Skip to content

Commit

Permalink
feat: error handling/data-module core (#14)
Browse files Browse the repository at this point in the history
* Prevent data streams with associated errors from re-requesting over
  and over
 * Refactor subscription store to utilize RXJS to broadcast subscription
  updates to subscribers
 * Update types on @iot-app-kit/components to be any query
 * Remove unused dependency on redux-thunk
 * Up required code coverage targets
 * run linting on test
 * add subscribeToDataStreamsFrom stubs
  • Loading branch information
diehbria authored Nov 29, 2021
1 parent 025fcc4 commit 376929f
Show file tree
Hide file tree
Showing 55 changed files with 906 additions and 599 deletions.
2 changes: 2 additions & 0 deletions .prettierignore
Original file line number Diff line number Diff line change
Expand Up @@ -2,5 +2,7 @@ node_modules
www
build
dist
**/build/*
**/dist/*
coverage
packages/components/loader/*
3 changes: 2 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,8 @@
"fix": "npm-run-all -p fix:prettier fix:eslint",
"fix:eslint": "eslint --fix --ext .js,.ts,.tsx .",
"fix:prettier": "prettier --write packages/**/*.{ts,tsx,js,json}",
"test": "lerna run test --stream --concurrency 1",
"test": "npm-run-all -p test:unit",
"test:unit": "lerna run test --stream --concurrency 1",
"pack": "lerna run pack"
},
"devDependencies": {
Expand Down
48 changes: 31 additions & 17 deletions packages/components/src/components.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,55 +5,62 @@
* It contains typing information for all components that exist in this project.
*/
import { HTMLStencilElement, JSXBase } from "@stencil/core/internal";
import { AssetSummaryQuery, DataStreamQuery, Request } from "@iot-app-kit/core";
import { AnyDataStreamQuery, AssetSummaryQuery, DataModule, Request } from "@iot-app-kit/core";
import { DataStream, MinimalViewPortConfig } from "@synchro-charts/core";
export namespace Components {
interface IotAssetDetails {
"query": AssetSummaryQuery;
}
interface IotBarChart {
"appKit": DataModule;
"isEditing": boolean | undefined;
"query": DataStreamQuery;
"query": AnyDataStreamQuery;
"viewport": MinimalViewPortConfig;
"widgetId": string;
}
interface IotConnector {
"query": DataStreamQuery;
"appKit": DataModule;
"query": AnyDataStreamQuery;
"renderFunc": ({ dataStreams }: { dataStreams: DataStream[] }) => unknown;
"requestInfo": Request;
}
interface IotKpi {
"appKit": DataModule;
"isEditing": boolean | undefined;
"query": DataStreamQuery;
"query": AnyDataStreamQuery;
"viewport": MinimalViewPortConfig;
"widgetId": string;
}
interface IotLineChart {
"appKit": DataModule;
"isEditing": boolean | undefined;
"query": DataStreamQuery;
"query": AnyDataStreamQuery;
"viewport": MinimalViewPortConfig;
"widgetId": string;
}
interface IotScatterChart {
"appKit": DataModule;
"isEditing": boolean | undefined;
"query": DataStreamQuery;
"query": AnyDataStreamQuery;
"viewport": MinimalViewPortConfig;
"widgetId": string;
}
interface IotStatusGrid {
"appKit": DataModule;
"isEditing": boolean | undefined;
"query": DataStreamQuery;
"query": AnyDataStreamQuery;
"viewport": MinimalViewPortConfig;
"widgetId": string;
}
interface IotStatusTimeline {
"appKit": DataModule;
"isEditing": boolean | undefined;
"query": DataStreamQuery;
"query": AnyDataStreamQuery;
"viewport": MinimalViewPortConfig;
"widgetId": string;
}
interface IotTable {
"query": DataStreamQuery;
"query": AnyDataStreamQuery;
"viewport": MinimalViewPortConfig;
"widgetId": string;
}
Expand Down Expand Up @@ -148,48 +155,55 @@ declare namespace LocalJSX {
"query"?: AssetSummaryQuery;
}
interface IotBarChart {
"appKit"?: DataModule;
"isEditing"?: boolean | undefined;
"query"?: DataStreamQuery;
"query"?: AnyDataStreamQuery;
"viewport"?: MinimalViewPortConfig;
"widgetId"?: string;
}
interface IotConnector {
"query"?: DataStreamQuery;
"appKit"?: DataModule;
"query"?: AnyDataStreamQuery;
"renderFunc"?: ({ dataStreams }: { dataStreams: DataStream[] }) => unknown;
"requestInfo"?: Request;
}
interface IotKpi {
"appKit"?: DataModule;
"isEditing"?: boolean | undefined;
"query"?: DataStreamQuery;
"query"?: AnyDataStreamQuery;
"viewport"?: MinimalViewPortConfig;
"widgetId"?: string;
}
interface IotLineChart {
"appKit"?: DataModule;
"isEditing"?: boolean | undefined;
"query"?: DataStreamQuery;
"query"?: AnyDataStreamQuery;
"viewport"?: MinimalViewPortConfig;
"widgetId"?: string;
}
interface IotScatterChart {
"appKit"?: DataModule;
"isEditing"?: boolean | undefined;
"query"?: DataStreamQuery;
"query"?: AnyDataStreamQuery;
"viewport"?: MinimalViewPortConfig;
"widgetId"?: string;
}
interface IotStatusGrid {
"appKit"?: DataModule;
"isEditing"?: boolean | undefined;
"query"?: DataStreamQuery;
"query"?: AnyDataStreamQuery;
"viewport"?: MinimalViewPortConfig;
"widgetId"?: string;
}
interface IotStatusTimeline {
"appKit"?: DataModule;
"isEditing"?: boolean | undefined;
"query"?: DataStreamQuery;
"query"?: AnyDataStreamQuery;
"viewport"?: MinimalViewPortConfig;
"widgetId"?: string;
}
interface IotTable {
"query"?: DataStreamQuery;
"query"?: AnyDataStreamQuery;
"viewport"?: MinimalViewPortConfig;
"widgetId"?: string;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ import { newSpecPage } from '@stencil/core/testing';
import { MinimalLiveViewport } from '@synchro-charts/core';
import { IotBarChart } from './iot-bar-chart';
import { Components } from '../../components.d';
import { getDataModule, initialize, SiteWiseDataStreamQuery } from '@iot-app-kit/core';
import { registerDataSource, initialize, SiteWiseDataStreamQuery } from '@iot-app-kit/core';
import { createMockSource } from '../../testing/createMockSource';
import { DATA_STREAM } from '../../testing/mockWidgetProperties';
import { IotConnector } from '../iot-connector/iot-connector';
Expand All @@ -14,8 +14,8 @@ const viewport: MinimalLiveViewport = {
};

const barChartSpecPage = async (propOverrides: Partial<Components.IotBarChart> = {}) => {
initialize({ registerDataSources: false });
getDataModule().registerDataSource(createMockSource([DATA_STREAM]));
const appKit = initialize({ registerDataSources: false });
registerDataSource(appKit, createMockSource([DATA_STREAM]));

const page = await newSpecPage({
components: [IotBarChart, IotConnector],
Expand All @@ -24,6 +24,7 @@ const barChartSpecPage = async (propOverrides: Partial<Components.IotBarChart> =
});
const barChart = page.doc.createElement('iot-bar-chart') as CustomHTMLElement<Components.IotBarChart>;
const props: Partial<Components.IotBarChart> = {
appKit,
widgetId: 'test-bar-chart-widget',
isEditing: false,
query: {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { Component, Prop, h } from '@stencil/core';
import { MinimalViewPortConfig } from '@synchro-charts/core';
import { DataStreamQuery, Request } from '@iot-app-kit/core';
import { AnyDataStreamQuery, DataModule, Request } from '@iot-app-kit/core';

const DEFAULT_VIEWPORT = { duration: 10 * 1000 * 60 };

Expand All @@ -9,7 +9,9 @@ const DEFAULT_VIEWPORT = { duration: 10 * 1000 * 60 };
shadow: false,
})
export class IotBarChart {
@Prop() query: DataStreamQuery;
@Prop() appKit: DataModule;

@Prop() query: AnyDataStreamQuery;

@Prop() viewport: MinimalViewPortConfig = DEFAULT_VIEWPORT;

Expand All @@ -28,6 +30,7 @@ export class IotBarChart {
const requestInfo = this.requestInfo();
return (
<iot-connector
appKit={this.appKit}
query={this.query}
requestInfo={requestInfo}
renderFunc={({ dataStreams }) => (
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { newSpecPage } from '@stencil/core/testing';
import { MinimalLiveViewport } from '@synchro-charts/core';
import flushPromises from 'flush-promises';
import { getDataModule, initialize, SiteWiseDataStreamQuery } from '@iot-app-kit/core';
import { registerDataSource, initialize, SiteWiseDataStreamQuery } from '@iot-app-kit/core';
import { IotConnector } from './iot-connector';
import { createMockSource } from '../../testing/createMockSource';
import { update } from '../../testing/update';
Expand All @@ -16,8 +16,8 @@ const viewport: MinimalLiveViewport = {

const connectorSpecPage = async (propOverrides: Partial<Components.IotConnector> = {}) => {
/** Initialize data source and register mock data source */
initialize({ registerDataSources: false });
getDataModule().registerDataSource(createMockSource([DATA_STREAM]));
const appKit = initialize({ registerDataSources: false });
registerDataSource(appKit, createMockSource([DATA_STREAM]));

const page = await newSpecPage({
components: [IotConnector],
Expand All @@ -26,6 +26,7 @@ const connectorSpecPage = async (propOverrides: Partial<Components.IotConnector>
});
const connector = page.doc.createElement('iot-connector') as CustomHTMLElement<Components.IotConnector>;
const props: Partial<Components.IotConnector> = {
appKit,
query: {
source: 'test-mock',
assets: [],
Expand Down
Original file line number Diff line number Diff line change
@@ -1,28 +1,31 @@
import { Component, Prop, State, Watch } from '@stencil/core';
import { DataStream } from '@synchro-charts/core';
import isEqual from 'lodash.isequal';
import { getDataModule, Request, DataStreamQuery, SubscriptionUpdate } from '@iot-app-kit/core';
import { Request, AnyDataStreamQuery, SubscriptionUpdate, subscribeToDataStreams, DataModule } from '@iot-app-kit/core';

@Component({
tag: 'iot-connector',
shadow: false,
})
export class IotConnector {
@Prop() query: DataStreamQuery;
@Prop() appKit: DataModule;

@Prop() query: AnyDataStreamQuery;

@Prop() requestInfo: Request;

@Prop() renderFunc: ({ dataStreams }: { dataStreams: DataStream[] }) => unknown;

@State() dataStreams: DataStream[] = [];

private update: (subscriptionUpdate: SubscriptionUpdate<DataStreamQuery>) => void;
private update: (subscriptionUpdate: SubscriptionUpdate<AnyDataStreamQuery>) => void;

private unsubscribe: () => void;

componentWillLoad() {
// Subscribe to data module for the requested `query`
const { update, unsubscribe } = getDataModule().subscribeToDataStreams(
const { update, unsubscribe } = subscribeToDataStreams(
this.appKit,
{
query: this.query,
requestInfo: this.requestInfo,
Expand Down
7 changes: 4 additions & 3 deletions packages/components/src/components/iot-kpi/iot-kpi.spec.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { newSpecPage } from '@stencil/core/testing';
import { getDataModule, SiteWiseDataStreamQuery, initialize } from '@iot-app-kit/core';
import { registerDataSource, SiteWiseDataStreamQuery, initialize } from '@iot-app-kit/core';
import { MinimalLiveViewport } from '@synchro-charts/core';
import { IotKpi } from './iot-kpi';
import { Components } from '../../components.d';
Expand All @@ -14,8 +14,8 @@ const viewport: MinimalLiveViewport = {
};

const kpiSpecPage = async (propOverrides: Partial<Components.IotKpi> = {}) => {
initialize({ registerDataSources: false });
getDataModule().registerDataSource(createMockSource([DATA_STREAM]));
const appKit = initialize({ registerDataSources: false });
registerDataSource(appKit, createMockSource([DATA_STREAM]));

const page = await newSpecPage({
components: [IotKpi, IotConnector],
Expand All @@ -24,6 +24,7 @@ const kpiSpecPage = async (propOverrides: Partial<Components.IotKpi> = {}) => {
});
const kpi = page.doc.createElement('iot-kpi') as CustomHTMLElement<Components.IotKpi>;
const props: Partial<Components.IotKpi> = {
appKit,
widgetId: 'test-kpi-widget',
isEditing: false,
query: {
Expand Down
7 changes: 5 additions & 2 deletions packages/components/src/components/iot-kpi/iot-kpi.tsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { Component, Prop, h } from '@stencil/core';
import { MinimalViewPortConfig } from '@synchro-charts/core';
import { DataStreamQuery } from '@iot-app-kit/core';
import { AnyDataStreamQuery, DataModule } from '@iot-app-kit/core';

const DEFAULT_VIEWPORT = { duration: 10 * 1000 };

Expand All @@ -9,7 +9,9 @@ const DEFAULT_VIEWPORT = { duration: 10 * 1000 };
shadow: false,
})
export class IotKpi {
@Prop() query: DataStreamQuery;
@Prop() appKit: DataModule;

@Prop() query: AnyDataStreamQuery;

@Prop() viewport: MinimalViewPortConfig = DEFAULT_VIEWPORT;

Expand All @@ -28,6 +30,7 @@ export class IotKpi {
const requestInfo = this.requestInfo();
return (
<iot-connector
appKit={this.appKit}
query={this.query}
requestInfo={requestInfo}
renderFunc={({ dataStreams }) => (
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { newSpecPage } from '@stencil/core/testing';
import { MinimalLiveViewport } from '@synchro-charts/core';
import { IotLineChart } from './iot-line-chart';
import { getDataModule } from '@iot-app-kit/core/src/data-module';
import { registerDataSource } from '@iot-app-kit/core';
import { Components } from '../../components.d';
import { createMockSource } from '../../testing/createMockSource';
import { DATA_STREAM } from '../../testing/mockWidgetProperties';
Expand All @@ -15,8 +15,8 @@ const viewport: MinimalLiveViewport = {
};

const lineChartSpecPage = async (propOverrides: Partial<Components.IotKpi> = {}) => {
initialize({ registerDataSources: false });
getDataModule().registerDataSource(createMockSource([DATA_STREAM]));
const appKit = initialize({ registerDataSources: false });
registerDataSource(appKit, createMockSource([DATA_STREAM]));

const page = await newSpecPage({
components: [IotLineChart, IotConnector],
Expand All @@ -25,6 +25,7 @@ const lineChartSpecPage = async (propOverrides: Partial<Components.IotKpi> = {})
});
const lineChart = page.doc.createElement('iot-line-chart') as CustomHTMLElement<Components.IotStatusGrid>;
const props: Partial<Components.IotStatusGrid> = {
appKit,
widgetId: 'test-line-chart-widget',
isEditing: false,
query: {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
import { Component, Prop, h } from '@stencil/core';
import { MinimalViewPortConfig } from '@synchro-charts/core';
import { DataStreamQuery, Request } from '@iot-app-kit/core';
import { AnyDataStreamQuery, DataModule, Request } from '@iot-app-kit/core';

const DEFAULT_VIEWPORT = { duration: 10 * 1000 * 60 };

Expand All @@ -9,7 +9,9 @@ const DEFAULT_VIEWPORT = { duration: 10 * 1000 * 60 };
shadow: false,
})
export class IotLineChart {
@Prop() query: DataStreamQuery;
@Prop() appKit: DataModule;

@Prop() query: AnyDataStreamQuery;

@Prop() viewport: MinimalViewPortConfig = DEFAULT_VIEWPORT;

Expand All @@ -28,6 +30,7 @@ export class IotLineChart {
const requestInfo = this.requestInfo();
return (
<iot-connector
appKit={this.appKit}
query={this.query}
requestInfo={requestInfo}
renderFunc={({ dataStreams }) => (
Expand Down
Loading

0 comments on commit 376929f

Please sign in to comment.