From aec733797e8ddbfcce0e9b34be7ef2f4f569a950 Mon Sep 17 00:00:00 2001 From: Tim Kourt Date: Fri, 11 Sep 2015 15:47:12 -0700 Subject: [PATCH] Android: Added Device/Platform discovery client/server examples. In addition it also modifies the respective functions in the base Note: this commit depends on gerrit.iotivity.org/gerrit/#/c/2433/ Set 3: Changed the tests to be in accordance with the modified API Change-Id: I1a91166a54ef7f277d95c4d62961888c782383c5 Signed-off-by: Tim Kourt Reviewed-on: https://gerrit.iotivity.org/gerrit/2489 Tested-by: jenkins-iotivity Reviewed-by: Patrick Lankswert --- .../java/org/iotivity/base/SmokeTest.java | 17 +- .../src/main/java/org/iotivity/base/ErrorCode.java | 10 +- .../main/java/org/iotivity/base/OcDeviceInfo.java | 17 +- .../main/java/org/iotivity/base/OcPlatform.java | 10 +- .../java/org/iotivity/base/OcPlatformInfo.java | 131 ++++++------ .../java/org/iotivity/base/OcPlatformStatus.java | 1 - .../main/java/org/iotivity/base/OcStackConfig.java | 33 --- android/examples/devicediscoveryclient/.gitignore | 1 + .../examples/devicediscoveryclient/build.gradle | 24 +++ .../devicediscoveryclient.iml | 91 ++++++++ .../devicediscoveryclient/proguard-rules.pro | 17 ++ .../src/main/AndroidManifest.xml | 33 +++ .../base/examples/DeviceDiscoveryClient.java | 233 +++++++++++++++++++++ .../src/main/res/drawable/iotivityicon.png | Bin 0 -> 2300 bytes .../src/main/res/drawable/iotivitylogo.png | Bin 0 -> 123110 bytes .../layout/activity_device_discovery_client.xml | 37 ++++ .../src/main/res/mipmap-hdpi/iotivityicon.png | Bin 0 -> 2300 bytes .../src/main/res/mipmap-mdpi/iotivityicon.png | Bin 0 -> 1300 bytes .../src/main/res/mipmap-xhdpi/iotivityicon.png | Bin 0 -> 2862 bytes .../src/main/res/mipmap-xxhdpi/iotivityicon.png | Bin 0 -> 2651 bytes .../src/main/res/values-v21/styles.xml | 5 + .../src/main/res/values-w820dp/dimens.xml | 6 + .../src/main/res/values/dimens.xml | 5 + .../src/main/res/values/strings.xml | 3 + .../src/main/res/values/styles.xml | 8 + android/examples/devicediscoveryserver/.gitignore | 1 + .../examples/devicediscoveryserver/build.gradle | 24 +++ .../devicediscoveryserver.iml | 91 ++++++++ .../devicediscoveryserver/proguard-rules.pro | 17 ++ .../src/main/AndroidManifest.xml | 33 +++ .../base/examples/DeviceDiscoveryServer.java | 163 ++++++++++++++ .../src/main/res/drawable/iotivityicon.png | Bin 0 -> 2300 bytes .../src/main/res/drawable/iotivitylogo.png | Bin 0 -> 123110 bytes .../layout/activity_device_discovery_server.xml | 37 ++++ .../src/main/res/mipmap-hdpi/iotivityicon.png | Bin 0 -> 2300 bytes .../src/main/res/mipmap-mdpi/iotivityicon.png | Bin 0 -> 1300 bytes .../src/main/res/mipmap-xhdpi/iotivityicon.png | Bin 0 -> 2862 bytes .../src/main/res/mipmap-xxhdpi/iotivityicon.png | Bin 0 -> 2651 bytes .../src/main/res/values-v21/styles.xml | 5 + .../src/main/res/values-w820dp/dimens.xml | 6 + .../src/main/res/values/dimens.xml | 5 + .../src/main/res/values/strings.xml | 3 + .../src/main/res/values/styles.xml | 8 + android/examples/presenceclient/build.gradle | 2 +- .../presenceclient/src/main/AndroidManifest.xml | 24 +-- .../org/iotivity/base/examples/PresenceClient.java | 36 +--- .../main/res/layout/activity_presence_client.xml | 38 ++-- .../src/main/res/values-v21/styles.xml | 4 +- .../org/iotivity/base/examples/PresenceServer.java | 13 +- android/examples/settings.gradle | 2 +- 50 files changed, 990 insertions(+), 204 deletions(-) delete mode 100644 android/android_api/base/src/main/java/org/iotivity/base/OcStackConfig.java create mode 100644 android/examples/devicediscoveryclient/.gitignore create mode 100644 android/examples/devicediscoveryclient/build.gradle create mode 100644 android/examples/devicediscoveryclient/devicediscoveryclient.iml create mode 100644 android/examples/devicediscoveryclient/proguard-rules.pro create mode 100644 android/examples/devicediscoveryclient/src/main/AndroidManifest.xml create mode 100644 android/examples/devicediscoveryclient/src/main/java/org/iotivity/base/examples/DeviceDiscoveryClient.java create mode 100644 android/examples/devicediscoveryclient/src/main/res/drawable/iotivityicon.png create mode 100644 android/examples/devicediscoveryclient/src/main/res/drawable/iotivitylogo.png create mode 100644 android/examples/devicediscoveryclient/src/main/res/layout/activity_device_discovery_client.xml create mode 100644 android/examples/devicediscoveryclient/src/main/res/mipmap-hdpi/iotivityicon.png create mode 100644 android/examples/devicediscoveryclient/src/main/res/mipmap-mdpi/iotivityicon.png create mode 100644 android/examples/devicediscoveryclient/src/main/res/mipmap-xhdpi/iotivityicon.png create mode 100644 android/examples/devicediscoveryclient/src/main/res/mipmap-xxhdpi/iotivityicon.png create mode 100644 android/examples/devicediscoveryclient/src/main/res/values-v21/styles.xml create mode 100644 android/examples/devicediscoveryclient/src/main/res/values-w820dp/dimens.xml create mode 100644 android/examples/devicediscoveryclient/src/main/res/values/dimens.xml create mode 100644 android/examples/devicediscoveryclient/src/main/res/values/strings.xml create mode 100644 android/examples/devicediscoveryclient/src/main/res/values/styles.xml create mode 100644 android/examples/devicediscoveryserver/.gitignore create mode 100644 android/examples/devicediscoveryserver/build.gradle create mode 100644 android/examples/devicediscoveryserver/devicediscoveryserver.iml create mode 100644 android/examples/devicediscoveryserver/proguard-rules.pro create mode 100644 android/examples/devicediscoveryserver/src/main/AndroidManifest.xml create mode 100644 android/examples/devicediscoveryserver/src/main/java/org/iotivity/base/examples/DeviceDiscoveryServer.java create mode 100644 android/examples/devicediscoveryserver/src/main/res/drawable/iotivityicon.png create mode 100644 android/examples/devicediscoveryserver/src/main/res/drawable/iotivitylogo.png create mode 100644 android/examples/devicediscoveryserver/src/main/res/layout/activity_device_discovery_server.xml create mode 100644 android/examples/devicediscoveryserver/src/main/res/mipmap-hdpi/iotivityicon.png create mode 100644 android/examples/devicediscoveryserver/src/main/res/mipmap-mdpi/iotivityicon.png create mode 100644 android/examples/devicediscoveryserver/src/main/res/mipmap-xhdpi/iotivityicon.png create mode 100644 android/examples/devicediscoveryserver/src/main/res/mipmap-xxhdpi/iotivityicon.png create mode 100644 android/examples/devicediscoveryserver/src/main/res/values-v21/styles.xml create mode 100644 android/examples/devicediscoveryserver/src/main/res/values-w820dp/dimens.xml create mode 100644 android/examples/devicediscoveryserver/src/main/res/values/dimens.xml create mode 100644 android/examples/devicediscoveryserver/src/main/res/values/strings.xml create mode 100644 android/examples/devicediscoveryserver/src/main/res/values/styles.xml diff --git a/android/android_api/base/src/androidTest/java/org/iotivity/base/SmokeTest.java b/android/android_api/base/src/androidTest/java/org/iotivity/base/SmokeTest.java index b0fadd7..350091e 100644 --- a/android/android_api/base/src/androidTest/java/org/iotivity/base/SmokeTest.java +++ b/android/android_api/base/src/androidTest/java/org/iotivity/base/SmokeTest.java @@ -995,12 +995,8 @@ public class SmokeTest extends InstrumentationTestCase { }; OcPlatformInfo platformInfo = null; - try { - platformInfo = new OcPlatformInfo("myPlatformID", "myManuName", "myManuUrl"); - } catch (OcException e) { - Log.e(TAG, "Could not construct platformInfo. " + e.getMessage()); - assertTrue(false); - } + + platformInfo = new OcPlatformInfo("myPlatformID", "myManuName", "myManuUrl"); platformInfo.setModelNumber("myModelNumber"); platformInfo.setDateOfManufacture("myDateOfManufacture"); @@ -1013,13 +1009,12 @@ public class SmokeTest extends InstrumentationTestCase { try { //server - OcPlatform.registerPlatformInfo(platformInfo); //client OcPlatform.getPlatformInfo( "", - OcPlatform.MULTICAST_PREFIX + "/oic/p", + OcPlatform.WELL_KNOWN_PLATFORM_QUERY, EnumSet.of(OcConnectivityType.CT_DEFAULT), platformFoundListener); @@ -1053,17 +1048,15 @@ public class SmokeTest extends InstrumentationTestCase { } }; - OcDeviceInfo devInfo = new OcDeviceInfo(); - devInfo.setDeviceName("myDeviceName"); + OcDeviceInfo devInfo = new OcDeviceInfo("myDeviceName"); try { //server OcPlatform.registerDeviceInfo(devInfo); - //client OcPlatform.getDeviceInfo( "", - OcPlatform.MULTICAST_PREFIX + OcPlatform.DEVICE_URI, + OcPlatform.WELL_KNOWN_DEVICE_QUERY, EnumSet.of(OcConnectivityType.CT_DEFAULT), deviceFoundListener); diff --git a/android/android_api/base/src/main/java/org/iotivity/base/ErrorCode.java b/android/android_api/base/src/main/java/org/iotivity/base/ErrorCode.java index b83e4b3..5951750 100644 --- a/android/android_api/base/src/main/java/org/iotivity/base/ErrorCode.java +++ b/android/android_api/base/src/main/java/org/iotivity/base/ErrorCode.java @@ -55,20 +55,12 @@ public enum ErrorCode { PERSISTENT_BUFFER_REQUIRED("PERSISTENT_BUFFER_REQUIRED", ""), INVALID_REQUEST_HANDLE("INVALID_REQUEST_HANDLE", ""), INVALID_DEVICE_INFO("INVALID_DEVICE_INFO", ""), - INVALID_PLATFORM_INFO_PLATFORMID("INVALID_PLATFORM_INFO_PLATFORMID", - "PlatformID cannot be null or empty"), - INVALID_PLATFORM_INFO_MANUFACTURER_NAME("INVALID_PLATFORM_INFO_MANUFACTURER_NAME", - "ManufacturerName cannot be null, empty or greater than " + - OcStackConfig.MAX_MANUFACTURER_NAME_LENGTH + " characters long"), - INVALID_PLATFORM_INFO_PLATFORMID_MANUFACTURER_URL("INVALID_PLATFORM_INFO_MANUFACTURER_URL", - "MANUFACTURER_URL cannot be null, empty or greater than " + - OcStackConfig.MAX_MANUFACTURER_URL_LENGTH + " characters long"), + ERROR("ERROR", "Generic error"), JNI_EXCEPTION("JNI_EXCEPTION", "Generic Java binder error"), JNI_NO_NATIVE_OBJECT("JNI_NO_NATIVE_OBJECT", ""), JNI_INVALID_VALUE("JNI_INVALID_VALUE", ""), - INVALID_CLASS_CAST("INVALID_CLASS_CAST", ""),; private String error; diff --git a/android/android_api/base/src/main/java/org/iotivity/base/OcDeviceInfo.java b/android/android_api/base/src/main/java/org/iotivity/base/OcDeviceInfo.java index 2cb794b..a793e7a 100644 --- a/android/android_api/base/src/main/java/org/iotivity/base/OcDeviceInfo.java +++ b/android/android_api/base/src/main/java/org/iotivity/base/OcDeviceInfo.java @@ -24,25 +24,14 @@ package org.iotivity.base; public class OcDeviceInfo { - private String deviceName; - - public OcDeviceInfo() { - - deviceName = ""; - } + private String mDeviceName; public OcDeviceInfo(String deviceName) { - this.deviceName = deviceName; + this.mDeviceName = deviceName; } public String getDeviceName() { - - return deviceName; - } - - public void setDeviceName(String deviceName) { - - this.deviceName = deviceName; + return mDeviceName; } } diff --git a/android/android_api/base/src/main/java/org/iotivity/base/OcPlatform.java b/android/android_api/base/src/main/java/org/iotivity/base/OcPlatform.java index 63b482a..8004287 100644 --- a/android/android_api/base/src/main/java/org/iotivity/base/OcPlatform.java +++ b/android/android_api/base/src/main/java/org/iotivity/base/OcPlatform.java @@ -64,11 +64,9 @@ public final class OcPlatform { public static final String GROUP_INTERFACE = "oic.mi.grp"; public static final String WELL_KNOWN_QUERY = "/oic/res"; - public static final String MULTICAST_PREFIX = "224.0.1.187:5683"; - public static final String MULTICAST_IP = "224.0.1.187"; - public static final int MULTICAST_PORT = 5683; + public static final String WELL_KNOWN_DEVICE_QUERY = "/oic/d"; + public static final String WELL_KNOWN_PLATFORM_QUERY = "/oic/p"; public static final int DEFAULT_PRESENCE_TTL = 60; - public static final String DEVICE_URI = "/oic/d"; public static final String PRESENCE_URI = "/oic/ad"; private static volatile boolean sIsPlatformInitialized = false; @@ -491,7 +489,7 @@ public final class OcPlatform { resProperty |= prop.getValue(); } - if(null == entityHandler){ + if (null == entityHandler) { entityHandler = new EntityHandler() { @Override public EntityHandlerResult handleEntity(OcResourceRequest ocResourceRequest) { @@ -542,7 +540,7 @@ public final class OcPlatform { OcPlatformInfo ocPlatformInfo) throws OcException { OcPlatform.initCheck(); OcPlatform.registerPlatformInfo0( - ocPlatformInfo.getPlatformID(), + ocPlatformInfo.getPlatformId(), ocPlatformInfo.getManufacturerName(), ocPlatformInfo.getManufacturerUrl(), ocPlatformInfo.getModelNumber(), diff --git a/android/android_api/base/src/main/java/org/iotivity/base/OcPlatformInfo.java b/android/android_api/base/src/main/java/org/iotivity/base/OcPlatformInfo.java index 53d366d..1ed7a94 100644 --- a/android/android_api/base/src/main/java/org/iotivity/base/OcPlatformInfo.java +++ b/android/android_api/base/src/main/java/org/iotivity/base/OcPlatformInfo.java @@ -22,135 +22,138 @@ package org.iotivity.base; -import java.lang.Error; +import java.security.InvalidParameterException; public class OcPlatformInfo { - - private String platformID; - private String manufacturerName; - private String manufacturerUrl; - private String modelNumber; - private String dateOfManufacture; - private String platformVersion; - private String operatingSystemVersion; - private String hardwareVersion; - private String firmwareVersion; - private String supportUrl; - private String systemTime; + private String mPlatformId; + private String mManufacturerName; + private String mManufacturerUrl; + private String mModelNumber; + private String mDateOfManufacture; + private String mPlatformVersion; + private String mOperatingSystemVersion; + private String mHardwareVersion; + private String mFirmwareVersion; + private String mSupportUrl; + private String mSystemTime; // construct OcPlatformInfo with mandatory fields which cannot be null // manufacturerName cannot be > 16 chars // manufacturerUrl cannot be > 32 chars - protected OcPlatformInfo(String platformID, String manufacturerName, - String manufacturerUrl) throws OcException { - ErrorCode result = validatePlatformInfo(platformID, manufacturerName, manufacturerUrl); - if (ErrorCode.OK == result) { - this.platformID = platformID; - this.manufacturerName = manufacturerName; - this.manufacturerUrl = manufacturerUrl; - } else { - throw new OcException(result, result.getDescription()); - } - } - - public ErrorCode validatePlatformInfo(String platformID, String manufacturerName, - String manufacturerUrl) { - // checks to see if the mandatory fields have non-null values or not - if (platformID == null || platformID.isEmpty()) return ErrorCode.INVALID_PLATFORM_INFO_PLATFORMID; - if (manufacturerName == null || manufacturerName.isEmpty() || - manufacturerName.length() > OcStackConfig.MAX_MANUFACTURER_NAME_LENGTH) - return ErrorCode.INVALID_PLATFORM_INFO_MANUFACTURER_NAME; - if (manufacturerUrl == null || manufacturerUrl.isEmpty() || - manufacturerUrl.length() > OcStackConfig.MAX_MANUFACTURER_URL_LENGTH) - return ErrorCode.INVALID_PLATFORM_INFO_PLATFORMID_MANUFACTURER_URL; - return ErrorCode.OK; - } - - public String getPlatformID() { - return platformID; - } - - public void setPlatformID(String platformID) { - this.platformID = platformID; + public OcPlatformInfo(String platformId, String manufacturerName, + String manufacturerUrl) { + this.mPlatformId = platformId; + this.mManufacturerName = manufacturerName; + this.mManufacturerUrl = manufacturerUrl; + } + + public OcPlatformInfo(String platformId, + String manufacturerName, + String manufacturerUrl, + String modelNumber, + String dateOfManufacture, + String platformVersion, + String operatingSystemVersion, + String hardwareVersion, + String firmwareVersion, + String supportUrl, + String systemTime) { + this(platformId, manufacturerName, manufacturerUrl); + this.mModelNumber = modelNumber; + this.mDateOfManufacture = dateOfManufacture; + this.mPlatformVersion = platformVersion; + this.mOperatingSystemVersion = operatingSystemVersion; + this.mHardwareVersion = hardwareVersion; + this.mFirmwareVersion = firmwareVersion; + this.mSupportUrl = supportUrl; + this.mSystemTime = systemTime; + } + + public String getPlatformId() { + return mPlatformId; + } + + public void setPlatformId(String platformId) { + this.mPlatformId = platformId; } public String getManufacturerName() { - return manufacturerName; + return mManufacturerName; } public void setManufacturerName(String manufacturerName) { - this.manufacturerName = manufacturerName; + this.mManufacturerName = manufacturerName; } public String getManufacturerUrl() { - return manufacturerUrl; + return mManufacturerUrl; } public void setManufacturerUrl(String manufacturerUrl) { - this.manufacturerUrl = manufacturerUrl; + this.mManufacturerUrl = manufacturerUrl; } public String getModelNumber() { - return modelNumber; + return mModelNumber; } public void setModelNumber(String modelNumber) { - this.modelNumber = modelNumber; + this.mModelNumber = modelNumber; } public String getDateOfManufacture() { - return dateOfManufacture; + return mDateOfManufacture; } public void setDateOfManufacture(String dateOfManufacture) { - this.dateOfManufacture = dateOfManufacture; + this.mDateOfManufacture = dateOfManufacture; } public String getPlatformVersion() { - return platformVersion; + return mPlatformVersion; } public void setPlatformVersion(String platformVersion) { - this.platformVersion = platformVersion; + this.mPlatformVersion = platformVersion; } public String getOperatingSystemVersion() { - return operatingSystemVersion; + return mOperatingSystemVersion; } public void setOperatingSystemVersion(String operatingSystemVersion) { - this.operatingSystemVersion = operatingSystemVersion; + this.mOperatingSystemVersion = operatingSystemVersion; } public String getHardwareVersion() { - return hardwareVersion; + return mHardwareVersion; } public void setHardwareVersion(String hardwareVersion) { - this.hardwareVersion = hardwareVersion; + this.mHardwareVersion = hardwareVersion; } public String getFirmwareVersion() { - return firmwareVersion; + return mFirmwareVersion; } public void setFirmwareVersion(String firmwareVersion) { - this.firmwareVersion = firmwareVersion; + this.mFirmwareVersion = firmwareVersion; } public String getSupportUrl() { - return supportUrl; + return mSupportUrl; } public void setSupportUrl(String supportUrl) { - this.supportUrl = supportUrl; + this.mSupportUrl = supportUrl; } public String getSystemTime() { - return systemTime; + return mSystemTime; } public void setSystemTime(String systemTime) { - this.systemTime = systemTime; + this.mSystemTime = systemTime; } } diff --git a/android/android_api/base/src/main/java/org/iotivity/base/OcPlatformStatus.java b/android/android_api/base/src/main/java/org/iotivity/base/OcPlatformStatus.java index 9addbf7..3087e5a 100644 --- a/android/android_api/base/src/main/java/org/iotivity/base/OcPlatformStatus.java +++ b/android/android_api/base/src/main/java/org/iotivity/base/OcPlatformStatus.java @@ -22,7 +22,6 @@ package org.iotivity.base; - public enum OcPlatformStatus { UP(0), DOWN(1),; diff --git a/android/android_api/base/src/main/java/org/iotivity/base/OcStackConfig.java b/android/android_api/base/src/main/java/org/iotivity/base/OcStackConfig.java deleted file mode 100644 index ff48d22..0000000 --- a/android/android_api/base/src/main/java/org/iotivity/base/OcStackConfig.java +++ /dev/null @@ -1,33 +0,0 @@ -/* - * //****************************************************************** - * // - * // Copyright 2015 Intel Corporation. - * // - * //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= - * // - * // Licensed under the Apache License, Version 2.0 (the "License"); - * // you may not use this file except in compliance with the License. - * // You may obtain a copy of the License at - * // - * // http://www.apache.org/licenses/LICENSE-2.0 - * // - * // Unless required by applicable law or agreed to in writing, software - * // distributed under the License is distributed on an "AS IS" BASIS, - * // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * // See the License for the specific language governing permissions and - * // limitations under the License. - * // - * //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= - */ - -// This file contains all the variables which can be configured/modified as -// per platform or specific product usage scenarios. - -package org.iotivity.base; - -public interface OcStackConfig { - // max manufacturer name length for OcPlatformInfo supported by server - public static final int MAX_MANUFACTURER_NAME_LENGTH = 16; - // max manufacturer url length for OcPlatformInfo supported by server - public static final int MAX_MANUFACTURER_URL_LENGTH = 32; -} \ No newline at end of file diff --git a/android/examples/devicediscoveryclient/.gitignore b/android/examples/devicediscoveryclient/.gitignore new file mode 100644 index 0000000..796b96d --- /dev/null +++ b/android/examples/devicediscoveryclient/.gitignore @@ -0,0 +1 @@ +/build diff --git a/android/examples/devicediscoveryclient/build.gradle b/android/examples/devicediscoveryclient/build.gradle new file mode 100644 index 0000000..0f56669 --- /dev/null +++ b/android/examples/devicediscoveryclient/build.gradle @@ -0,0 +1,24 @@ +apply plugin: 'com.android.application' + +android { + compileSdkVersion 22 + buildToolsVersion "22.0.1" + + defaultConfig { + applicationId "org.iotivity.base.examples.devicediscoveryclient" + minSdkVersion 21 + targetSdkVersion 22 + versionCode 1 + versionName "1.0" + } + buildTypes { + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' + } + } +} + +dependencies { + compile fileTree(dir: 'libs', include: ['*.jar']) +} diff --git a/android/examples/devicediscoveryclient/devicediscoveryclient.iml b/android/examples/devicediscoveryclient/devicediscoveryclient.iml new file mode 100644 index 0000000..ee457b3 --- /dev/null +++ b/android/examples/devicediscoveryclient/devicediscoveryclient.iml @@ -0,0 +1,91 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/android/examples/devicediscoveryclient/proguard-rules.pro b/android/examples/devicediscoveryclient/proguard-rules.pro new file mode 100644 index 0000000..d26150c --- /dev/null +++ b/android/examples/devicediscoveryclient/proguard-rules.pro @@ -0,0 +1,17 @@ +# Add project specific ProGuard rules here. +# By default, the flags in this file are appended to flags specified +# in /home/rahul/sdk/tools/proguard/proguard-android.txt +# You can edit the include path and order by changing the proguardFiles +# directive in build.gradle. +# +# For more details, see +# http://developer.android.com/guide/developing/tools/proguard.html + +# Add any project specific keep options here: + +# If your project uses WebView with JS, uncomment the following +# and specify the fully qualified class name to the JavaScript interface +# class: +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { +# public *; +#} diff --git a/android/examples/devicediscoveryclient/src/main/AndroidManifest.xml b/android/examples/devicediscoveryclient/src/main/AndroidManifest.xml new file mode 100644 index 0000000..ca6ea94 --- /dev/null +++ b/android/examples/devicediscoveryclient/src/main/AndroidManifest.xml @@ -0,0 +1,33 @@ + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/android/examples/devicediscoveryclient/src/main/java/org/iotivity/base/examples/DeviceDiscoveryClient.java b/android/examples/devicediscoveryclient/src/main/java/org/iotivity/base/examples/DeviceDiscoveryClient.java new file mode 100644 index 0000000..309fc3b --- /dev/null +++ b/android/examples/devicediscoveryclient/src/main/java/org/iotivity/base/examples/DeviceDiscoveryClient.java @@ -0,0 +1,233 @@ +/* + * //****************************************************************** + * // + * // Copyright 2015 Intel Corporation. + * // + * //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= + * // + * // Licensed under the Apache License, Version 2.0 (the "License"); + * // you may not use this file except in compliance with the License. + * // You may obtain a copy of the License at + * // + * // http://www.apache.org/licenses/LICENSE-2.0 + * // + * // Unless required by applicable law or agreed to in writing, software + * // distributed under the License is distributed on an "AS IS" BASIS, + * // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * // See the License for the specific language governing permissions and + * // limitations under the License. + * // + * //-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= + */ +package org.iotivity.base.examples; + +import android.app.Activity; +import android.content.Context; +import android.os.Bundle; +import android.text.method.ScrollingMovementMethod; +import android.util.Log; +import android.view.View; +import android.widget.Button; +import android.widget.ScrollView; +import android.widget.TextView; + +import org.iotivity.base.ModeType; +import org.iotivity.base.OcConnectivityType; +import org.iotivity.base.OcException; +import org.iotivity.base.OcPlatform; +import org.iotivity.base.OcRepresentation; +import org.iotivity.base.PlatformConfig; +import org.iotivity.base.QualityOfService; +import org.iotivity.base.ServiceType; + +import java.util.EnumSet; +import java.util.HashMap; +import java.util.Map; + +/** + * This sample demonstrates the device discovery feature. + * The client queries for the device related information stored by the server. + */ +public class DeviceDiscoveryClient extends Activity implements + OcPlatform.OnDeviceFoundListener, + OcPlatform.OnPlatformFoundListener { + private void startDeviceDiscoveryClient() { + Context context = this; + + PlatformConfig platformConfig = new PlatformConfig( + context, + ServiceType.IN_PROC, + ModeType.CLIENT, + "0.0.0.0", // By setting to "0.0.0.0", it binds to all available interfaces + 0, // Uses randomly available port + QualityOfService.LOW + ); + + msg("Configuring platform."); + OcPlatform.Configure(platformConfig); + sleep(1); + + try { + msg("Querying for platform information..."); + OcPlatform.getPlatformInfo("", + OcPlatform.WELL_KNOWN_PLATFORM_QUERY, + EnumSet.of(OcConnectivityType.CT_DEFAULT), + this); + } catch (OcException e) { + Log.e(TAG, e.toString()); + msg("Failed to query for platform information"); + } + sleep(2); + + try { + msg("Querying for device information..."); + OcPlatform.getDeviceInfo("", + OcPlatform.WELL_KNOWN_DEVICE_QUERY, + EnumSet.of(OcConnectivityType.CT_DEFAULT), + this); + } catch (OcException e) { + Log.e(TAG, e.toString()); + msg("Failed to query for device information"); + } + sleep(2); + + enableStartButton(); + printLine(); + } + + private final static Map PLATFORM_INFO_KEYS = new HashMap() {{ + put("pi", "Platform ID: "); + put("mnmn", "Manufacturer name: "); + put("mnml", "Manufacturer url: "); + put("mnmo", "Manufacturer Model No: "); + put("mndt", "Manufactured Date: "); + put("mnpv", "Manufacturer Platform Version: "); + put("mnos", "Manufacturer OS version: "); + put("mnhw", "Manufacturer hardware version: "); + put("mnfv", "Manufacturer firmware version: "); + put("mnsl", "Manufacturer support url: "); + put("st", "Manufacturer system time: "); + }}; + + @Override + public synchronized void onPlatformFound(OcRepresentation ocRepresentation) { + msg("Platform Information received:"); + try { + for (String key : PLATFORM_INFO_KEYS.keySet()) { + msg("\t" + PLATFORM_INFO_KEYS.get(key) + ocRepresentation.getValue(key)); + } + } catch (OcException e) { + Log.e(TAG, e.toString()); + msg("Failed to read platform info values."); + } + + printLine(); + } + + private final static Map DEVICE_INFO_KEYS = new HashMap() {{ + put("di", "Device ID: "); + put("n", "Device name: "); + put("lcv", "Spec version url: "); + put("dmv", "Data Model: "); + }}; + + @Override + public synchronized void onDeviceFound(OcRepresentation ocRepresentation) { + msg("Device Information received:"); + try { + for (String key : DEVICE_INFO_KEYS.keySet()) { + msg("\t" + DEVICE_INFO_KEYS.get(key) + ocRepresentation.getValue(key)); + } + } catch (OcException e) { + Log.e(TAG, e.toString()); + msg("Failed to read device info values."); + } + + printLine(); + } + + //****************************************************************************** + // End of the OIC specific code + //****************************************************************************** + + private final static String TAG = DeviceDiscoveryClient.class.getSimpleName(); + private TextView mConsoleTextView; + private ScrollView mScrollView; + + @Override + protected void onCreate(Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + setContentView(R.layout.activity_device_discovery_client); + + mConsoleTextView = (TextView) findViewById(R.id.consoleTextView); + mConsoleTextView.setMovementMethod(new ScrollingMovementMethod()); + mScrollView = (ScrollView) findViewById(R.id.scrollView); + mScrollView.fullScroll(View.FOCUS_DOWN); + final Button button = (Button) findViewById(R.id.button); + + if (null == savedInstanceState) { + button.setOnClickListener(new View.OnClickListener() { + @Override + public void onClick(View v) { + button.setText("Re-start"); + button.setEnabled(false); + new Thread(new Runnable() { + public void run() { + startDeviceDiscoveryClient(); + } + }).start(); + } + }); + } else { + String consoleOutput = savedInstanceState.getString("consoleOutputString"); + mConsoleTextView.setText(consoleOutput); + } + } + + @Override + protected void onSaveInstanceState(Bundle outState) { + super.onSaveInstanceState(outState); + outState.putString("consoleOutputString", mConsoleTextView.getText().toString()); + } + + @Override + protected void onRestoreInstanceState(Bundle savedInstanceState) { + super.onRestoreInstanceState(savedInstanceState); + + String consoleOutput = savedInstanceState.getString("consoleOutputString"); + mConsoleTextView.setText(consoleOutput); + } + + private void enableStartButton() { + runOnUiThread(new Runnable() { + public void run() { + Button button = (Button) findViewById(R.id.button); + button.setEnabled(true); + } + }); + } + + private void sleep(int seconds) { + try { + Thread.sleep(seconds * 1000); + } catch (InterruptedException e) { + e.printStackTrace(); + Log.e(TAG, e.toString()); + } + } + + private void msg(final String text) { + runOnUiThread(new Runnable() { + public void run() { + mConsoleTextView.append("\n"); + mConsoleTextView.append(text); + mScrollView.fullScroll(View.FOCUS_DOWN); + } + }); + Log.i(TAG, text); + } + + private void printLine() { + msg("------------------------------------------------------------------------"); + } +} diff --git a/android/examples/devicediscoveryclient/src/main/res/drawable/iotivityicon.png b/android/examples/devicediscoveryclient/src/main/res/drawable/iotivityicon.png new file mode 100644 index 0000000000000000000000000000000000000000..e1e4aa71b5449b12e60072f3999fc75186d8946c GIT binary patch literal 2300 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 OpaH}C@9H;h$^QYPfGv3d literal 0 HcmV?d00001 diff --git a/android/examples/devicediscoveryclient/src/main/res/layout/activity_device_discovery_client.xml b/android/examples/devicediscoveryclient/src/main/res/layout/activity_device_discovery_client.xml new file mode 100644 index 0000000..621533b --- /dev/null +++ b/android/examples/devicediscoveryclient/src/main/res/layout/activity_device_discovery_client.xml @@ -0,0 +1,37 @@ + +