1.1.5.5. カスタムウィジェット・microserviceコード例を確認する

本項では、以下の実装において必要なカスタムウィジェット・microserviceのコード例を記載しています。

  • 実装2: IoT機器からの画像送信・Things Cloudでの画像表示

なお、本書中の設定値の「< >」の表記については、ご利用の環境により各自入力いただく箇所となります("<"から">"までを設定値に置き換えてください)。

注釈

  • 本項では例として、開発用ディレクトリを「C:\tmp」としています。お使いの環境に合わせて読み替えてください。

microservice用コードを確認する

本実装のためのmicroservice作成用コードをまとめています。

注釈

  • 本項では、標準提供されていない「microservice」機能を利用しています。
    「microservice」のご利用にあたっては、提供環境や費用などの諸条件について個別対応とさせていただいております。
    「microservice」機能に関しては iot-app@ntt.com までご相談ください。
    案件相談、提供環境や費用などの諸条件に関しては iot-info@ntt.com までご相談ください。
  • 本項では開発用ディレクトリを「C:\tmp\microservice」としています。お使いの環境に合わせて読み替えてください。

「app.ts」

C:\tmp\microservice\app.ts

"use strict";

require("dotenv").config();

import * as express from "express";
import { routes } from "./routes";

const app: express.Express = express();

// Application endpoints
routes(app);

// Server listening on port 80
app.use(express.json());
app.listen(80);
console.log(`node-microservice started on port 80`);

「routes.ts」

C:\tmp\microservice\routes.ts

"use strict";

import * as express from "express";
import * as AWS from "aws-sdk";
import { FetchClient, IFetchOptions, IAuthentication } from "@c8y/client";

export function routes(app: express.Express): any {
const baseUrl = process.env.C8Y_BASEURL;
        // Temporary Credentialsの取得
        app.route("/tempcred").get((req, res) => {
                const authorization = req.get("Authorization");
                const cookie = req.get("Cookie");
                const headers = { authorization, cookie };
                const auth = new MicroserviceClientRequestAuth(headers);
                const fetchClient = new FetchClient(auth, baseUrl);

                (async () => {
                        try {
                                const options: IFetchOptions = {
                                        method: "GET",
                                        headers: { "Content-Type": "application/json" },
                                };
                                //keystore APIを叩く
                                const resp = await fetchClient.fetch(`/inventory/managedObjects/${req.query.keystoreid}`, options);
                                //結果(wasabi鍵情報)をcredsに入れる
                                const creds = await resp.json();
                                //AWS STSインスタンスを生成
                                const sts = new AWS.STS({
                                        apiVersion: "2011-06-15",
                                        endpoint: new AWS.Endpoint("wasabisys.com"),
                                        accessKeyId: creds.com_WasabiCredentials.accessKey,
                                        secretAccessKey: creds.com_WasabiCredentials.secretKey,
                                        region: "us-east-1",
                                });
                                //Temporary Credentialsを取得し返す
                                sts.getSessionToken({ DurationSeconds: 900 }, (err, data) => {
                                        if (err) {
                                                res.json(err);
                                        } else {
                                                data.Credentials["region"] = creds.com_WasabiCredentials.region;
                                                data.Credentials["bucket"] = creds.com_WasabiCredentials.bucket;
                                                console.log(data.Credentials);
                                                res.json(data.Credentials);
                                        }
                                });
                        } catch (err) {
                                res.json(err);
                        }
                })();
        });
}

/**
* Allows to use either Cookie-Auth or Basic-Auth
* of a microservice client request header
* for Authorization to the Cumulocity API.
*/
class MicroserviceClientRequestAuth implements IAuthentication {
        xsrfToken: any;
        authTokenFromCookie: any;
        authTokenFromHeader: any;

/**
* Authenticates using the credentials which were
* provided within the request headers of the
* client call to the microservice.
* @param headers
*/
        constructor(headers: any = {}) {
                this.xsrfToken = this.getCookieValue(headers, "XSRF-TOKEN");
                this.authTokenFromCookie = this.getCookieValue(headers, "authorization");
                this.authTokenFromHeader = headers.authorization;
        }
        updateCredentials(credentials = {}) {
                return undefined;
        }
        getFetchOptions(options) {
                const headers = Object.assign(
                        { Authorization: this.authTokenFromCookie ? `Bearer ${this.authTokenFromCookie}` : this.
                        authTokenFromHeader },
                        this.xsrfToken ? { "X-XSRF-TOKEN": this.xsrfToken } : undefined
                );
                options.headers = Object.assign(headers, options.headers);
                return options;
        }
        getCometdHandshake(config: any = {}) {
                const KEY = "com.cumulocity.authn";
                const xsrfToken = this.xsrfToken;
                let token = this.authTokenFromCookie;
                if (!token && this.authTokenFromHeader) {
                        token = this.authTokenFromHeader.replace("Basic ", "").replace("Bearer ", "");
                }
                const ext = (config.ext = config.ext || {});
                ext[KEY] = Object.assign(ext[KEY] || {}, Object.assign({ token }, xsrfToken ? { xsrfToken } : undefined));
                return config;
        }
        logout() {
                if (this.authTokenFromCookie) {
                        delete this.authTokenFromCookie;
                }
                if (this.authTokenFromHeader) {
                        delete this.authTokenFromHeader;
                }
                if (this.xsrfToken) {
                        delete this.xsrfToken;
                }
        }
        getCookieValue(headers, name) {
                try {
                        const value = headers && headers.cookie && headers.cookie.match("(^|;)\\s*" + name + "\\s*=\\s*([^;]+)");
                        return value ? value.pop() : undefined;
                } catch (ex) {
                        return undefined;
                }
        }
}

