From 52d4d835cc4189ebee57925e4274025ccae605e4 Mon Sep 17 00:00:00 2001 From: zhangxin_T Date: Wed, 15 Jun 2022 08:25:39 +0800 Subject: [PATCH] modify continuous task xts testcase Signed-off-by: zhangxin_T --- .../resourceschedule_standard/BUILD.gn | 1 + .../backgroundtaskmanager/Test.json | 24 + .../default/test/ContinuousTaskJsunit.test.js | 516 ++++++------------ .../continuoustaskrelyhap/BUILD.gn | 38 ++ .../entry/src/main/config.json | 78 +++ .../src/main/js/ServiceAbility/service.js | 223 ++++++++ .../entry/src/main/js/default/app.js | 22 + .../entry/src/main/js/default/i18n/en-US.json | 6 + .../entry/src/main/js/default/i18n/zh-CN.json | 6 + .../src/main/js/default/pages/index/index.css | 9 + .../src/main/js/default/pages/index/index.hml | 5 + .../src/main/js/default/pages/index/index.js | 29 + .../main/resources/base/element/string.json | 16 + .../src/main/resources/base/media/icon.png | Bin 0 -> 6790 bytes .../signature/openharmony_sx.p7b | Bin 0 -> 3442 bytes 15 files changed, 620 insertions(+), 353 deletions(-) create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/BUILD.gn create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/config.json create mode 100644 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/ServiceAbility/service.js create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/app.js create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/i18n/en-US.json create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/i18n/zh-CN.json create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.css create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.hml create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.js create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/resources/base/element/string.json create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/resources/base/media/icon.png create mode 100755 resourceschedule/resourceschedule_standard/continuoustaskrelyhap/signature/openharmony_sx.p7b diff --git a/resourceschedule/resourceschedule_standard/BUILD.gn b/resourceschedule/resourceschedule_standard/BUILD.gn index 98e9ed0e8..7aab0b9e0 100755 --- a/resourceschedule/resourceschedule_standard/BUILD.gn +++ b/resourceschedule/resourceschedule_standard/BUILD.gn @@ -18,6 +18,7 @@ group("resourceschedule_standard") { if (is_standard_system) { deps = [ "backgroundtaskmanager:resourceschedule_backgroundtaskmanager_js_test", + "continuoustaskrelyhap:ContinuousTaskRelyHap", "deviceusagestatisticsjsunit:resourceschedule_deviceusagestatisticsjsunit_js_test", "reminderagent:resourceschedule_reminderagent_js_test", "workscheduler:resourceschedule_workscheduler_js_test", diff --git a/resourceschedule/resourceschedule_standard/backgroundtaskmanager/Test.json b/resourceschedule/resourceschedule_standard/backgroundtaskmanager/Test.json index e22867d19..904d83621 100644 --- a/resourceschedule/resourceschedule_standard/backgroundtaskmanager/Test.json +++ b/resourceschedule/resourceschedule_standard/backgroundtaskmanager/Test.json @@ -13,6 +13,30 @@ ], "type": "AppInstallKit", "cleanup-apps": true + }, + { + "type": "ShellKit", + "run-command": [ + "remount", + "mkdir /data/test" + ] + }, + { + "type": "PushKit", + "push": [ + "ContinuousTaskRelyHap.hap->/data/test/ContinuousTaskRelyHap.hap" + ] + }, + { + "type": "ShellKit", + "run-command": [ + "bm install -p /data/test/ContinuousTaskRelyHap.hap", + "aa start -a com.example.continuoustaskserver.MainAbility -b com.example.continuoustaskserver" + ], + "teardown-command": [ + "bm uninstall -n com.example.continuoustaskserver", + "rm -rf data/test" + ] } ] } diff --git a/resourceschedule/resourceschedule_standard/backgroundtaskmanager/src/main/js/default/test/ContinuousTaskJsunit.test.js b/resourceschedule/resourceschedule_standard/backgroundtaskmanager/src/main/js/default/test/ContinuousTaskJsunit.test.js index 00e264899..860bdf5be 100644 --- a/resourceschedule/resourceschedule_standard/backgroundtaskmanager/src/main/js/default/test/ContinuousTaskJsunit.test.js +++ b/resourceschedule/resourceschedule_standard/backgroundtaskmanager/src/main/js/default/test/ContinuousTaskJsunit.test.js @@ -13,13 +13,13 @@ * limitations under the License. */ -import notification from '@ohos.notification'; -import wantAgent from '@ohos.wantAgent'; -import particleAbility from '@ohos.ability.particleAbility' -import backgroundTaskManager from '@ohos.backgroundTaskManager' -import featureAbility from '@ohos.ability.featureAbility' +import backgroundTaskManager from '@ohos.backgroundTaskManager'; +import featureAbility from '@ohos.ability.featureAbility'; +import commonEvent from '@ohos.commonEvent'; -import {describe, beforeAll, beforeEach, afterEach, afterAll, it, expect} from 'deccjsunit/index' +import {describe, beforeAll, beforeEach, afterEach, afterAll, it, expect} from 'deccjsunit/index'; + +const TAG = "ContinuousTaskJsTest "; describe("ContinuousTaskJsTest", function () { beforeAll(function() { @@ -27,7 +27,7 @@ describe("ContinuousTaskJsTest", function () { /* * @tc.setup: setup invoked before all testcases */ - console.info('beforeAll called') + console.info('beforeAll called'); }) afterAll(function() { @@ -35,7 +35,7 @@ describe("ContinuousTaskJsTest", function () { /* * @tc.teardown: teardown invoked after all testcases */ - console.info('afterAll called') + console.info('afterAll called'); }) beforeEach(function() { @@ -43,7 +43,7 @@ describe("ContinuousTaskJsTest", function () { /* * @tc.setup: setup invoked before each testcases */ - console.info('beforeEach called') + console.info('beforeEach called'); }) afterEach(function() { @@ -51,431 +51,241 @@ describe("ContinuousTaskJsTest", function () { /* * @tc.teardown: teardown invoked after each testcases */ - console.info('afterEach called') - particleAbility.cancelBackgroundRunning(); - setTimeout(() => {}, 500); - backgroundTaskManager.stopBackgroundRunning(featureAbility.getContext()); - setTimeout(() => {}, 500); + console.info('afterEach called'); }) + async function startAbilityWithOption(subscribeInfo, option, event, done) { + commonEvent.createSubscriber(subscribeInfo, (err, subscriber) => { + if (subscriber !== null && subscriber !== undefined) { + console.info(TAG + "Subscribe begin"); + commonEvent.subscribe(subscriber, (err, data) => { + if (data !== null && data !== undefined) { + console.info(TAG + "Get comment event: " + JSON.stringify(data)); + if (data.event === event) { + expect(true).assertTrue(); + } else { + expect(false).assertTrue(); + } + } else { + console.error(TAG + "SubscribeCallBack failed"); + expect(false).assertTrue(); + } + done(); + }) + } else { + console.error(TAG + "createSubscriber failed"); + expect(false).assertTrue(); + done(); + } + }) + + setTimeout(() => { + console.info(TAG + "Start ability with option:" + option + " begin"); + featureAbility.startAbility( + { + want: { + bundleName: "com.example.continuoustaskserver", + abilityName: "com.example.continuoustaskserver.ServiceAbility", + parameters: { + option: option + } + } + } + ).catch(() => { + console.error(TAG + "Start ability failed"); + expect(false).assertTrue(); + done(); + }); + }, 1000); + } + /* * @tc.name:ContinuousTaskJsTest001 - * @tc.desc:verify new startBackgroundrunning interface promise mode work properly + * @tc.desc:verify old startBackgroundrunning interface callback mode work properly * @tc.type: FUNC * @tc.require: */ it("ContinuousTaskJsTest001", 0, async function (done) { - let wantAgentInfo = { - wants: [ - { - bundleName: "com.continuoustask.test", - abilityName: "com.continuoustask.test.MainAbility" - } - ], - operationType: 2, - requestCode: 0, - wantAgentFlags: [3] - }; - wantAgent.getWantAgent(wantAgentInfo).then((data) => { - backgroundTaskManager.startBackgroundRunning(featureAbility.getContext(), - backgroundTaskManager.BackgroundMode.DATA_TRANSFER, data).then(() => { - console.log("ContinuousTaskJsTest001 startBackgroundRunning success"); - expect(true).assertTrue(); - setTimeout(() => { - done(); - }, 500); - }).catch((err) => { - expect(false).assertTrue(); - console.log("ContinuousTaskJsTest001 startBackgroundRunning failure"); - setTimeout(() => { - done(); - }, 500); - }); - }); + console.info(TAG + '-----------------ContinuousTaskJsTest001 begin--------------------'); + startAbilityWithOption({ events: ["startTaskUseApi7Callback"] }, "testcase1", "startTaskUseApi7Callback", done); }) /* * @tc.name:ContinuousTaskJsTest002 - * @tc.desc:verify new startBackgroundrunning interface callback mode work properly + * @tc.desc:verify cancelBackgroundrunning interface callback mode work properly * @tc.type: FUNC * @tc.require: */ it("ContinuousTaskJsTest002", 0, async function (done) { - function conTaskCallback(err, data) { - if (err) { - console.info('ContinuousTaskJsTest002 startBackgroundRunning failed'); - expect(false).assertTrue(); - } else { - console.info('ContinuousTaskJsTest002 startBackgroundRunning succeed'); - expect(true).assertTrue(); - } - setTimeout(()=>{ - done(); - }, 500); - } - let wantAgentInfo = { - wants: [ - { - bundleName: "com.continuoustask.test", - abilityName: "com.continuoustask.test.MainAbility" - } - ], - operationType: 2, - requestCode: 0, - wantAgentFlags: [3] - }; - wantAgent.getWantAgent(wantAgentInfo).then((data) => { - backgroundTaskManager.startBackgroundRunning(featureAbility.getContext(), - backgroundTaskManager.BackgroundMode.DATA_TRANSFER, data, conTaskCallback); - }); + console.info(TAG + '-----------------ContinuousTaskJsTest002 begin--------------------'); + startAbilityWithOption({ events: ["stopTaskUseApi7Callback"] }, "testcase2", "stopTaskUseApi7Callback", done); }) /* * @tc.name:ContinuousTaskJsTest003 - * @tc.desc:verify old startBackgroundrunning interface promise mode work properly + * @tc.desc:verify new startBackgroundrunning interface callback mode work properly * @tc.type: FUNC * @tc.require: */ it("ContinuousTaskJsTest003", 0, async function (done) { - let wantAgentInfo = { - wants: [ - { - bundleName: "com.continuoustask.test", - abilityName: "com.continuoustask.test.MainAbility" - } - ], - operationType: 2, - requestCode: 0, - wantAgentFlags: [3] - }; - wantAgent.getWantAgent(wantAgentInfo).then((data) => { - let basicContent = { - title: "title", - text: "text" - }; - - let notificationContent = { - contentType: notification.ContentType.NOTIFICATION_CONTENT_BASIC_TEXT, - normal: basicContent - }; - - let request = { - content: notificationContent, - wantAgent: data - } - - let id = 1; - - particleAbility.startBackgroundRunning(id, request).then((data) => { - console.log("ContinuousTaskJsTest003 startBackgroundRunning success"); - expect(true).assertTrue(); - setTimeout(() => { - done(); - }, 500); - }).catch((err) => { - expect(false).assertTrue(); - console.log("ContinuousTaskJsTest003 startBackgroundRunning failure"); - setTimeout(() => { - done(); - }, 500); - }); - }); + console.info(TAG + '-----------------ContinuousTaskJsTest003 begin--------------------'); + startAbilityWithOption({ events: ["startTaskUseApi8Callback"] }, "testcase3", "startTaskUseApi8Callback", done); }) /* * @tc.name:ContinuousTaskJsTest004 - * @tc.desc:verify old startBackgroundrunning interface callback mode work properly + * @tc.desc:verify new stopBackgroundrunning interface callback mode work properly * @tc.type: FUNC * @tc.require: */ it("ContinuousTaskJsTest004", 0, async function (done) { - function conTaskCallback(err, data) { - if (err) { - console.info('ContinuousTaskJsTest004 startBackgroundRunning failure'); - expect(false).assertTrue(); - } else { - console.info('ContinuousTaskJsTest004 startBackgroundRunning success'); - expect(true).assertTrue(); - } - setTimeout(()=>{ - done(); - }, 500); - } - let wantAgentInfo = { - wants: [ - { - bundleName: "com.continuoustask.test", - abilityName: "com.continuoustask.test.MainAbility" - } - ], - operationType: 2, - requestCode: 0, - wantAgentFlags: [3] - }; - wantAgent.getWantAgent(wantAgentInfo).then((data) => { - let basicContent = { - title: "title", - text: "text" - }; - - let notificationContent = { - contentType: notification.ContentType.NOTIFICATION_CONTENT_BASIC_TEXT, - normal: basicContent - }; - - let request = { - content: notificationContent, - wantAgent: data - } - - let id = 1; - - particleAbility.startBackgroundRunning(id, request, conTaskCallback); - }); + console.info(TAG + '-----------------ContinuousTaskJsTest004 begin--------------------'); + startAbilityWithOption({ events: ["stopTaskUseApi8Callback"] }, "testcase4", "stopTaskUseApi8Callback", done); }) /* * @tc.name:ContinuousTaskJsTest005 - * @tc.desc:verify new api stopBackgroundrunning interface promise mode work properly + * @tc.desc:verify old startBackgroundrunning interface promise mode work properly * @tc.type: FUNC * @tc.require: */ it("ContinuousTaskJsTest005", 0, async function (done) { - let wantAgentInfo = { - wants: [ - { - bundleName: "com.continuoustask.test", - abilityName: "com.continuoustask.test.MainAbility" - } - ], - operationType: 2, - requestCode: 0, - wantAgentFlags: [3] - }; - wantAgent.getWantAgent(wantAgentInfo).then((data) => { - backgroundTaskManager.startBackgroundRunning(featureAbility.getContext(), - backgroundTaskManager.BackgroundMode.DATA_TRANSFER, data).then((data) => { - backgroundTaskManager.stopBackgroundRunning(featureAbility.getContext()).then((data) => { - console.log("ContinuousTaskJsTest005 cancelBackgroundRunning success"); - expect(true).assertTrue(); - setTimeout(() => { - done(); - }, 500); - }).catch((err) => { - expect(false).assertTrue(); - console.log("ContinuousTaskJsTest005 cancelBackgroundRunning failure"); - setTimeout(() => { - done(); - }, 500); - }); - }) - }); + console.info(TAG + '-----------------ContinuousTaskJsTest005 begin--------------------'); + startAbilityWithOption({ events: ["startTaskUseApi7Promise"] }, "testcase5", "startTaskUseApi7Promise", done); }) /* - * @tc.name:ContinuousTaskJsTest006 - * @tc.desc:verify new api stopBackgroundrunning interface callback mode work properly - * @tc.type: FUNC - * @tc.require: - */ + * @tc.name:ContinuousTaskJsTest006 + * @tc.desc:verify cancelBackgroundrunning interface promise mode work properly + * @tc.type: FUNC + * @tc.require: + */ it("ContinuousTaskJsTest006", 0, async function (done) { - function conTaskCallback(err, data) { - if (err) { - console.info('ContinuousTaskJsTest006 startBackgroundRunning failure'); - expect(false).assertTrue(); - } else { - console.info('ContinuousTaskJsTest006 startBackgroundRunning success'); - expect(true).assertTrue(); - } - setTimeout(()=>{ - done(); - }, 500); - } - let wantAgentInfo = { - wants: [ - { - bundleName: "com.continuoustask.test", - abilityName: "com.continuoustask.test.MainAbility" - } - ], - operationType: 2, - requestCode: 0, - wantAgentFlags: [3] - }; - wantAgent.getWantAgent(wantAgentInfo).then((data) => { - backgroundTaskManager.startBackgroundRunning(featureAbility.getContext(), - backgroundTaskManager.BackgroundMode.DATA_TRANSFER, data).then((data) => { - backgroundTaskManager.stopBackgroundRunning(featureAbility.getContext(), conTaskCallback); - }) - }); + console.info(TAG + '-----------------ContinuousTaskJsTest006 begin--------------------'); + startAbilityWithOption({ events: ["stopTaskUseApi7Promise"] }, "testcase6", "stopTaskUseApi7Promise", done); }) /* * @tc.name:ContinuousTaskJsTest007 - * @tc.desc:verify old api cancelBackgroundrunning interface promise mode work properly + * @tc.desc:verify new startBackgroundrunning interface promise mode work properly * @tc.type: FUNC * @tc.require: */ it("ContinuousTaskJsTest007", 0, async function (done) { - let wantAgentInfo = { - wants: [ - { - bundleName: "com.continuoustask.test", - abilityName: "com.continuoustask.test.MainAbility" - } - ], - operationType: 2, - requestCode: 0, - wantAgentFlags: [3] - }; - await wantAgent.getWantAgent(wantAgentInfo).then((data) => { - particleAbility.startBackgroundRunning(data); - setTimeout(()=>{ - }, 500); - }); - - particleAbility.cancelBackgroundRunning().then(() => { - console.log("ContinuousTaskJsTest007 cancelBackgroundRunning success"); - expect(true).assertTrue(); - setTimeout(() => { - done(); - }, 500); - }).catch( (err) => { - expect(false).assertTrue(); - console.log("ContinuousTaskJsTest007 cancelBackgroundRunning failure"); - setTimeout(() => { - done(); - }, 500); - }); + console.info(TAG + '-----------------ContinuousTaskJsTest007 begin--------------------'); + startAbilityWithOption({ events: ["startTaskUseApi8Promise"] }, "testcase7", "startTaskUseApi8Promise", done); }) /* * @tc.name:ContinuousTaskJsTest008 - * @tc.desc:verify old cancelBackgroundrunning interface callback mode work properly + * @tc.desc:verify new stopBackgroundrunning interface promise mode work properly * @tc.type: FUNC * @tc.require: */ it("ContinuousTaskJsTest008", 0, async function (done) { - function conTaskCallback(err, data) { - if (err) { - console.info('ContinuousTaskJsTest008 startBackgroundRunning failure'); - expect(false).assertTrue(); - } else { - console.info('ContinuousTaskJsTest008 startBackgroundRunning success'); - expect(true).assertTrue(); - } - setTimeout(()=>{ - done(); - }, 500); - } - let wantAgentInfo = { - wants: [ - { - bundleName: "com.continuoustask.test", - abilityName: "com.continuoustask.test.MainAbility" - } - ], - operationType: 2, - requestCode: 0, - wantAgentFlags: [3] - }; - await wantAgent.getWantAgent(wantAgentInfo).then((data) => { - particleAbility.startBackgroundRunning(data); - setTimeout(()=>{ - }, 500); - }); + console.info(TAG + '-----------------ContinuousTaskJsTest008 begin--------------------'); + startAbilityWithOption({ events: ["stopTaskUseApi8Promise"] }, "testcase8", "stopTaskUseApi8Promise", done); + }) - particleAbility.cancelBackgroundRunning(conTaskCallback); + /* + * @tc.name:ContinuousTaskJsTest009 + * @tc.desc:verify continuous task request failed with invalid background mode + * @tc.type: FUNC + * @tc.require: + */ + it("ContinuousTaskJsTest009", 0, async function (done) { + console.info(TAG + '-----------------ContinuousTaskJsTest009 begin--------------------'); + startAbilityWithOption({ events: ["startTaskInvalidBgmode"] }, "testcase9", "startTaskInvalidBgmode", done); }) - - /* - * @tc.name: ContinuousTaskJsTest009 - * @tc.desc: test work scheduler constant - * @tc.type: FUNC - * @tc.require: - */ - it("ContinuousTaskJsTest009", 0, function (done) { - console.info('----------------------ContinuousTaskJsTest009---------------------------'); + + /* + * @tc.name: ContinuousTaskJsTest009 + * @tc.desc: test work scheduler constant + * @tc.type: FUNC + * @tc.require: + */ + it("ContinuousTaskJsTest010", 0, async function (done) { + console.info(TAG + '-----------------ContinuousTaskJsTest010 begin--------------------'); let value1 = backgroundTaskManager.BackgroundMode.DATA_TRANSFER; - expect(value1).assertEqual(1) + expect(value1).assertEqual(1); done(); }) - - /* - * @tc.name: ContinuousTaskJsTest010 - * @tc.desc: test work scheduler constant - * @tc.type: FUNC - * @tc.require: - */ - it("ContinuousTaskJsTest010", 0, function (done) { - console.info('----------------------ContinuousTaskJsTest010---------------------------'); + + /* + * @tc.name: ContinuousTaskJsTest010 + * @tc.desc: test work scheduler constant + * @tc.type: FUNC + * @tc.require: + */ + it("ContinuousTaskJsTest011", 0, async function (done) { + console.info(TAG + '-----------------ContinuousTaskJsTest011 begin--------------------'); let value1 = backgroundTaskManager.BackgroundMode.AUDIO_PLAYBACK; - expect(value1).assertEqual(2) + expect(value1).assertEqual(2); done(); }) - - /* - * @tc.name: ContinuousTaskJsTest011 - * @tc.desc: test work scheduler constant - * @tc.type: FUNC - * @tc.require: - */ - it("ContinuousTaskJsTest011", 0, function (done) { - console.info('----------------------ContinuousTaskJsTest011---------------------------'); + + /* + * @tc.name: ContinuousTaskJsTest011 + * @tc.desc: test work scheduler constant + * @tc.type: FUNC + * @tc.require: + */ + it("ContinuousTaskJsTest012", 0, async function (done) { + console.info(TAG + '-----------------ContinuousTaskJsTest012 begin--------------------'); let value1 = backgroundTaskManager.BackgroundMode.AUDIO_RECORDING; - expect(value1).assertEqual(3) + expect(value1).assertEqual(3); done(); }) - - /* - * @tc.name: ContinuousTaskJsTest012 - * @tc.desc: test work scheduler constant - * @tc.type: FUNC - * @tc.require: - */ - it("ContinuousTaskJsTest012", 0, function (done) { - console.info('----------------------ContinuousTaskJsTest012---------------------------'); + + /* + * @tc.name: ContinuousTaskJsTest012 + * @tc.desc: test work scheduler constant + * @tc.type: FUNC + * @tc.require: + */ + it("ContinuousTaskJsTest013", 0, async function (done) { + console.info(TAG + '-----------------ContinuousTaskJsTest013 begin--------------------'); let value1 = backgroundTaskManager.BackgroundMode.LOCATION; - expect(value1).assertEqual(4) + expect(value1).assertEqual(4); done(); }) - - /* - * @tc.name: ContinuousTaskJsTest013 - * @tc.desc: test work scheduler constant - * @tc.type: FUNC - * @tc.require: - */ - it("ContinuousTaskJsTest013", 0, function (done) { - console.info('----------------------ContinuousTaskJsTest013---------------------------'); + + /* + * @tc.name: ContinuousTaskJsTest013 + * @tc.desc: test work scheduler constant + * @tc.type: FUNC + * @tc.require: + */ + it("ContinuousTaskJsTest014", 0, async function (done) { + console.info(TAG + '-----------------ContinuousTaskJsTest014 begin--------------------'); let value1 = backgroundTaskManager.BackgroundMode.BLUETOOTH_INTERACTION; - expect(value1).assertEqual(5) + expect(value1).assertEqual(5); done(); }) - - /* - * @tc.name: ContinuousTaskJsTest014 - * @tc.desc: test work scheduler constant - * @tc.type: FUNC - * @tc.require: - */ - it("ContinuousTaskJsTest014", 0, function (done) { - console.info('----------------------ContinuousTaskJsTest014---------------------------'); + + /* + * @tc.name: ContinuousTaskJsTest014 + * @tc.desc: test work scheduler constant + * @tc.type: FUNC + * @tc.require: + */ + it("ContinuousTaskJsTest015", 0, async function (done) { + console.info(TAG + '-----------------ContinuousTaskJsTest015 begin--------------------'); let value1 = backgroundTaskManager.BackgroundMode.MULTI_DEVICE_CONNECTION; - expect(value1).assertEqual(6) + expect(value1).assertEqual(6); done(); }) - - /* - * @tc.name: ContinuousTaskJsTest017 - * @tc.desc: test work scheduler constant - * @tc.type: FUNC - * @tc.require: - */ - it("ContinuousTaskJsTest017", 0, function (done) { - console.info('----------------------ContinuousTaskJsTest017---------------------------'); + + /* + * @tc.name: ContinuousTaskJsTest017 + * @tc.desc: test work scheduler constant + * @tc.type: FUNC + * @tc.require: + */ + it("ContinuousTaskJsTest016", 0, async function (done) { + console.info(TAG + '-----------------ContinuousTaskJsTest016 begin--------------------'); let value1 = backgroundTaskManager.BackgroundMode.TASK_KEEPING; - expect(value1).assertEqual(9) + expect(value1).assertEqual(9); done(); }) }) \ No newline at end of file diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/BUILD.gn b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/BUILD.gn new file mode 100755 index 000000000..cf8d1054d --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/BUILD.gn @@ -0,0 +1,38 @@ +# Copyright (c) 2021 Huawei Device Co., Ltd. +# 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. +import("//test/xts/tools/build/suite.gni") + +ohos_hap("ContinuousTaskRelyHap") { + hap_profile = "./entry/src/main/config.json" + hap_name = "ContinuousTaskRelyHap" + subsystem_name = XTS_SUITENAME + final_hap_path = + "${SUITES_OUTPUT_ROOT}/${XTS_SUITENAME}/testcases/${hap_name}.hap" + testonly = true + deps = [ + ":hjs_demo_js_assets", + ":hjs_demo_js_service_assets", + ":hjs_demo_resources", + ] + certificate_profile = "./signature/openharmony_sx.p7b" +} +ohos_js_assets("hjs_demo_js_assets") { + source_dir = "./entry/src/main/js/default" +} +ohos_js_assets("hjs_demo_js_service_assets") { + source_dir = "./entry/src/main/js/ServiceAbility" +} +ohos_resources("hjs_demo_resources") { + sources = [ "./entry/src/main/resources" ] + hap_profile = "./entry/src/main/config.json" +} diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/config.json b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/config.json new file mode 100755 index 000000000..eb91b3a99 --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/config.json @@ -0,0 +1,78 @@ +{ + "app": { + "bundleName": "com.example.continuoustaskserver", + "vendor": "example", + "version": { + "code": 1000000, + "name": "1.0.0" + }, + "apiVersion": { + "compatible": 7, + "target": 7 + } + }, + "deviceConfig": {}, + "module": { + "package": "com.example.continuoustaskserver", + "name": ".MyApplication", + "deviceType": [ + "phone" + ], + "distro": { + "deliveryWithInstall": true, + "moduleName": "entry", + "moduleType": "entry", + "installationFree": false + }, + "abilities": [ + { + "skills": [ + { + "entities": [ + "entity.system.home" + ], + "actions": [ + "action.system.home" + ] + } + ], + "name": "com.example.continuoustaskserver.MainAbility", + "icon": "$media:icon", + "description": "$string:mainability_description", + "label": "$string:app_name", + "type": "page", + "visible": true, + "launchType": "standard" + }, + { + "backgroundModes": [ + "location" + ], + "visible": true, + "srcPath": "ServiceAbility", + "name": ".ServiceAbility", + "icon": "$media:icon", + "srcLanguage": "js", + "description": "$string:description_serviceability", + "type": "service" + } + ], + "js": [ + { + "pages": [ + "pages/index/index" + ], + "name": "default", + "window": { + "designWidth": 720, + "autoDesignWidth": false + } + } + ], + "reqPermissions": [ + { + "name": "ohos.permission.KEEP_BACKGROUND_RUNNING" + } + ] + } +} \ No newline at end of file diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/ServiceAbility/service.js b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/ServiceAbility/service.js new file mode 100644 index 000000000..66098e2d6 --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/ServiceAbility/service.js @@ -0,0 +1,223 @@ +/* + * Copyright (C) 2022 Huawei Device Co., Ltd. + * 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. + */ + +import notification from '@ohos.notification'; +import featureAbility from '@ohos.ability.featureAbility'; +import wantAgent from '@ohos.wantAgent'; +import backgroundTaskManager from '@ohos.backgroundTaskManager'; +import particleAbility from '@ohos.ability.particleAbility'; +import commonEvent from '@ohos.commonEvent'; + +const TAG = "BGMODE_TEST "; + +let event; + +let request = { + content: { + contentType: notification.ContentType.NOTIFICATION_CONTENT_BASIC_TEXT, + normal: { + title: "title", + text: "text" + } + }, + id: 1 +}; + +let wantAgentInfo = { + wants: [ + { + bundleName: "com.example.continuoustaskserver", + abilityName: "com.example.continuoustaskserver.MainAbility" + } + ], + operationType: wantAgent.OperationType.START_ABILITY, + requestCode: 0, + wantAgentFlags: [wantAgent.WantAgentFlags.UPDATE_PRESENT_FLAG] +}; + +function callback(err, data) { + if (err) { + console.error(TAG + " Operation failed"); + } else { + console.info(TAG + " Operation succeed"); + commonEvent.publish(event, (err) => { + if (err.code) { + console.error(TAG + "PublishCallBack failed"); + } else { + console.info(TAG + "Publish succeed"); + } + }) + } +} + +function startContinuousTaskUseApi7Callback() { + event = "startTaskUseApi7Callback"; + particleAbility.startBackgroundRunning(1, request, callback); +} + +function stopContinuousTaskUseApi7Callback() { + event = "stopTaskUseApi7Callback"; + particleAbility.startBackgroundRunning(1, request).then(() => { + particleAbility.cancelBackgroundRunning(callback); + }); +} + +function startContinuousTaskUseApi8Callback() { + event = "startTaskUseApi8Callback"; + wantAgent.getWantAgent(wantAgentInfo).then((data) => { + backgroundTaskManager.startBackgroundRunning(featureAbility.getContext(), + backgroundTaskManager.BackgroundMode.LOCATION, data, callback); + }); +} + +function stopContinuousTaskUseApi8Callback() { + event = "stopTaskUseApi8Callback"; + wantAgent.getWantAgent(wantAgentInfo).then((data) => { + return backgroundTaskManager.startBackgroundRunning(featureAbility.getContext(), + backgroundTaskManager.BackgroundMode.LOCATION, data); + }).then(() => { + backgroundTaskManager.stopBackgroundRunning(featureAbility.getContext(), callback); + }); +} + +function startContinuousTaskUseApi7Promise() { + particleAbility.startBackgroundRunning(1, request).then(() => { + commonEvent.publish("startTaskUseApi7Promise", (err) => { + if (err.code) { + console.error(TAG + "PublishCallBack failed"); + } else { + console.info(TAG + "Publish succeed"); + } + }); + }); +} + +function stopContinuousTaskUseApi7Promise() { + particleAbility.startBackgroundRunning(1, request).then(() => { + return particleAbility.cancelBackgroundRunning(); + }).then(() => { + commonEvent.publish("stopTaskUseApi7Promise", (err) => { + if (err.code) { + console.error(TAG + "PublishCallBack failed"); + } else { + console.info(TAG + "Publish succeed"); + } + }); + }); +} + +function startContinuousTaskUseApi8Promise() { + wantAgent.getWantAgent(wantAgentInfo).then((data) => { + return backgroundTaskManager.startBackgroundRunning(featureAbility.getContext(), + backgroundTaskManager.BackgroundMode.LOCATION, data); + }).then(() => { + commonEvent.publish("startTaskUseApi8Promise", (err) => { + if (err.code) { + console.error(TAG + "PublishCallBack failed"); + } else { + console.info(TAG + "Publish succeed"); + } + }); + }); +} + +function stopContinuousTaskUseApi8Promise() { + wantAgent.getWantAgent(wantAgentInfo).then((data) => { + return backgroundTaskManager.startBackgroundRunning(featureAbility.getContext(), + backgroundTaskManager.BackgroundMode.LOCATION, data); + }).then(() => { + return backgroundTaskManager.stopBackgroundRunning(featureAbility.getContext()); + }).then(() => { + commonEvent.publish("stopTaskUseApi8Promise", (err) => { + if (err.code) { + console.error(TAG + "PublishCallBack failed"); + } else { + console.info(TAG + "Publish succeed"); + } + }) + }) +} + +function startContinuousTaskInvalidBgmode() { + wantAgent.getWantAgent(wantAgentInfo).then((data) => { + return backgroundTaskManager.startBackgroundRunning(featureAbility.getContext(), + backgroundTaskManager.BackgroundMode.BLUETOOTH_INTERACTION, data); + }).catch(() => { + commonEvent.publish("startTaskInvalidBgmode", (err) => { + if (err.code) { + console.error(TAG + "PublishCallBack failed"); + } else { + console.info(TAG + "Publish succeed"); + } + }); + }); +} + +function handleOption(want) { + switch (want.parameters.option) { + case "testcase1": + startContinuousTaskUseApi7Callback(); + break; + case "testcase2": + stopContinuousTaskUseApi7Callback(); + break; + case "testcase3": + startContinuousTaskUseApi8Callback(); + break; + case "testcase4": + stopContinuousTaskUseApi8Callback(); + break; + case "testcase5": + startContinuousTaskUseApi7Promise(); + break; + case "testcase6": + stopContinuousTaskUseApi7Promise(); + break; + case "testcase7": + startContinuousTaskUseApi8Promise(); + break; + case "testcase8": + stopContinuousTaskUseApi8Promise(); + break; + case "testcase9": + startContinuousTaskInvalidBgmode(); + break; + default: + console.warn(TAG + 'Unknown option'); + break; + } +} + +export default { + onStart() { + console.info(TAG + 'ServiceAbility onStart'); + }, + onStop() { + console.info(TAG + 'ServiceAbility onStop'); + }, + onCommand(want, startId) { + console.info(TAG + 'ServiceAbility onCommand'); + console.info(TAG + "Get onCommand want: " + JSON.stringify(want)); + particleAbility.cancelBackgroundRunning(); + backgroundTaskManager.stopBackgroundRunning(featureAbility.getContext()); + handleOption(want); + }, + onConnect(want) { + console.info(TAG + 'ServiceAbility onConnect'); + }, + onDisConnect(want) { + console.info(TAG + 'ServiceAbility onDisConnect'); + }, +}; \ No newline at end of file diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/app.js b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/app.js new file mode 100755 index 000000000..2a68c1992 --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/app.js @@ -0,0 +1,22 @@ +/* + * Copyright (c) 2021 Huawei Device Co., Ltd. + * 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. + */ +export default { + onCreate() { + console.info('AceApplication onCreate'); + }, + onDestroy() { + console.info('AceApplication onDestroy'); + } +}; diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/i18n/en-US.json b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/i18n/en-US.json new file mode 100755 index 000000000..e63c70d97 --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/i18n/en-US.json @@ -0,0 +1,6 @@ +{ + "strings": { + "hello": "Hello", + "world": "World" + } +} \ No newline at end of file diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/i18n/zh-CN.json b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/i18n/zh-CN.json new file mode 100755 index 000000000..de6ee5748 --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/i18n/zh-CN.json @@ -0,0 +1,6 @@ +{ + "strings": { + "hello": "您好", + "world": "世界" + } +} \ No newline at end of file diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.css b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.css new file mode 100755 index 000000000..6fda79275 --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.css @@ -0,0 +1,9 @@ +.container { + flex-direction: column; + justify-content: center; + align-items: center; +} + +.title { + font-size: 100px; +} diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.hml b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.hml new file mode 100755 index 000000000..c45422b42 --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.hml @@ -0,0 +1,5 @@ +
+ + {{title}} + +
diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.js b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.js new file mode 100755 index 000000000..f445115a1 --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/js/default/pages/index/index.js @@ -0,0 +1,29 @@ +/* + * Copyright (c) 2021 Huawei Device Co., Ltd. + * 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. + */ + +export default { + data: { + title: "" + }, + onInit() { + this.title = this.$t('strings.world'); + }, + onShow() { + console.info('onShow finish') + }, + onReady() { + console.info('onReady'); + }, +} \ No newline at end of file diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/resources/base/element/string.json b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/resources/base/element/string.json new file mode 100755 index 000000000..d88552332 --- /dev/null +++ b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/resources/base/element/string.json @@ -0,0 +1,16 @@ +{ + "string": [ + { + "name": "app_name", + "value": "ContinuousTaskServer" + }, + { + "name": "mainability_description", + "value": "JS_Phone_Empty Feature Ability" + }, + { + "name": "description_serviceability", + "value": "hap sample empty service" + } + ] +} \ No newline at end of file diff --git a/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/resources/base/media/icon.png b/resourceschedule/resourceschedule_standard/continuoustaskrelyhap/entry/src/main/resources/base/media/icon.png new file mode 100755 index 0000000000000000000000000000000000000000..ce307a8827bd75456441ceb57d530e4c8d45d36c GIT binary patch literal 6790 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 YX9HpW+h1JFRc8v8e%3PML|^*<0J%6(*8l(j literal 0 HcmV?d00001 -- GitLab