/** * ThingSpeak Java Client * Copyright 2014, Andrew Bythell <abythell@ieee.org> * http://angryelectron.com * * The ThingSpeak Java Client is free software: you can redistribute it and/or * modify it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or (at your * option) any later version. * * The ThingSpeak Java Client is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General * Public License for more details. * * You should have received a copy of the GNU General Public License along with * theThingSpeak Java Client. If not, see <http://www.gnu.org/licenses/>. */ package com.angryelectron.thingspeak; import com.google.gson.Gson; import com.google.gson.GsonBuilder; import com.mashape.unirest.http.HttpResponse; import com.mashape.unirest.http.JsonNode; import com.mashape.unirest.http.Unirest; import com.mashape.unirest.http.exceptions.UnirestException; import com.mashape.unirest.request.GetRequest; import java.util.HashMap; /** * Thingspeak Channel. Methods for updating and requesting feeds and entries * from Thingspeak channels. */ public class Channel { //TODO: the API url should be configurable so the client can be used with //self-hosted servers. private String APIURL = "http://api.thingspeak.com"; private static final String APIHEADER = "X-THINGSPEAKAPIKEY"; private final Integer channelId; private String readAPIKey; private String writeAPIKey; private final Boolean isPublic; private final HashMap<String, Object> fields = new HashMap<>(); private final Gson gson = new GsonBuilder().setDateFormat("yyyy-MM-dd'T'HH:mm:ssXXX").create(); /** * Constructor for a public, read-only, Thingspeak channel. This type of * channel cannot be updated. * * @param channelId Channel Id. */ public Channel(Integer channelId) { this.isPublic = true; this.channelId = channelId; } /** * Constructor for a public, writeable, Thingspeak channel. * * @param channelId Channel Id. * @param writeKey API Key for the channel. See * https://thingspeak.com/channels/<channelId>#apikeys */ public Channel(Integer channelId, String writeKey) { this.isPublic = true; this.channelId = channelId; this.writeAPIKey = writeKey; } /** * Constructor for a private, writeable, Thingspeak channel. * * @param channelId Channel Id. * @param writeKey Write API Key. See * https://thingspeak.com/channels/<channelId>#apikeys. * @param readKey Read API Key. See * https://thingspeak.com/channels/<channelId>#apikeys. */ public Channel(Integer channelId, String writeKey, String readKey) { this.channelId = channelId; this.readAPIKey = readKey; this.writeAPIKey = writeKey; this.isPublic = false; } /** * Make GET requests to the Thingspeak API without additional feed * parameters. * * @param url The API url. * @return JSON string. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ private String thingRequest(String url) throws UnirestException, ThingSpeakException { GetRequest request = Unirest.get(url); if (!this.isPublic) { request.field("key", this.readAPIKey); } HttpResponse<JsonNode> response = request.asJson(); if (response.getCode() != 200) { throw new ThingSpeakException("Request failed with code " + response.getCode()); } return response.getBody().toString(); } /** * Make GET requests to the Thingspeak API with additional feed parameters. * * @param url The API url. * @param options Optional feed parameters. * @return JSON string. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ private String thingRequest(String url, FeedParameters options) throws UnirestException, ThingSpeakException { GetRequest request = Unirest.get(url); if (!this.isPublic) { request.field("key", this.readAPIKey); } request.fields(options.fields); HttpResponse<JsonNode> response = request.asJson(); if (response.getCode() != 200) { throw new ThingSpeakException("Request failed with code " + response.getCode()); } return response.getBody().toString(); } /** * Use a server other than thingspeak.com. If you are hosting your own * Thingspeak server, set the url of the server here. The url of the public * Thingspeak server is http://api.thingspeak.com * * @param url eg. http://localhost, http://thingspeak.local:3000, etc. */ public void setUrl(String url) { this.APIURL = url; } /** * Update channel with new data. * * @param entry The new data to be posted. * @return The id of the new entry. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Integer update(Entry entry) throws UnirestException, ThingSpeakException { HttpResponse<String> response = Unirest.post(APIURL + "/update") .header(APIHEADER, this.writeAPIKey) .header("Connection", "close") .fields(entry.getUpdateMap()) .asString(); if (response.getCode() != 200) { throw new ThingSpeakException("Request failed with code " + response.getCode()); } else if (response.getBody().equals("0")) { throw new ThingSpeakException("Update failed."); } return Integer.parseInt(response.getBody()); } /** * Get a channel feed with default feed options. Does not include location or status info. Only fields that * have been named in the channel's settings (via the web) will be returned. * * @return Feed for this channel. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Feed getChannelFeed() throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/feed.json"; return gson.fromJson(thingRequest(url), Feed.class); } /** * Get a channel feed with additional feed options. Only fields that have been named in * the channel's settings (via the web) will be returned. * * @param options Additional feed parameters. * @return Feed for this channel. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Feed getChannelFeed(FeedParameters options) throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/feed.json"; return gson.fromJson(thingRequest(url, options), Feed.class); } /** * Get last entry in this channel with default feed options. This is a * faster alternative to getting a Channel Feed and then calling * {@link Feed#getChannelLastEntry()}. * * @return Entry. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Entry getLastChannelEntry() throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/feed/last.json"; return gson.fromJson(thingRequest(url), Entry.class); } /** * Get last entry in this channel with additional feed options. This is a * faster alternative to getting a Channel Feed and then calling * {@link Feed#getChannelLastEntry()} * * @param options Supported options: offset, status, and location. * @return Entry. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Entry getLastChannelEntry(FeedParameters options) throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/feed/last.json"; return gson.fromJson(thingRequest(url, options), Entry.class); } /** * Get a field feed with default feed options. * * @param fieldId The field to include in the field (1-8). * @return Feed. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Feed getFieldFeed(Integer fieldId) throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/field/" + fieldId + ".json"; return gson.fromJson(thingRequest(url), Feed.class); } /** * Get a field feed with additional feed options. * * @param fieldId The field to include in the field (1-8). * @param options Optional parameters that control the format of the feed. * @return Feed. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Feed getFieldFeed(Integer fieldId, FeedParameters options) throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/field/" + fieldId + ".json"; return gson.fromJson(thingRequest(url, options), Feed.class); } /** * Get the last entry in a field feed with default feed options. * * @param fieldId The field to return (0-8). * @return Last entry for the specified field. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Entry getLastFieldEntry(Integer fieldId) throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/field/" + fieldId + "/last.json"; return gson.fromJson(thingRequest(url), Entry.class); } /** * Get the last entry in a field feed with additional feed options. * * @param fieldId The field to return (0-8). * @param options Supported options: offset, status, and location. * @return Last entry for the specified field. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Entry getLastFieldEntry(Integer fieldId, FeedParameters options) throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/field/" + fieldId + "/last.json"; return gson.fromJson(thingRequest(url, options), Entry.class); } /** * Get channel status updates. Uses the default feed options. * * @return Status feed. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Feed getStatusFeed() throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/status.json"; return gson.fromJson(thingRequest(url), Feed.class); } /** * Get channel status updates. * * @param options Only {@link FeedParameters#offset(java.lang.Integer)} is * supported. * @return Status feed. * @throws UnirestException The request cannot be made. * @throws ThingSpeakException The request is invalid. */ public Feed getStatusFeed(FeedParameters options) throws UnirestException, ThingSpeakException { String url = APIURL + "/channels/" + this.channelId + "/status.json"; return gson.fromJson(thingRequest(url, options), Feed.class); } /** * Not implemented. */ public void getUserInfo() { throw new UnsupportedOperationException("Not implemented."); } /** * Not implemented. */ public void getUserChannels() { throw new UnsupportedOperationException("Not implemented."); } /** * Checks if a channel is available/reachable. Use this method if you want to avoid handling exceptions. * * @return channel availability */ public boolean isAvailable() { String url = APIURL + "/channels/" + this.channelId + "/feed.json" + "?key=" + this.readAPIKey + "&results=0"; try { thingRequest(url); } catch (UnirestException | ThingSpeakException e) { return false; } return true; } }