「package.json」

C:\tmp\microservice\package.json

{
        "name": "wasabi-img",
        "version": "1.0.0",
        "description": "Things Cloud microservice application",
        "main": "app.js",
        "dependencies": {
                "@c8y/client": "^1004.6.15",
                "@types/express": "^4.17.3",
                "@types/node": "^13.9.1",
                "@types/source-map-support": "^0.5.1",
                "aws-sdk": "^2.1220.0",
                "dotenv": "^8.1.0",
                "express": "^4.17.0",
                "source-map-support": "^0.5.16"
        },
        "scripts": {
                "start": "node app.js",
                "build": "tsc -p .",
                "build:watch": "tsc -p . -w",
                "microservice2": "rm wasabi-img.tar && rm wasabi-img.zip && tsc -p . && docker rmi -f wasabi-img && docker build -t wasabi-img . && docker save -o wasabi-img.tar wasabi-img && jar cvf wasabi-img.zip cumulocity.json wasabi-img.tar",
                "microservice": "tsc -p . && docker rmi -f wasabi-img && docker build -t wasabi-img . && docker save -o wasabi-img.tar wasabi-img && jar cvf wasabi-img.zip cumulocity.json wasabi-img.tar",
                "test": "echo \"Error: no test specified\" && exit 1"
        },
        "author": "Your name",
        "license": "MIT"
}

「tsconfig.json」

C:\tmp\microservice\tsconfig.json

{
        "compilerOptions": {
                "target": "es5",
                "module": "commonjs",
                "sourceMap": true,
                "forceConsistentCasingInFileNames": true
        },
        "exclude": ["node_modules"]
}

「Dockerfile」

C:\tmp\microservice\Dockerfile

FROM node:alpine

WORKDIR /usr/app

