| | |
| | | |
| | | |
| | | import com.google.gson.Gson; |
| | | import com.google.gson.JsonArray; |
| | | import com.google.gson.JsonObject; |
| | | import com.google.gson.reflect.TypeToken; |
| | | import com.hdl.linkpm.sdk.core.exception.HDLException; |
| | |
| | | import com.hdl.linkpm.sdk.ota.bean.CloudDeviceFirmwaresBean; |
| | | import com.hdl.linkpm.sdk.ota.bean.CloudGatewayDriversBean; |
| | | import com.hdl.linkpm.sdk.ota.bean.DeviceFirmwareBean; |
| | | import com.hdl.linkpm.sdk.ota.bean.DownloadUrlBean; |
| | | import com.hdl.linkpm.sdk.ota.bean.GatewayDriverBean; |
| | | import com.hdl.photovoltaic.HDLApp; |
| | | import com.hdl.photovoltaic.config.UserConfigManage; |
| | |
| | | import com.hdl.photovoltaic.listener.CloudCallBeak; |
| | | import com.hdl.photovoltaic.listener.LinkCallBack; |
| | | import com.hdl.photovoltaic.utils.AesUtils; |
| | | import com.hdl.photovoltaic.utils.AppManagerUtils; |
| | | import com.hdl.photovoltaic.utils.Md5Utils; |
| | | import com.hdl.photovoltaic.utils.NetworkUtils; |
| | | import com.hdl.sdk.link.common.exception.HDLLinkException; |
| | | import com.hdl.sdk.link.common.utils.ByteUtils; |
| | | import com.hdl.sdk.link.core.bean.eventbus.BaseEventBus; |
| | | import com.hdl.sdk.link.core.callback.HDLLinkCallBack; |
| | | |
| | |
| | | |
| | | import java.io.File; |
| | | import java.io.FileOutputStream; |
| | | import java.io.IOException; |
| | | import java.io.InputStream; |
| | | import java.lang.reflect.Type; |
| | | import java.util.ArrayList; |
| | | import java.util.List; |
| | | import java.util.Objects; |
| | | |
| | | import okhttp3.ResponseBody; |
| | | |
| | |
| | | private Intent intentServer = null; |
| | | |
| | | private ServiceConnection mServiceConnection = null; |
| | | /** |
| | | * 服务是否开启 true=开启 |
| | | */ |
| | | public boolean mServiceStart = false; |
| | | |
| | | /** |
| | | * true表示终止下载升级文件(用户行为) |
| | | */ |
| | | private boolean stopDownload = false; |
| | | |
| | | |
| | | //region -----固件升级--------- |
| | | |
| | |
| | | |
| | | } |
| | | |
| | | /** |
| | | * 向云端获取【设备固件】升级包下载地址 |
| | | * |
| | | * @param firmwareVersionId 固件版本Id |
| | | */ |
| | | public void getDeviceFirmwareDownloadUrl(String firmwareVersionId, CloudCallBeak<DownloadUrlBean> callBack) { |
| | | // "protocolType":"bus", |
| | | // "deviceModel":"029B9988", |
| | | // "firmwareNameEn":"核心层", |
| | | // "firmwareNameCn":"核心层", |
| | | // "imageId":"12345678", |
| | | // "version":"V01.02.20" |
| | | String requestUrl = HttpApi.POST_OTA_GET_LinkDeviceFirmwareDownloadUrl; |
| | | JsonObject json = new JsonObject(); |
| | | json.addProperty("firmwareVersionId", firmwareVersionId); |
| | | HttpClient.getInstance().requestHttp(requestUrl, json.toString(), new CloudCallBeak<String>() { |
| | | @Override |
| | | public void onSuccess(String json) { |
| | | |
| | | if (TextUtils.isEmpty(json)) { |
| | | if (callBack != null) { |
| | | callBack.onSuccess(new DownloadUrlBean()); |
| | | } |
| | | return; |
| | | } |
| | | Gson gson = new Gson(); |
| | | Type typeOfT = new TypeToken<DownloadUrlBean>() { |
| | | }.getType(); |
| | | DownloadUrlBean downloadUrlBean = gson.fromJson(json, typeOfT); |
| | | if (callBack != null) { |
| | | callBack.onSuccess(downloadUrlBean); |
| | | } |
| | | } |
| | | |
| | | @Override |
| | | public void onFailure(HDLException e) { |
| | | if (callBack != null) { |
| | | callBack.onFailure(e); |
| | | } |
| | | } |
| | | }); |
| | | } |
| | | |
| | | /** |
| | | * 向云端获取【设备固件】下载升级文件 |
| | | * |
| | | * @param downloadUrl 云端固件地址 |
| | | */ |
| | | public void getDeviceFirmwareDownloadFile(String downloadUrl, CloudCallBeak<DownloadUrlBean> callBack) { |
| | | String requestUrl = HttpApi.POST_OTA_GET_LinkDeviceFirmwareDownloadUrl; |
| | | JsonObject json = new JsonObject(); |
| | | |
| | | } |
| | | |
| | | //endregion |
| | | |
| | |
| | | |
| | | } |
| | | |
| | | |
| | | /** |
| | | * 云端发起【设备驱动】升级OTA指令 |
| | | * 云端发起【设备驱动升级】OTA指令 |
| | | * |
| | | * @param oid 网关设备oid |
| | | * @param driverVersionId 驱动版本id |
| | |
| | | }); |
| | | } |
| | | |
| | | |
| | | //endregion |
| | | |
| | | |
| | | /** |
| | | * 向云端获取【设备驱动】升级包下载地址 |
| | | * |
| | | * @param driverVersionId 驱动版本Id |
| | | * 取消终止下载文件指令 |
| | | */ |
| | | public void getDeviceDriverDownloadUrl(String driverVersionId, CloudCallBeak<DownloadUrlBean> callBack) { |
| | | |
| | | String requestUrl = HttpApi.POST_OTA_GET_LinkDeviceDriverDownloadUrl; |
| | | JsonObject json = new JsonObject(); |
| | | json.addProperty("driverVersionId", driverVersionId); |
| | | HttpClient.getInstance().requestHttp(requestUrl, json.toString(), new CloudCallBeak<String>() { |
| | | @Override |
| | | public void onSuccess(String s) { |
| | | if (TextUtils.isEmpty(s)) { |
| | | if (callBack != null) { |
| | | callBack.onSuccess(new DownloadUrlBean()); |
| | | } |
| | | return; |
| | | } |
| | | Gson gson = new Gson(); |
| | | Type typeOfT = new TypeToken<DownloadUrlBean>() { |
| | | }.getType(); |
| | | DownloadUrlBean downloadUrlBean = gson.fromJson(s, typeOfT); |
| | | if (callBack != null) { |
| | | callBack.onSuccess(downloadUrlBean); |
| | | } |
| | | } |
| | | |
| | | @Override |
| | | public void onFailure(HDLException e) { |
| | | if (callBack != null) { |
| | | callBack.onFailure(e); |
| | | } |
| | | } |
| | | }); |
| | | public void stopDownloadUpgradeFile() { |
| | | this.stopDownload = true; |
| | | } |
| | | |
| | | /** |
| | | * 向云端获取【设备驱动】下载升级文件 |
| | | * 向云端获取【设备升级文件(驱动,固件)下载】升级文件 |
| | | * |
| | | * @param downloadUrl 云端驱动地址 |
| | | */ |
| | | public void getDeviceDriverDownloadFile(String downloadUrl, CloudCallBeak<ResponseBody> callBack) { |
| | | public void getDeviceUpgradeDownloadFile(String downloadUrl, CloudCallBeak<ResponseBody> callBack) { |
| | | HttpClient.getInstance().downLoadFile(downloadUrl, new CloudCallBeak<ResponseBody>() { |
| | | @Override |
| | | public void onSuccess(ResponseBody s) { |
| | |
| | | |
| | | } |
| | | |
| | | |
| | | //endregion |
| | | |
| | | /** |
| | | * 本地升级包推送(告诉网关下载升级包地址等信息) |
| | | * |
| | |
| | | */ |
| | | public void pushUpgradePacketInfo(String mac, String oid, String module, String version, String size, String url, String sign_method, LinkCallBack<String> linkCallBack) { |
| | | String requestUrl = TopicApi.SET_GATEWAY_UPGRADE_INFO; |
| | | JsonArray jsonArray = new JsonArray(); |
| | | JsonObject json = new JsonObject(); |
| | | json.addProperty("oid", oid);// 物理设备oid |
| | | json.addProperty("module", module);//模块信息 |
| | |
| | | // json.addProperty("is_diff", is_diff);//true=增量包,false=全量包 |
| | | json.addProperty("size", size);//升级包大小 |
| | | json.addProperty("url", url);//升级包下载地址 |
| | | json.addProperty("sign_method", sign_method);//升级包签名类型:md5 |
| | | TcpClient.getInstance().sendDataToLinkGateway(mac, requestUrl, json, "", new HDLLinkCallBack() { |
| | | json.addProperty("sign", sign_method);//升级包签名类型:md5 |
| | | json.addProperty("sign_method", "md5");//升级包签名类型:md5 |
| | | jsonArray.add(json); |
| | | TcpClient.getInstance().sendDataToLinkGateway(mac, requestUrl, jsonArray, "", new HDLLinkCallBack() { |
| | | @Override |
| | | public void onSuccess(String msg) { |
| | | if (linkCallBack != null) { |
| | |
| | | }); |
| | | } |
| | | |
| | | /** |
| | | * 取消升级驱动,固件文件指令 |
| | | * |
| | | * @param mac 设备mac |
| | | * @param linkCallBack 回调update |
| | | */ |
| | | public void cancelUpgrade(String mac, LinkCallBack<Boolean> linkCallBack) { |
| | | String requestUrl = TopicApi.SET_GATEWAY_REMOTE_EDIT; |
| | | // JsonObject json = new JsonObject(); |
| | | // json.addProperty("homeId", UserConfigManage.getInstance().getHomeId()); |
| | | // json.addProperty("server_addr", AppConfigManage.getUserRegionUrl()); |
| | | // json.addProperty("local_secret", UserConfigManage.getInstance().getLocalSecret()); |
| | | // //解密负载数据(写密钥给网关一定是明文,因为那时网关还没有密钥) |
| | | // TcpClient.getInstance().sendDataToLinkGateway(mac, false, requestUrl, json, "", new HDLLinkCallBack() { |
| | | // @Override |
| | | // public void onSuccess(String msg) { |
| | | // GatewayBean gatewayBean = HDLLinkLocalGateway.getInstance().getLocalGateway(mac); |
| | | // if (gatewayBean != null) { |
| | | // if (!TextUtils.isEmpty(UserConfigManage.getInstance().getLocalSecret())) { |
| | | // gatewayBean.setLocalEncrypt(true); |
| | | // } |
| | | // } |
| | | // if (linkCallBack != null) { |
| | | // linkCallBack.onSuccess(true); |
| | | // } |
| | | // |
| | | // } |
| | | // |
| | | // @Override |
| | | // public void onError(HDLLinkException e) { |
| | | // if (linkCallBack != null) { |
| | | // linkCallBack.onError(e); |
| | | // } |
| | | // } |
| | | // }); |
| | | } |
| | | |
| | | |
| | | /** |
| | | * 启动本地服务 |
| | |
| | | //1,建立本地服务; |
| | | //2,告诉网关手机ip和端口; |
| | | //实例化 获取ip 地址 |
| | | MyNanoHttpServer.getInstance(AppManagerUtils.getAppManager().getIPAddress(HDLApp.getInstance())); |
| | | MyNanoHttpServer.getInstance(NetworkUtils.getIPAddress(HDLApp.getInstance())); |
| | | //MyNanoHttpServer.getInstance("127.0.0.1"); |
| | | intentServer = new Intent(); |
| | | intentServer.setClass(HDLApp.getInstance(), MyNanoHttpService.class); |
| | | this.mServiceConnection = serviceConnection; |
| | | this.mServiceStart = true; |
| | | //启动服务监听 |
| | | HDLApp.getInstance().bindService(intentServer, serviceConnection, Context.BIND_AUTO_CREATE); |
| | | } |
| | |
| | | public void unLocalService() { |
| | | if (intentServer != null && this.mServiceConnection != null) { |
| | | HDLApp.getInstance().unbindService(this.mServiceConnection); |
| | | this.mServiceStart = false; |
| | | } |
| | | } |
| | | |
| | | /** |
| | | * 【测试】处理下载文件数据 |
| | | * |
| | | * @param zipData 文件数据流对象 |
| | | * @param md5 检验文件MD5值 |
| | | * @return 返回解密的byte[] |
| | | */ |
| | | public boolean testDisposeDownLoadFile(String fullPath, ResponseBody zipData, String md5) { |
| | | String driver = driver_type; |
| | | //-100:没有数据;-101:没有md5秘钥;-102:md5秘钥不对;-103:有异常; |
| | | if (zipData == null) { |
| | | this.eventBusPost(driver, -100, "没有数据"); |
| | | return false; |
| | | } |
| | | if (TextUtils.isEmpty(md5)) { |
| | | this.eventBusPost(driver, -101, "没有md5秘钥"); |
| | | return false; |
| | | } |
| | | InputStream is = null; |
| | | byte[] buf = new byte[1204 * 4]; |
| | | int len = 0; |
| | | FileOutputStream fos = null; |
| | | File file = null; |
| | | try { |
| | | file = new File(fullPath); |
| | | is = zipData.byteStream(); |
| | | long total = zipData.contentLength(); |
| | | if (total == 0) { |
| | | this.eventBusPost(driver, -100, "没有数据"); |
| | | return false; |
| | | } |
| | | fos = new FileOutputStream(file); |
| | | long sum = 0; |
| | | while ((len = is.read(buf)) != -1) { |
| | | fos.write(buf, 0, len); |
| | | sum += len; |
| | | //todo 下载中,可以自己计算的进度条(可以广播出去) |
| | | int progress = (int) (100 * sum * 1.0f / total); |
| | | if (sum != total) { |
| | | //不等于100都要报,等于100处理整块逻辑完之后再报100; |
| | | this.eventBusPost(driver, progress, "正常上报"); |
| | | } |
| | | } |
| | | fos.flush(); |
| | | fos.close(); |
| | | is.close(); |
| | | byte[] bytes = FileUtils.readFileToByteArray(file); |
| | | String fileMD5 = Md5Utils.encodeMD5(bytes); |
| | | if (TextUtils.isEmpty(fileMD5)) { |
| | | this.eventBusPost(driver, -106, "文件生成md5失败."); |
| | | return false; |
| | | } |
| | | if (!md5.equals(fileMD5)) { |
| | | this.eventBusPost(driver, -107, "md5比对失败."); |
| | | return false; |
| | | } |
| | | this.eventBusPost(driver, 100, "下载完成."); |
| | | |
| | | |
| | | return true; |
| | | } catch (Exception e) { |
| | | this.eventBusPost(driver, -103, e.getMessage()); |
| | | e.printStackTrace(); |
| | | } |
| | | return false; |
| | | } |
| | | |
| | | /** |
| | | * 处理下载文件数据 |
| | | * |
| | | * @param fullPath 文件全路径 |
| | | * @param zipData 文件数据流对象 |
| | | * @param md5 检验文件MD5值 |
| | | * @param driver 驱动或者固件 |
| | | * @param fileFullPath 文件全路径 |
| | | * @param zipData 文件数据流对象 |
| | | * @param md5 检验文件MD5值 |
| | | * @param driver 驱动或者固件 |
| | | * @return 成功true |
| | | */ |
| | | public boolean disposeDownLoadFile(String fullPath, ResponseBody zipData, String md5, String driver) { |
| | | public boolean disposeDownLoadFile(String fileFullPath, ResponseBody zipData, String md5, String driver) { |
| | | this.stopDownload = false; |
| | | //-100:没有数据;-101:没有md5秘钥;-102:md5秘钥不对;-103:有异常; |
| | | if (zipData == null) { |
| | | this.eventBusPost(driver, -100, "没有数据"); |
| | | this.eventBusPost(fileFullPath, driver, -100, "拉不到云端升级文件数据"); |
| | | HdlLogLogic.print("拉不到云端升级文件数据(-100)", true); |
| | | return false; |
| | | } |
| | | if (TextUtils.isEmpty(md5)) { |
| | | this.eventBusPost(driver, -101, "没有md5秘钥"); |
| | | this.eventBusPost(fileFullPath, driver, -101, "拿不到云端上升级文件md5秘钥"); |
| | | HdlLogLogic.print("拿不到云端上升级文件md5秘钥(-101)", true); |
| | | return false; |
| | | } |
| | | InputStream is = null; |
| | |
| | | FileOutputStream fos = null; |
| | | File file = null; |
| | | try { |
| | | file = new File(fullPath); |
| | | file = new File(fileFullPath); |
| | | is = zipData.byteStream(); |
| | | long total = zipData.contentLength(); |
| | | if (total == 0) { |
| | | this.eventBusPost(driver, -100, "没有数据"); |
| | | this.eventBusPost(fileFullPath, driver, -102, "拉不到云端升级文件数据"); |
| | | HdlLogLogic.print("拉不到云端升级文件数据(-102)", true); |
| | | return false; |
| | | } |
| | | fos = new FileOutputStream(file); |
| | | long sum = 0; |
| | | while ((len = is.read(buf)) != -1) { |
| | | if (this.stopDownload) { |
| | | this.eventBusPost(fileFullPath, driver, -108, "用户取消下载升级文件."); |
| | | HdlLogLogic.print("用户取消下载升级文件(-108)", true); |
| | | fos.flush(); |
| | | fos.close(); |
| | | is.close(); |
| | | //todo 用户终止继续读取数据; |
| | | return false; |
| | | } |
| | | fos.write(buf, 0, len); |
| | | sum += len; |
| | | //todo 下载中,可以自己计算的进度条(可以广播出去) |
| | | int progress = (int) (100 * sum * 1.0f / total); |
| | | if (sum != total) { |
| | | //不等于100都要报,等于100处理整块逻辑完之后再报100; |
| | | this.eventBusPost(driver, progress, "正常上报"); |
| | | this.eventBusPost(fileFullPath, driver, progress, "正常上报进度值->" + progress); |
| | | HdlLogLogic.print("正常上报进度值->" + progress, true); |
| | | } |
| | | } |
| | | fos.flush(); |
| | |
| | | //下载文件需要解密之后再进行md5去做比对 |
| | | byte[] decrypt = AesUtils.decrypt(bytes); |
| | | if (decrypt == null) { |
| | | this.eventBusPost(driver, -105, "aes解密失败."); |
| | | this.eventBusPost(fileFullPath, driver, -105, "升级文件aes解密失败."); |
| | | HdlLogLogic.print("升级文件aes解密失败(-105)", true); |
| | | return false; |
| | | } |
| | | String fileMD5 = Md5Utils.encodeMD5(decrypt); |
| | | if (TextUtils.isEmpty(fileMD5)) { |
| | | this.eventBusPost(driver, -106, "文件生成md5失败."); |
| | | this.eventBusPost(fileFullPath, driver, -106, "升级文件生成md5失败失败."); |
| | | HdlLogLogic.print("升级文件生成md5失败失败(-106)", true); |
| | | return false; |
| | | } |
| | | if (!md5.equals(fileMD5)) { |
| | | this.eventBusPost(driver, -107, "md5比对失败."); |
| | | this.eventBusPost(fileFullPath, driver, -107, "升级文件md5比对失败."); |
| | | HdlLogLogic.print("升级文件md5比对失败(-107)", true); |
| | | return false; |
| | | } |
| | | //todo 注意:解密之后,要重新写数据; |
| | | FileUtils.writeByteArrayToFile(file, decrypt, false); |
| | | this.eventBusPost(driver, 100, "下载完成."); |
| | | this.eventBusPost(fileFullPath, driver, 100, "升级文件下载完成."); |
| | | HdlLogLogic.print("升级文件下载完成.", true); |
| | | return true; |
| | | } catch (Exception e) { |
| | | this.eventBusPost(driver, -103, e.getMessage()); |
| | | this.eventBusPost(fileFullPath, driver, -103, e.getMessage()); |
| | | e.printStackTrace(); |
| | | } |
| | | return false; |
| | |
| | | * @param progressValue 进度值 |
| | | * @param describe 描述文本 |
| | | */ |
| | | public void eventBusPost(String type, int progressValue, String describe) { |
| | | public void eventBusPost(String fileFullPath, String type, int progressValue, String describe) { |
| | | |
| | | |
| | | if (this.stopDownload) { |
| | | // HdlFileLogic.getInstance().deleteFile(fileFullPath);//删除下载不完整数据; |
| | | //取消下载广播一次 |
| | | BaseEventBus baseEventBus = new BaseEventBus(); |
| | | baseEventBus.setTopic(localDownloadProgress); |
| | | Progress progress = new Progress(); |
| | | progress.step = -108; |
| | | progress.describe = describe; |
| | | baseEventBus.setType(type); |
| | | baseEventBus.setData(progress); |
| | | EventBus.getDefault().post(baseEventBus); |
| | | return; |
| | | } |
| | | // HdlThreadLogic.runMainThread(new Runnable() { |
| | | // @Override |
| | | // public void run() { |
| | |
| | | Progress progress = new Progress(); |
| | | progress.step = progressValue; |
| | | progress.describe = describe; |
| | | progress.type = type; |
| | | baseEventBus.setType(type); |
| | | baseEventBus.setData(progress); |
| | | EventBus.getDefault().post(baseEventBus); |
| | | // } |
| | | // }); |
| | | |
| | | |
| | | } |
| | | |
| | | |
| | | //自定义上报下载驱动或固件进度主题 |
| | | public static final String localDownloadProgress = "cloud/download/progress/up"; |
| | | |
| | | |
| | | |
| | | public static final String driver_type = "driver"; |
| | | public static final String firmware_type = "firmware"; |
| | |
| | | public int step; |
| | | //进度描述信息 |
| | | public String describe; |
| | | //表示是驱动,固件; |
| | | public String type; |
| | | } |
| | | |
| | | |