diff --git a/examples/scripts/countdown.js b/examples/scripts/countdown.js index c0b9971f7..3c4f425bd 100644 --- a/examples/scripts/countdown.js +++ b/examples/scripts/countdown.js @@ -80,7 +80,7 @@ WoT.produce({ const listToDelete = []; for (const id of countdowns.keys()) { const as = countdowns.get(id); - if (as.output !== undefined) { + if (as !== undefined && as.output !== undefined) { const prev = as.output; as.output--; console.log("\t" + id + ", from " + prev + " to " + as.output); @@ -125,7 +125,7 @@ WoT.produce({ }; const ii = resp; console.log("init countdown value = " + JSON.stringify(resp)); - countdowns.set(resp.href, resp); + countdowns.set(resp.href !== undefined ? resp.href : "", resp); return ii; }); thing.setActionHandler("stopCountdown", async (params, options) => { @@ -133,10 +133,14 @@ WoT.produce({ const value = await params.value(); if (typeof value === "string" && countdowns.has(value)) { const as = countdowns.get(value); - as.output = 0; - as.status = Status.completed; - console.log("Countdown stopped for href: " + value); - return undefined; + if (as !== undefined) { + as.output = 0; + as.status = Status.completed; + console.log("Countdown stopped for href: " + value); + return null; + } else { + throw Error("Countdown value is undefined for href, " + value); + } } else { throw Error("Input provided for stopCountdown is no string or invalid href, " + value); } diff --git a/examples/scripts/counter-client.js b/examples/scripts/counter-client.js index a8cf994cf..d98973408 100644 --- a/examples/scripts/counter-client.js +++ b/examples/scripts/counter-client.js @@ -13,6 +13,19 @@ * SPDX-License-Identifier: EPL-2.0 OR W3C-20150513 ********************************************************************************/ +function getFormIndexForDecrementWithCoAP(thing) { + var _a; + const forms = (_a = thing.getThingDescription().actions) === null || _a === void 0 ? void 0 : _a.decrement.forms; + if (forms !== undefined) { + for (let i = 0; i < forms.length; i++) { + if (/^coaps?:\/\/.*/.test(forms[i].href)) { + return i; + } + } + } + // return formIndex: 0 if no CoAP target IRI found + return 0; +} WoTHelpers.fetch("coap://localhost:5683/counter") .then(async (td) => { // using await for serial execution (note 'async' in then() of fetch()) @@ -45,13 +58,3 @@ WoTHelpers.fetch("coap://localhost:5683/counter") .catch((err) => { console.error("Fetch error:", err); }); -function getFormIndexForDecrementWithCoAP(thing) { - const forms = thing.getThingDescription().actions.decrement.forms; - for (let i = 0; i < forms.length; i++) { - if (/^coaps?:\/\/.*/.test(forms[i].href)) { - return i; - } - } - // return formIndex: 0 if no CoAP target IRI found - return 0; -} diff --git a/examples/scripts/counter.js b/examples/scripts/counter.js index ea7828f39..19939def7 100644 --- a/examples/scripts/counter.js +++ b/examples/scripts/counter.js @@ -204,7 +204,7 @@ WoT.produce({ let fill = "black"; if (options && typeof options === "object" && "uriVariables" in options) { console.log("options = " + JSON.stringify(options)); - if ("fill" in options.uriVariables) { + if (options.uriVariables && "fill" in options.uriVariables) { const uriVariables = options.uriVariables; fill = uriVariables.fill; } @@ -229,7 +229,7 @@ WoT.produce({ let step = 1; if (options && typeof options === "object" && "uriVariables" in options) { console.log("options = " + JSON.stringify(options)); - if ("step" in options.uriVariables) { + if (options.uriVariables && "step" in options.uriVariables) { const uriVariables = options.uriVariables; step = uriVariables.step; } @@ -246,7 +246,7 @@ WoT.produce({ let step = 1; if (options && typeof options === "object" && "uriVariables" in options) { console.log("options = " + JSON.stringify(options)); - if ("step" in options.uriVariables) { + if (options.uriVariables && "step" in options.uriVariables) { const uriVariables = options.uriVariables; step = uriVariables.step; } diff --git a/examples/scripts/smart-coffee-machine-client.js b/examples/scripts/smart-coffee-machine-client.js index 8427eb2a9..9dba78129 100644 --- a/examples/scripts/smart-coffee-machine-client.js +++ b/examples/scripts/smart-coffee-machine-client.js @@ -15,6 +15,14 @@ // This is an example of Web of Things consumer ("client" mode) Thing script. // It considers a fictional smart coffee machine in order to demonstrate the capabilities of Web of Things. // An accompanying tutorial is available at http://www.thingweb.io/smart-coffee-machine.html. + +// Print data and an accompanying message in a distinguishable way +function log(msg, data) { + console.info("======================"); + console.info(msg); + console.dir(data); + console.info("======================"); +} WoTHelpers.fetch("http://127.0.0.1:8080/smart-coffee-machine").then(async (td) => { try { const thing = await WoT.consume(td); @@ -40,7 +48,7 @@ WoTHelpers.fetch("http://127.0.0.1:8080/smart-coffee-machine").then(async (td) = const makeCoffee = await thing.invokeAction("makeDrink", undefined, { uriVariables: { drinkId: "latte", size: "l", quantity: 3 }, }); - const makeCoffeep = await makeCoffee.value(); + const makeCoffeep = await (makeCoffee === null || makeCoffee === void 0 ? void 0 : makeCoffee.value()); if (makeCoffeep.result) { log("Enjoy your drink!", makeCoffeep); } else { @@ -57,7 +65,9 @@ WoTHelpers.fetch("http://127.0.0.1:8080/smart-coffee-machine").then(async (td) = time: "10:00", mode: "everyday", }); - const scheduledTaskp = await scheduledTask.value(); + const scheduledTaskp = await (scheduledTask === null || scheduledTask === void 0 + ? void 0 + : scheduledTask.value()); log(scheduledTaskp.message, scheduledTaskp); // See how it has been added to the schedules property const schedules = await (await thing.readProperty("schedules")).value(); @@ -74,10 +84,3 @@ WoTHelpers.fetch("http://127.0.0.1:8080/smart-coffee-machine").then(async (td) = console.error("Script error:", err); } }); -// Print data and an accompanying message in a distinguishable way -function log(msg, data) { - console.info("======================"); - console.info(msg); - console.dir(data); - console.info("======================"); -} diff --git a/examples/security/oauth/consumer.js b/examples/security/oauth/consumer.js index b6a6fd436..5726a0186 100644 --- a/examples/security/oauth/consumer.js +++ b/examples/security/oauth/consumer.js @@ -1,5 +1,5 @@ /******************************************************************************** - * Copyright (c) 2018 - 2020 Contributors to the Eclipse Foundation + * Copyright (c) 2023 Contributors to the Eclipse Foundation * * See the NOTICE file(s) distributed with this work for additional * information regarding copyright ownership. @@ -15,7 +15,8 @@ WoTHelpers.fetch("https://localhost:8080/oauth").then((td) => { WoT.consume(td).then(async (thing) => { try { - const result = await (await thing.invokeAction("sayOk")).value(); + const resp = await thing.invokeAction("sayOk"); + const result = resp === null || resp === void 0 ? void 0 : resp.value(); console.log("oAuth token was", result); } catch (error) { console.log("It seems that I couldn't access the resource"); diff --git a/examples/testthing/testclient.js b/examples/testthing/testclient.js index 8c75c1140..c0815cd3c 100644 --- a/examples/testthing/testclient.js +++ b/examples/testthing/testclient.js @@ -25,7 +25,7 @@ async function testPropertyRead(thing, name) { const value = await res.value(); console.info("PASS " + name + " READ:", value); } catch (err) { - console.error("FAIL " + name + " READ:", err.message); + console.error("FAIL " + name + " READ:", JSON.stringify(err)); } } async function testPropertyWrite(thing, name, value, shouldFail) { @@ -35,8 +35,8 @@ async function testPropertyWrite(thing, name, value, shouldFail) { if (!shouldFail) console.info("PASS " + name + " WRITE (" + displayValue + ")"); else console.error("FAIL " + name + " WRITE: (" + displayValue + ")"); } catch (err) { - if (!shouldFail) console.error("FAIL " + name + " WRITE (" + displayValue + "):", err.message); - else console.info("PASS " + name + " WRITE (" + displayValue + "):", err.message); + if (!shouldFail) console.error("FAIL " + name + " WRITE (" + displayValue + "):", JSON.stringify(err)); + else console.info("PASS " + name + " WRITE (" + displayValue + "):", JSON.stringify(err)); } } WoTHelpers.fetch("http://localhost:8080/testthing") diff --git a/packages/core/src/helpers.ts b/packages/core/src/helpers.ts index 621615b17..8718c643a 100644 --- a/packages/core/src/helpers.ts +++ b/packages/core/src/helpers.ts @@ -153,6 +153,7 @@ export default class Helpers implements Resolver { } // TODO: specialize fetch to retrieve just thing descriptions + // see https://github.com/eclipse-thingweb/node-wot/issues/1055 public fetch(uri: string): Promise { return new Promise((resolve, reject) => { const client = this.srv.getClientFor(Helpers.extractScheme(uri)); diff --git a/packages/examples/src/scripts/countdown.ts b/packages/examples/src/scripts/countdown.ts index 922880a2f..fa7c2e25c 100644 --- a/packages/examples/src/scripts/countdown.ts +++ b/packages/examples/src/scripts/countdown.ts @@ -13,8 +13,6 @@ * SPDX-License-Identifier: EPL-2.0 OR W3C-20150513 ********************************************************************************/ -import { InteractionOptions } from "wot-typescript-definitions"; - function uuidv4(): string { return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, function (c) { const r = (Math.random() * 16) | 0; @@ -91,8 +89,8 @@ WoT.produce({ console.log("Update countdowns"); const listToDelete: string[] = []; for (const id of countdowns.keys()) { - const as: ActionStatus = countdowns.get(id); - if (as.output !== undefined) { + const as = countdowns.get(id); + if (as?.output !== undefined) { const prev = as.output; as.output--; console.log("\t" + id + ", from " + prev + " to " + as.output); @@ -117,20 +115,17 @@ WoT.produce({ }, 1000); // set property handlers (using async-await) - thing.setPropertyReadHandler( - "countdowns", - async (options: InteractionOptions): Promise => { - const cts: string[] = []; - for (const id of countdowns.keys()) { - cts.push(id); - } - return cts; + thing.setPropertyReadHandler("countdowns", async (options): Promise => { + const cts: string[] = []; + for (const id of countdowns.keys()) { + cts.push(id); } - ); + return cts; + }); // set action handlers (using async-await) thing.setActionHandler( "startCountdown", - async (params: WoT.InteractionOutput, options: InteractionOptions): Promise => { + async (params: WoT.InteractionOutput, options): Promise => { let initValue = 100; if (params) { const value = await params.value(); @@ -145,21 +140,25 @@ WoT.produce({ }; const ii: WoT.InteractionInput = resp; console.log("init countdown value = " + JSON.stringify(resp)); - countdowns.set(resp.href, resp); + countdowns.set(resp.href ?? "", resp); return ii; } ); thing.setActionHandler( "stopCountdown", - async (params: WoT.InteractionOutput, options: InteractionOptions): Promise => { + async (params: WoT.InteractionOutput, options): Promise => { if (params) { const value = await params.value(); if (typeof value === "string" && countdowns.has(value)) { - const as: ActionStatus = countdowns.get(value); - as.output = 0; - as.status = Status.completed; - console.log("Countdown stopped for href: " + value); - return undefined; + const as = countdowns.get(value); + if (as !== undefined) { + as.output = 0; + as.status = Status.completed; + console.log("Countdown stopped for href: " + value); + return null; + } else { + throw Error("Countdown value is undefined for href, " + value); + } } else { throw Error("Input provided for stopCountdown is no string or invalid href, " + value); } @@ -170,11 +169,11 @@ WoT.produce({ ); thing.setActionHandler( "monitorCountdown", - async (params: WoT.InteractionOutput, options: InteractionOptions): Promise => { + async (params: WoT.InteractionOutput, options): Promise => { if (params) { const value = await params.value(); if (typeof value === "string" && countdowns.has(value)) { - const as: ActionStatus = countdowns.get(value); + const as = countdowns.get(value); return JSON.stringify(as); } else { throw Error("Input provided for monitorCountdown is no string or invalid href, " + value); diff --git a/packages/examples/src/scripts/counter-client.ts b/packages/examples/src/scripts/counter-client.ts index 376c72366..6389b4dcb 100644 --- a/packages/examples/src/scripts/counter-client.ts +++ b/packages/examples/src/scripts/counter-client.ts @@ -16,13 +16,15 @@ import { Helpers } from "@node-wot/core"; import { ThingDescription } from "wot-typescript-definitions"; -let WoTHelpers: Helpers; +let WoTHelpers!: Helpers; function getFormIndexForDecrementWithCoAP(thing: WoT.ConsumedThing): number { - const forms = thing.getThingDescription().actions.decrement.forms; - for (let i = 0; i < forms.length; i++) { - if (/^coaps?:\/\/.*/.test(forms[i].href)) { - return i; + const forms = thing.getThingDescription().actions?.decrement.forms; + if (forms !== undefined) { + for (let i = 0; i < forms.length; i++) { + if (/^coaps?:\/\/.*/.test(forms[i].href)) { + return i; + } } } // return formIndex: 0 if no CoAP target IRI found diff --git a/packages/examples/src/scripts/counter.ts b/packages/examples/src/scripts/counter.ts index 4c6dda51b..c70545bdc 100644 --- a/packages/examples/src/scripts/counter.ts +++ b/packages/examples/src/scripts/counter.ts @@ -208,7 +208,7 @@ WoT.produce({ let fill = "black"; if (options && typeof options === "object" && "uriVariables" in options) { console.log("options = " + JSON.stringify(options)); - if ("fill" in options.uriVariables) { + if (options.uriVariables && "fill" in options.uriVariables) { const uriVariables = options.uriVariables as Record; fill = uriVariables.fill; } @@ -234,7 +234,7 @@ WoT.produce({ let step = 1; if (options && typeof options === "object" && "uriVariables" in options) { console.log("options = " + JSON.stringify(options)); - if ("step" in options.uriVariables) { + if (options.uriVariables && "step" in options.uriVariables) { const uriVariables = options.uriVariables as Record; step = uriVariables.step as number; } @@ -251,7 +251,7 @@ WoT.produce({ let step = 1; if (options && typeof options === "object" && "uriVariables" in options) { console.log("options = " + JSON.stringify(options)); - if ("step" in options.uriVariables) { + if (options.uriVariables && "step" in options.uriVariables) { const uriVariables = options.uriVariables as Record; step = uriVariables.step as number; } diff --git a/packages/examples/src/scripts/smart-coffee-machine-client.ts b/packages/examples/src/scripts/smart-coffee-machine-client.ts index b365146cc..05e745c35 100644 --- a/packages/examples/src/scripts/smart-coffee-machine-client.ts +++ b/packages/examples/src/scripts/smart-coffee-machine-client.ts @@ -19,7 +19,7 @@ import { ThingDescription } from "wot-typescript-definitions"; import { Helpers } from "@node-wot/core"; -let WoTHelpers: Helpers; +let WoTHelpers!: Helpers; // Print data and an accompanying message in a distinguishable way function log(msg: string, data: unknown) { @@ -60,7 +60,7 @@ WoTHelpers.fetch("http://127.0.0.1:8080/smart-coffee-machine").then(async (td) = const makeCoffee = await thing.invokeAction("makeDrink", undefined, { uriVariables: { drinkId: "latte", size: "l", quantity: 3 }, }); - const makeCoffeep = (await makeCoffee.value()) as Record; + const makeCoffeep = (await makeCoffee?.value()) as Record; if (makeCoffeep.result) { log("Enjoy your drink!", makeCoffeep); } else { @@ -79,7 +79,7 @@ WoTHelpers.fetch("http://127.0.0.1:8080/smart-coffee-machine").then(async (td) = time: "10:00", mode: "everyday", }); - const scheduledTaskp = (await scheduledTask.value()) as Record; + const scheduledTaskp = (await scheduledTask?.value()) as Record; log(scheduledTaskp.message, scheduledTaskp); // See how it has been added to the schedules property diff --git a/packages/examples/src/security/oauth/consumer.ts b/packages/examples/src/security/oauth/consumer.ts index 8288b6622..ccb16fb29 100644 --- a/packages/examples/src/security/oauth/consumer.ts +++ b/packages/examples/src/security/oauth/consumer.ts @@ -15,12 +15,13 @@ import { Helpers } from "@node-wot/core"; import { ThingDescription } from "wot-typescript-definitions"; -let WoTHelpers: Helpers; +let WoTHelpers!: Helpers; WoTHelpers.fetch("https://localhost:8080/oauth").then((td) => { WoT.consume(td as ThingDescription).then(async (thing) => { try { - const result = await (await thing.invokeAction("sayOk")).value(); + const resp = await thing.invokeAction("sayOk"); + const result = resp?.value(); console.log("oAuth token was", result); } catch (error) { console.log("It seems that I couldn't access the resource"); diff --git a/packages/examples/src/testthing/testclient.ts b/packages/examples/src/testthing/testclient.ts index 3abab1967..ad2315975 100644 --- a/packages/examples/src/testthing/testclient.ts +++ b/packages/examples/src/testthing/testclient.ts @@ -15,7 +15,7 @@ import { Helpers } from "@node-wot/core"; import { ThingDescription } from "wot-typescript-definitions"; -let WoTHelpers: Helpers; +let WoTHelpers!: Helpers; console.log = () => { /* empty */ @@ -30,7 +30,7 @@ async function testPropertyRead(thing: WoT.ConsumedThing, name: string) { const value = await res.value(); console.info("PASS " + name + " READ:", value); } catch (err) { - console.error("FAIL " + name + " READ:", err.message); + console.error("FAIL " + name + " READ:", JSON.stringify(err)); } } @@ -46,8 +46,8 @@ async function testPropertyWrite( if (!shouldFail) console.info("PASS " + name + " WRITE (" + displayValue + ")"); else console.error("FAIL " + name + " WRITE: (" + displayValue + ")"); } catch (err) { - if (!shouldFail) console.error("FAIL " + name + " WRITE (" + displayValue + "):", err.message); - else console.info("PASS " + name + " WRITE (" + displayValue + "):", err.message); + if (!shouldFail) console.error("FAIL " + name + " WRITE (" + displayValue + "):", JSON.stringify(err)); + else console.info("PASS " + name + " WRITE (" + displayValue + "):", JSON.stringify(err)); } } diff --git a/packages/examples/tsconfig.json b/packages/examples/tsconfig.json index bfabb0e56..61b6c09bc 100644 --- a/packages/examples/tsconfig.json +++ b/packages/examples/tsconfig.json @@ -1,11 +1,10 @@ { "extends": "../../tsconfig.json", "compilerOptions": { + "strict": true, "outDir": "dist", "rootDir": "src", "target": "ES2018", - "alwaysStrict": false, - "noImplicitUseStrict": true, "sourceMap": false, "removeComments": false },