COPY ./package.json ./
RUN npm install
COPY ./*.js ./

CMD ["npm", "start"]

「cumulocity.json」

C:\tmp\microservice\cumulocity.json

{
        "apiVersion": "v1",
        "version": "1.0.0-SNAPSHOT",
        "contextPath": "wasabi-img",
        "provider": {
                "name": "NTT Communications"
        },
        "isolation": "MULTI_TENANT",
        "requiredRoles": [],
        "roles": []
}

カスタムウィジェット用コードを確認する

本実装のためのカスタムウィジェット作成用コードをまとめています。

注釈

  • 本項では開発用ディレクトリを「C:\tmp\event-widget」としています。お使いの環境に合わせて読み替えてください。

「package.json」

C:\tmp\event-widget\package.json

{
  "name": "cockpit",
  "version": "0.1.0",
  "description": "[TEST]View Image from wasabi",
  "main": "index.js",
  "scripts": {
    "start": "c8ycli server -u https://<Things Cloudのテナント名>.je1.thingscloud.ntt.com",
    "build": "c8ycli build",
    "deploy": "c8ycli deploy",
    "test": "karma start karma.conf.js"
  },
  "keywords": [],
  "author": "",
  "license": "ISC",
  "dependencies": {
    "@angular/animations": "8.2.13",
    "@angular/cdk": "8.2.3",
    "@angular/common": "8.2.13",
    "@angular/compiler": "8.2.13",
    "@angular/core": "^8.2.13",
    "@angular/forms": "8.2.13",
    "@angular/platform-browser": "8.2.13",
    "@angular/platform-browser-dynamic": "8.2.13",
    "@angular/router": "8.2.13",
    "@angular/upgrade": "8.2.13",
    "@aws-sdk/client-s3": "^3.347.1",
    "@c8y/ng1-modules": "1006.6.32",
    "@c8y/ngx-components": "1006.6.32",
    "@c8y/style": "1006.6.32",
    "@observablehq/plot": "^0.3.2",
    "@types/dom-mediacapture-record": "^1.0.7",
    "angular": "1.6.9",
    "aws-sdk": "^2.1392.0",
    "core-js": "^3.4.0",
    "cumulocity-ui-build": "http://resources.cumulocity.com/webapps/ui/9.20.16.tar.gz",
    "hls": "0.0.1",
    "hls.js": "^0.14.16",
    "jquery": "^3.5.1",
    "rxjs": "~6.4.0",
    "url-search-params-polyfill": "6.0.0",
    "zone.js": "~0.9.1"
  },
  "devDependencies": {
    "@angular-devkit/build-angular": "0.803.23",
    "@angular/compiler-cli": "8.2.13",
    "@angular/language-service": "8.2.13",
    "@c8y/cli": "1006.6.32",
    "@types/jasmine": "^3.5.7",
    "bootstrap": "^4.5.0",
    "css-loader": "^3.4.2",
    "html-loader": "^1.1.0",
    "jasmine": "^3.6.1",
    "jasmine-core": "^3.6.0",
    "karma": "^5.1.0",
    "karma-chrome-launcher": "^3.1.0",
    "karma-jasmine": "^3.3.1",
    "karma-jasmine-html-reporter": "^1.5.4",
    "karma-safari-launcher": "^1.0.0",
    "karma-sourcemap-loader": "^0.3.7",
    "karma-webpack": "^4.0.2",
    "mockdate": "^3.0.5",
    "to-string-loader": "^1.1.6",
    "ts-loader": "^8.0.1",
    "typescript": "3.5.3",
    "webpack-dev-server": "^3.11.2"
  },
  "c8y": {
    "application": {
      "name": "cockpit",
      "contextPath": "cockpit",
      "key": "cockpit-application-key",
      "brandingCssVars": {
        "brand-primary": "#1357ad",
        "brand-complementary": "#a4c8cd",
        "brand-dark": "#0e407f",
        "brand-light": "#94aac5",
        "gray-text": "#444",
        "link-color": "#1357ad",
        "link-hover-color": "#0b3468",
        "body-background-color": "#f8f8f8",
        "brand-logo-img": "url(/apps/public/ui-assets-ntt/logo.svg?nocache=8324918234)",
        "brand-logo-img-height": "25%",
        "navigator-platform-logo": "url(/apps/public/ui-assets-ntt/tenant-logo.svg?nocache=8324918234)",
        "navigator-platform-logo-height": "23%",
        "navigator-font-family": "'Frutiger Neue', 'Helvetica Neue', Helvetica, Arial, sans-serif",
        "navigator-app-name-size": "16px",
        "navigator-app-icon-size": "0",
        "navigator-bg-color": "#1357ad",
        "navigator-header-bg": "white",
        "navigator-title-color": "#1357ad",
        "navigator-text-color": "rgba(255, 255, 255, 0.7)",
        "navigator-separator-color": "rgba(255, 255, 255, 0.05)",
        "navigator-active-color": "#1357ad",
        "navigator-active-bg": "#fdd34c",
        "header-color": "white",
        "header-text-color": "#1357ad",
        "header-hover-color": "var(--brand-primary, #1357ad)",
        "header-border-color": "rgba(117, 117, 117, 0.05)",
        "font-family-base": "'Frutiger Neue', 'Helvetica Neue', Helvetica, Arial, sans-serif",
        "headings-font-family": "'Frutiger Neue', 'Helvetica Neue', Helvetica, Arial, sans-serif"
      },
      "extraCssUrls": [
        "/apps/public/ui-assets-ntt/style.css"
      ],
      "faviconUrl": "/apps/public/ui-assets-ntt/favicon.ico?nocache=08563753891373449",
      "globalTitle": "NTT",
      "docsBaseUrl": "https://developer.ntt.com/iot/docs",
      "tabsHorizontal": true,
      "upgrade": true,
      "rightDrawer": false,
      "sensorPhone": false,
      "contentSecurityPolicy": "base-uri 'self'; default-src 'self' 'unsafe-inline' http: https: ws: wss:; connect-src 'self' *.billwerk.com http: https: ws: wss: blob:;  script-src 'self' open.mapquestapi.com *.twitter.com *.twimg.com swc.safie.link 'unsafe-inline' 'unsafe-eval' data:; style-src * 'unsafe-inline' blob:; img-src * data: blob:; font-src * data:; frame-src *; media-src * data: blob:;"
    },
    "cli": {}
  }
}

「app.module.ts」

C:\tmp\event-widget\app.module.ts

import { NgModule, CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';
import { BrowserAnimationsModule } from '@angular/platform-browser/animations';
import { RouterModule as NgRouterModule } from '@angular/router';
import { UpgradeModule as NgUpgradeModule } from '@angular/upgrade/static';
import { HttpClientModule } from '@angular/common/http';

import { BsDropdownModule } from 'ngx-bootstrap/dropdown';
import { CollapseModule } from 'ngx-bootstrap/collapse';
import { BsDatepickerModule } from 'ngx-bootstrap/datepicker';
import { TimepickerModule } from 'ngx-bootstrap/timepicker';

import { CoreModule, RouterModule } from '@c8y/ngx-components';
import {
        DashboardUpgradeModule,
        UpgradeModule,
        HybridAppModule,
        UPGRADE_ROUTES
} from '@c8y/ngx-components/upgrade';
import { AssetsNavigatorModule } from '@c8y/ngx-components/assets-navigator';
import { CockpitDashboardModule } from '@c8y/ngx-components/context-dashboard';
import { ReportsModule } from '@c8y/ngx-components/reports';
import { SensorPhoneModule } from '@c8y/ngx-components/sensor-phone';
import { BinaryFileDownloadModule } from '@c8y/ngx-components/binary-file-download';

import { S3Module } from './custom-widgets/s3.module';

import { C8yService } from './custom-widgets/c8y.service';

@NgModule({
        imports: [
                // Upgrade module must be the first
                UpgradeModule,
                BrowserAnimationsModule,

                BsDropdownModule.forRoot(),
                CollapseModule.forRoot(),
                BsDatepickerModule.forRoot(),
                TimepickerModule.forRoot(),

                RouterModule.forRoot(),
                NgRouterModule.forRoot([...UPGRADE_ROUTES], { enableTracing: false, useHash: true }),
                CoreModule.forRoot(),
                AssetsNavigatorModule,
                ReportsModule,
                NgUpgradeModule,
                DashboardUpgradeModule,
                CockpitDashboardModule,
                SensorPhoneModule,
                BinaryFileDownloadModule,

                HttpClientModule,
                S3Module,
        ],
        declarations: [
        ],
        entryComponents: [
        ],
        providers: [
                C8yService,
        ],
        schemas: [
                CUSTOM_ELEMENTS_SCHEMA
        ]
})
export class AppModule extends HybridAppModule {
        constructor(protected upgrade: NgUpgradeModule) {
                super();
        }
}

「index.ts」

C:\tmp\event-widget\index.ts

import './polyfills';
import './ng1';

import { enableProdMode } from '@angular/core';
import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
import { AppModule } from './app.module';

declare const __MODE__: string;
if (__MODE__ === 'production') {
  enableProdMode();
}

export function bootstrap() {
  return platformBrowserDynamic()
    .bootstrapModule(AppModule).catch((err) => console.log(err));
}

「karma.conf.js」

C:\tmp\event-widget\karma.conf.js

// Karma configuration
// Generated on Mon Jul 27 2020 10:38:49 GMT+0900 (日本標準時)

module.exports = function(config) {
  config.set({

    // base path that will be used to resolve all patterns (eg. files, exclude)
    basePath: './custom-widgets',


    // frameworks to use
    // available frameworks: https://npmjs.org/browse/keyword/karma-adapter
    frameworks: ['jasmine'],


    plugins: [  // 追記箇所
      require("karma-jasmine"),
      require("karma-chrome-launcher"),
      require("karma-safari-launcher"),
      require("karma-jasmine-html-reporter"),
      require("karma-webpack"),
      require("karma-sourcemap-loader")
    ],

    client:{
      clearContext: false
    },

    // list of files / patterns to load in the browser
    files: [
      "../node_modules/zone.js/dist/zone.js"
    ],


    // list of files / patterns to exclude
    exclude: [
    ],


    // preprocess matching files before serving them to the browser
    // available preprocessors: https://npmjs.org/browse/keyword/karma-preprocessor
    preprocessors: {

    },

    webpack: { // 追記箇所: webpack の設定
      resolve: {
        extensions: [".ts", ".js"]
      },
      module: {
        rules: [
          { test: /\.ts$/, use: [{ loader: "ts-loader" }] },
          { test: /\.html$/, use: [{ loader: "html-loader" }] }, // 外部htmlファイルの読み込み用
          { test: /\.css$/, loaders: ["to-string-loader", "style-loader", "css-loader"] } // 外部cssファイルの読み込み用
        ]
      },
      mode: "development",
      devtool: 'inline-source-map'
    },



    // test results reporter to use
    // possible values: 'dots', 'progress'
    // available reporters: https://npmjs.org/browse/keyword/karma-reporter
    reporters: ['progress', 'kjhtml'],


    // web server port
    port: 9876,


    // enable / disable colors in the output (reporters and logs)
    colors: true,


    // level of logging
    // possible values: config.LOG_DISABLE || config.LOG_ERROR || config.LOG_WARN || config.LOG_INFO || config.LOG_DEBUG
    logLevel: config.LOG_INFO,


    // enable / disable watching file and executing tests whenever any file changes
    autoWatch: true,


    // start these browsers
    // available browser launchers: https://npmjs.org/browse/keyword/karma-launcher
    browsers: ['Chrome'],


    // Continuous Integration mode
    // if true, Karma captures browsers, runs the tests and exits
    singleRun: false,

    // Concurrency level
    // how many browser should be started simultaneous
    concurrency: Infinity,

    // timeout
    browserNoActivityTimeout: 60000
  })
}

「ng1.ts」

C:\tmp\event-widget\ng1.ts

import '@c8y/ng1-modules/core';
import '@c8y/ng1-modules/imageWidget/cumulocity.json';
import '@c8y/ng1-modules/assetPropertyWidget/cumulocity.json';
import '@c8y/ng1-modules/devicePropertyWidget/cumulocity.json';
import '@c8y/ng1-modules/alarms/cumulocity.json';
import '@c8y/ng1-modules/assetTable/cumulocity.json';
import '@c8y/ng1-modules/eventsBinary/cumulocity.json';
import '@c8y/ng1-modules/devicemanagement-alarmList/cumulocity.json';
import '@c8y/ng1-modules/deviceSelector/cumulocity.json';
import '@c8y/ng1-modules/kpi/cumulocity.json';
import '@c8y/ng1-modules/kpiAdmin/cumulocity.json';
import '@c8y/ng1-modules/devicemanagement-location/cumulocity.json';
import '@c8y/ng1-modules/dashboard2/cumulocity.json';
import '@c8y/ng1-modules/dashboardUI/cumulocity.json';
import '@c8y/ng1-modules/groupsHierarchy/cumulocity.json';
import '@c8y/ng1-modules/measurements/cumulocity.json';
import '@c8y/ng1-modules/map/cumulocity.json';
import '@c8y/ng1-modules/alarmAssets/cumulocity.json';
import '@c8y/ng1-modules/deviceControlMessage/cumulocity.json';
import '@c8y/ng1-modules/deviceControlRelay/cumulocity.json';
import '@c8y/ng1-modules/deviceControlRelayArray/cumulocity.json';
import '@c8y/ng1-modules/cockpit-cockpitConfig/cumulocity.json';
import '@c8y/ng1-modules/cockpit-assetCount/cumulocity.json';
import '@c8y/ng1-modules/cockpit-alarmRecent/cumulocity.json';
import '@c8y/ng1-modules/cockpit-reports/cumulocity.json';
import '@c8y/ng1-modules/cockpit-smartRulesUI/cumulocity.json';
import '@c8y/ng1-modules/cockpit-dataPointExplorerUI/cumulocity.json';
import '@c8y/ng1-modules/alarmsEventsExplorer/cumulocity.json';
import '@c8y/ng1-modules/deviceDatabase4/cumulocity.json';
import '@c8y/ng1-modules/modbusWidget4/cumulocity.json';
import '@c8y/ng1-modules/scada/cumulocity.json';
import '@c8y/ng1-modules/imageWidget/cumulocity.json';
import '@c8y/ng1-modules/htmlWidget/cumulocity.json';
import '@c8y/ng1-modules/applicationLinks/cumulocity.json';
import '@c8y/ng1-modules/quickLinks/cumulocity.json';
import '@c8y/ng1-modules/helpAndSupport/cumulocity.json';
import '@c8y/ng1-modules/eventList/cumulocity.json';
import '@c8y/ng1-modules/export/cumulocity.json';
import '@c8y/ng1-modules/dataPointTable/cumulocity.json';
import '@c8y/ng1-modules/switchDisplay/cumulocity.json';
import '@c8y/ng1-modules/trafficLightWidget/cumulocity.json';
import '@c8y/ng1-modules/infoGauge/cumulocity.json';

「polyfills.ts」

C:\tmp\event-widget\polyfills.ts

/**
 * This file includes polyfills needed by Angular and is loaded before the app.
 * You can add your own extra polyfills to this file.
 *
 * This file is divided into 2 sections:
 *   1. Browser polyfills. These are applied before loading ZoneJS and are sorted by browsers.
 *   2. Application imports. Files imported after ZoneJS that should be loaded before your main
 *      file.
 *
 * The current setup is for so-called "evergreen" browsers; the last versions of browsers that
 * automatically update themselves. This includes Safari >= 10, Chrome >= 55 (including Opera),
 * Edge >= 13 on the desktop, and iOS 10 and Chrome on mobile.
 *
 * Learn more in https://angular.io/docs/ts/latest/guide/browser-support.html
 */

/***************************************************************************************************
 * BROWSER POLYFILLS
 */

/** IE9, IE10, IE11, Evergreen browsers require the following polyfills. */
import "@angular-devkit/build-angular/src/angular-cli-files/models/es5-jit-polyfills.js";
import "@angular-devkit/build-angular/src/angular-cli-files/models/es5-polyfills.js";
import "@angular-devkit/build-angular/src/angular-cli-files/models/jit-polyfills.js";

/**
 * By default, zone.js will patch all possible macroTask and DomEvents
 * user can disable parts of macroTask/DomEvents patch by setting following flags
 */

(window as any).__Zone_disable_requestAnimationFrame = true; // disable patch requestAnimationFrame
// (window as any).__Zone_disable_on_property = true; // disable patch onProperty such as onclick
(window as any).__zone_symbol__BLACK_LISTED_EVENTS = [
  "scroll",
  "mousemove",
  "message",
  "mouseover",
  "mouseout",
  "mouseenter",
  "mouseleave",
]; // disable patch specified eventNames

/***************************************************************************************************
 * Zone JS is required by default for Angular itself.
 */
import "zone.js/dist/zone"; // Included with Angular CLI.

/***************************************************************************************************
 * APPLICATION IMPORTS
 */
import "url-search-params-polyfill";
import { addPolyfills } from "@c8y/ngx-components/polyfills";
addPolyfills();

「tsconfig.json」

C:\tmp\event-widget\tsconfig.json

{
  "compileOnSave": false,
  "compilerOptions": {
    "baseUrl": "./",
    "outDir": "./dist/out-tsc",
    "sourceMap": true,
    "declaration": false,
    "moduleResolution": "node",
    "emitDecoratorMetadata": true,
    "experimentalDecorators": true,
    "target": "es5",
    "typeRoots": [
      "node_modules/@types"
    ],
    "lib": [
      "dom",
      "es2016"
    ],
  }
}

「custom-widgets\image-widget\image-widget.component.css」

C:\tmp\event-widget\custom-widgets\image-widget\image-widget.component.css

.video-container {
        text-align: center;
}

.sticky-container {
        position: sticky;
        top: 0px;
        background-color: #ffffff;
        z-index: 100;
}

.video {
        width: 95%;
        margin: auto;
}
i.fa-play-circle-o {
        font-size: 30px;
}
.play-btn {
        color: #3f4a50;
        cursor: pointer;
}
.play-btn {
        color: #75bcf1;
}
.timeline-list>div.active .timeline-item-content .list-item-actions:before, .timeline-list>li.active .timeline-item-content .list-item-actions:before {
        display: none !important;
}

「custom-widgets\image-widget\image-widget.component.html」

C:\tmp\event-widget\custom-widgets\image-widget\image-widget.component.html

<div class="row" style="margin: 8px">
        <div class="col-xs-8 col-sm-7 sticky-container">
                <div class="card-header sticky-container">
                        <h4 class="card-title text-primary">撮影画像</h4>
                </div>
                <div class="image-container">
                        <img #img width="100%" />
                </div>
        </div>
        <div class="col-xs-4 col-sm-5">
                <div class="card">
                        <div class="card-header sticky-container">
                                <h4 class="card-title text-primary">イベント一覧</h4>
                        </div>
                        <div class="card-inner-scroll">
                                <ul class="timeline-list">
                                        <li class="timeline-list-item flex-row" [class.active]="event.active" *ngFor="let event of events | slice: 0:listNumber">
                                                <small class="timeline-item-date text-right">
                                                        <i c8y-icon="clock-o" class="fa fw fa-clock-o"></i>
                                                        <strong>{{ event.time | date: "yy/MM/dd" }}</strong>
                                                        <br />
                                                        <span class="text-muted" title="このイベントの最終発生日時">{{ event.time | date: "shortTime" }}</span>
                                                </small>
                                                <div class="timeline-item-content flex-row wrap play-btn" (click)="onPlayClick(event)" (click)="event.isCollapsed = !event.isCollapsed">
                                                        <div class="list-item-actions">
                                                                <button type="button" title="Expand" class="collapse-btn">
                                                                        <i c8yIcon="chevron-down"></i>
                                                                </button>
                                                        </div>
                                                        <div class="list-item-body">
                                                                <span>
                                                                        {{ event.text }}<br />
                                                                        <br class="visible-xs visible-sm" />
                                                                        <small>
                                                                                <a (click)="$event.stopPropagation()" class="text-muted" href="#/device/{{ event.source.id }}"> <i c8yIcon="cog"></i>{{ event.source.name }} </a>
                                                                        </small>
                                                                </span>
                                                        </div>
                                                        <div class="collapse no-cursor m-l-40" [collapse]="!event.isCollapsed" [isAnimated]="true" (click)="$event.stopPropagation()">
                                                                <div>
                                                                        <div class="legend form-block" translate>詳細</div>
                                                                        <dl class="dl-inline small text-muted">
                                                                                <dt>タイプ</dt>
                                                                                <dd>{{ event.type }}</dd>
                                                                        </dl>
                                                                        <dl class="dl-inline small text-muted">
                                                                                <dt translate>情報</dt>
                                                                                <dd>{{ event.com_ImageDownloadable | json }}</dd>
                                                                        </dl>
                                                                        <dl class="dl-inline small text-muted">
                                                                                <dt translate>サーバー時間</dt>
                                                                                <dd>{{ event.creationTime | date: "yy/MM/dd HH:mm:ss" }}</dd>
                                                                        </dl>
                                                                </div>
                                                        </div>
                                                </div>
                                        </li>
                                </ul>
                        </div>
                        <button id="more" type="button" class="btn btn-primary btn-lg btn-block" (click)="onMoreClick()"><i class="fa fw fa-repeat"></i>&nbsp;Load more...</button>
                </div>
        </div>
</div>

「custom-widgets\image-widget\image-widget.component.ts」

C:\tmp\event-widget\custom-widgets\image-widget\image-widget.component.ts

import { Component, Input, OnInit, ViewChild, ElementRef } from "@angular/core";
import { IEvent, IResultList } from "@c8y/client";
import { ImageService } from "../image.service";
import { C8yService, EventFilter } from "../c8y.service";

@Component({
        selector: "image-widget",
        templateUrl: "./image-widget.component.html",
        styleUrls: ["./image-widget.component.css"],
})
export class ImageWidget implements OnInit {
        private static EVENT_PAGE_SIZE = 50; //取得するイベントリスト数
        listNumber = 3; //表示するイベント数

        @Input() config;
        @ViewChild("img", { static: false }) private recordedVideo: ElementRef;
        events: IEvent[];
        eventList: Promise<IResultList<IEvent>>;
        resultList: IResultList<IEvent>;
        activeEvent: IEvent;

        constructor(private imageService: ImageService, private c8yService: C8yService) {}

        ngOnInit(): void {
                const realtime = this.c8yService.getRealtime();

                this.activeEvent = null;
                this.events = [];
                const filter: EventFilter = {
                        dateFrom: "2020-01-01",
                        dateTo: "2024-12-31",
                        revert: false,
                        pageSize: ImageWidget.EVENT_PAGE_SIZE,
                };

                this.eventList = this.c8yService.getEvents({
                        deviceId: this.config.device.id,
                        filter: filter,
                });
                this.eventList.then((resultList) => this.updateResultList(resultList));

                realtime.subscribe(`/events/${this.config.device.id}`, (data) => {
                        if (this.containsEventId(this.events, data.data.data)) return;
                        this.updateEvents([data.data.data]);
                });
        }

        /**
         * targetEvent が過去のイベントと重複しているか判定する
         *
         * @param {IEvent[]} events イベントの配列
         * @param {IEvent} targetEvent ターゲットイベントの配列
         *
         * @returns true(重複している) or false(重複していない)
         */
        containsEventId(events: IEvent[], targetEvent: IEvent): boolean {
                for (const event of events) {
                        if (event.id === targetEvent.id) return true;
                }
                return false;
        }

        /**
         * 再生ボタンクリック時、Wasabi からオブジェクトをダウンロードする
         *
         * @param {IEvent} event Event
         */
        onPlayClick(event: IEvent): void {
                if (this.activeEvent) this.activeEvent.active = false;
                event.active = true;
                this.activeEvent = event;

                this.imageService.getS3Object$(event.com_ImageDownloadable).subscribe((data) => {
                        const recordedVideoUrl = URL.createObjectURL(new Blob([data.Body], { type: "image/jpeg" }));

                        this.recordedVideo.nativeElement.src = recordedVideoUrl;
                        console.debug(data.Body.length);
                });
        }

        /**
         * Load more クリック時に過去のアラームを読み込む
         */
        onMoreClick(): void {
                this.listNumber += 3;
        }

        private updateResultList(resultList: IResultList<IEvent>): void {
                this.resultList = resultList;
                this.updateEvents(resultList.data);
        }

        private updateEvents(events: IEvent[]): void {
                for (let i = 0; i < events.length; i++) {
                        if (!events[i].com_ImageDownloadable) continue;
                        if (this.events.indexOf(events[i]) >= 0) continue;
                        this.events.push(events[i]);
                }
                this.events.sort((a, b) => {
                        if (a.time > b.time) return -1;
                        if (a.time < b.time) return 1;
                        return 0;
                });
        }
}

「custom-widgets\image-widget\image-widget-config.component.html」

C:\tmp\event-widget\custom-widgets\image-widget\image-widget-config.component.html

<div class="form-group">
        <c8y-form-group>
                <label translate>画像表示・画像情報一覧 設定</label>
        </c8y-form-group>
</div>

「custom-widgets\image-widget\image-widget-config.component.ts」

C:\tmp\event-widget\custom-widgets\image-widget\image-widget-config.component.ts

import { Component, Input } from '@angular/core';

@Component({
        selector: 'image-widget-config',
        templateUrl: './image-widget-config.component.html'
})
export class ImageWidgetConfig {
        @Input() config: any = {};
}

「custom-widgets\c8y.service.ts」

C:\tmp\event-widget\custom-widgets\c8y.service.ts

import { Injectable } from "@angular/core";
import {
        EventService,
        FetchClient,
        IResultList,
        IEvent,
        Realtime
} from "@c8y/client";

@Injectable()
export class C8yService {

        constructor(
                private eventService: EventService,
                private fetchClient: FetchClient
        ) { }

        /**
         * デバイスの Event 情報を取得する
         *
         * @param param {deviceId?, filter?} デバイスID, event 取得の際に指定する filter
         *
         * @returns ResultList<IEvent> の Promise
         */
        getEvents(param: {
                deviceId?: string;
                filter?: EventFilter;
        }): Promise<IResultList<IEvent>> {
                // 渡された filter を変更すると解決しづらいバグにつながるため、deep copy
                const requestFilter: any = (param.filter) ? JSON.parse(JSON.stringify(param.filter)) : {};
                if (param.deviceId) {
                        requestFilter.source = param.deviceId;
                }
                requestFilter.withTotalPages = true;

                return this.eventService.list(requestFilter);
        }


        /**
         * リアルタイムにイベントなどを取得するための Realtime オブジェクトを取得する
         *
         * @returns Realtime のインスタンス
         */
        getRealtime(): Realtime {
                return new Realtime(this.fetchClient);
        }

}

/**
 * getEvents で利用される filter。
 */
export interface EventFilter {
        /** ISO-8601 形式(2021-05-17T12:00:13.527+09:00 など) */
        dateFrom?: string;
        /** ISO-8601 形式(2021-05-17T12:00:13.527+09:00 など) */
        dateTo?: string;
        /**
         * dateTo または dateFrom とセットで利用する必要がある。
         * true の時、古い順、false の時、新しい順となる。
         */
        revert?: boolean;
        pageSize?: number;
        fragmentType?: string;
        source?: string;
        withTotalPages?: boolean;
}

「custom-widgets\image.service.ts」

C:\tmp\event-widget\custom-widgets\image.service.ts

import { Injectable } from "@angular/core";
import { IFetchOptions, FetchClient } from "@c8y/client";
import { Observable } from "rxjs";
import { IMediaDownloadable } from "./image.type";
import * as AWS from "aws-sdk";

@Injectable()
export class ImageService {
  private s3: AWS.S3;

  constructor(private fetchClient: FetchClient) {}

  /**
   * AWS S3 API から指定オブジェクトをダウンロードする
   *
   * @param {IMediaDownloadable} mediaDownloadable カスタムフラグメント
   *
   * @returns オブジェクトが通知される observable
   */
  getS3Object$(mediaDownloadable: IMediaDownloadable): Observable<any> {
    return new Observable((observer) => {
      (async () => {
        try {
          const options: IFetchOptions = {
            method: "GET",
            headers: { "Content-Type": "application/json" },
          };
          const resp = await this.fetchClient.fetch(`/service/wasabi-img/tempcred?keystoreid=${mediaDownloadable.keystoreId}`, options);
          const creds = await resp.json();
          this.s3 = this.createS3Instance(creds);
          this.s3.getObject(
            {
              Bucket: creds.bucket,
              Key: mediaDownloadable.fileInformation.name,
            },
            (err, data) => {
              if (err) {
                console.debug(err);
                observer.error(err);
              } else {
                observer.next(data);
              }
            }
          );
        } catch (err) {
          console.log("error occurred:" + err);
        }
      })();
    });
  }

  /**
   * AWS S3 のインスタンスを作成する
   *
   * @param {any} wasabiCredentials Wasabi keystore ID の wasabiCredentials フラグメント
   *
   * @returns AWS S3 インスタンス
   */
  private createS3Instance(wasabiCredentials: any): AWS.S3 {
    return new AWS.S3({
      endpoint: "s3.ap-northeast-1-ntt.wasabisys.com",
      accessKeyId: wasabiCredentials.AccessKeyId,
      secretAccessKey: wasabiCredentials.SecretAccessKey,
      sessionToken: wasabiCredentials.SessionToken,
      region: wasabiCredentials.region,
      signatureVersion: "v4",
    });
  }
}

「custom-widgets\image.type.ts」

C:\tmp\event-widget\custom-widgets\image.type.ts

export interface IMediaDownloadable {
                fileInformation: {
                                lastUpdated: string;
                                size: number;
                                name: string;
                                encoding: string;
                                'mime-type': string;
                };
                downloadURL: string;
                keystoreId: string;
}

export interface IVideoDownloadable extends IMediaDownloadable { }

「custom-widgets\preview-image.ts」

C:\tmp\event-widget\custom-widgets\preview-image.ts

export const previewImage = '';

「custom-widgets\s3.module.ts」

C:\tmp\event-widget\custom-widgets\s3.module.ts

import { NgModule, CUSTOM_ELEMENTS_SCHEMA } from '@angular/core';
import { CommonModule } from '@angular/common';
import { FormsModule } from '@angular/forms';
import { HOOK_COMPONENTS } from '@c8y/ngx-components';

import { CollapseModule } from 'ngx-bootstrap/collapse';

import { ImageWidget } from './image-widget/image-widget.component';
import { ImageWidgetConfig } from './image-widget/image-widget-config.component';

import { ImageService } from './image.service';

import * as previewImage from './preview-image';

@NgModule({
                imports: [
                                CommonModule,
                                FormsModule,
                                CollapseModule.forRoot()
                ],
                declarations: [
                                ImageWidget,
                                ImageWidgetConfig,
                ],
                entryComponents: [
                                ImageWidget,
                                ImageWidgetConfig,
                ],
                providers: [
                                ImageService,
                                {
                                                provide: HOOK_COMPONENTS,
                                                multi: true,
                                                useValue: {
                                                id: 'image-widget.widget',
                                                label: '画像表示・画像情報一覧',
                                                previewImage: previewImage.previewImage,
                                                description: 'IoT機器で撮影した画像の一覧と、選択した画像を表示します。',
                                                component: ImageWidget,
                                                configComponent: ImageWidgetConfig
                                                }
                                },
                ],
                schemas: [
                        CUSTOM_ELEMENTS_SCHEMA
                ]
})

export class S3Module {}