Skip to content

Commit 70d2762

Browse files
authored
Desktop build works with electron-packager (#8)
* Start working on desktop builds * More progress... * Building works! * Building works! * PostgreSQL working
1 parent a367469 commit 70d2762

15 files changed

+784
-99
lines changed

.gitignore

+3-1
Original file line numberDiff line numberDiff line change
@@ -5,4 +5,6 @@ ui/build
55
ui/data
66
data
77
build
8-
electron/build
8+
electron/build
9+
yarn-error.log
10+
releases

.prettierrc.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
11
{
2-
"singleQuote": true
2+
"singleQuote": true
33
}

README.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
# DataStation Community Edition
22

3-
DataStation allows you to seamless connect SQL queries, HTTP requests,
3+
DataStation allows you to seamlessly connect SQL queries, HTTP requests,
44
scripting, and visualization in a single platform.
55

66
![A screenshot of app.datastation.multiprocess.io](./screenshot.jpg)
@@ -29,4 +29,4 @@ interested in commercial support or to partner on the roadmap, please
2929

3030
## License
3131

32-
This software is licensed under an Apache 2 license.
32+
This software is licensed under an Apache 2.0 license.

desktop/app.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@ app.whenReady().then(() => {
2323
// In addition to removing the menu on Windows/Linux this also disables Chrome devtools.
2424
win.removeMenu();
2525
}
26-
win.loadFile('index.html');
26+
win.loadFile(path.join(__dirname, 'index.html'));
2727

2828
registerRPCHandlers(ipcMain, [
2929
...storeHandlers,

package.json

+50-41
Original file line numberDiff line numberDiff line change
@@ -1,43 +1,52 @@
11
{
2-
"name": "ui",
3-
"version": "1.0.0",
4-
"license": "MIT",
5-
"scripts": {
6-
"typecheck": "yarn tsc",
7-
"build-desktop": "python ./scripts/build.py ./desktop/scripts/desktop.build",
8-
"build-ui": "python ./scripts/build.py ./ui/scripts/ui.build",
9-
"start-desktop": "yarn build-ui && yarn build-desktop && yarn electron --trace-warnings build/desktop.js",
10-
"start-ui": "yarn build-ui && python3 -m http.server --directory build 8080",
11-
"format": "yarn prettier --write \"**/*.ts\" \"**/*.tsx\" \"**/*.css\""
12-
},
13-
"dependencies": {
14-
"@types/pg": "^8.6.0",
15-
"@types/uuid": "^8.3.0",
16-
"chart.js": "^3.3.2",
17-
"electron": "^13.0.1",
18-
"node-fetch": "^2.6.1",
19-
"pako": "^2.0.3",
20-
"papaparse": "^5.3.1",
21-
"parquetjs-lite": "^0.8.7",
22-
"pg": "^8.6.0",
23-
"prismjs": "^1.23.0",
24-
"react": "^17.0.2",
25-
"react-dom": "^17.0.2",
26-
"react-simple-code-editor": "^0.11.0",
27-
"sql.js": "^1.5.0",
28-
"uuid": "^8.3.2",
29-
"xlsx": "^0.17.0"
30-
},
31-
"devDependencies": {
32-
"@types/electron": "^1.6.10",
33-
"@types/node-fetch": "^2.5.10",
34-
"@types/pako": "^1.0.1",
35-
"@types/papaparse": "^5.2.5",
36-
"@types/prismjs": "^1.16.5",
37-
"@types/react": "^17.0.9",
38-
"@types/react-dom": "^17.0.6",
39-
"esbuild": "^0.12.7",
40-
"prettier": "^2.3.0",
41-
"typescript": "^4.3.2"
42-
}
2+
"name": "datastation",
3+
"productName": "DataStation Community Edition",
4+
"version": "0.0.1",
5+
"license": "Apache-2.0",
6+
"description": "The Data IDE for Developers",
7+
"author": "DataStation Authors",
8+
"main": "build/desktop.js",
9+
"scripts": {
10+
"typecheck": "yarn tsc",
11+
"build-desktop": "python ./scripts/build.py ./desktop/scripts/desktop.build",
12+
"build-ui": "python ./scripts/build.py ./ui/scripts/ui.build",
13+
"start-desktop": "yarn build-ui && yarn build-desktop && yarn electron --trace-warnings build/desktop.js",
14+
"start-ui": "yarn build-ui && python3 -m http.server --directory build 8080",
15+
"build-desktop-app": "yarn electron-packager --overwrite --out=releases .",
16+
"format": "yarn prettier --write \"**/*.ts\" \"**/*.tsx\" \"**/*.css\" \"**/*.json\""
17+
},
18+
"dependencies": {
19+
"@types/lodash.debounce": "^4.0.6",
20+
"@types/pg": "^8.6.0",
21+
"@types/uuid": "^8.3.0",
22+
"chart.js": "^3.3.2",
23+
"lodash.debounce": "^4.0.8",
24+
"node-fetch": "^2.6.1",
25+
"pako": "^2.0.3",
26+
"papaparse": "^5.3.1",
27+
"parquetjs-lite": "^0.8.7",
28+
"pg": "^8.6.0",
29+
"prismjs": "^1.23.0",
30+
"react": "^17.0.2",
31+
"react-dom": "^17.0.2",
32+
"react-simple-code-editor": "^0.11.0",
33+
"sql.js": "^1.5.0",
34+
"uuid": "^8.3.2",
35+
"xlsx": "^0.17.0"
36+
},
37+
"devDependencies": {
38+
"@types/electron": "^1.6.10",
39+
"@types/node-fetch": "^2.5.10",
40+
"@types/pako": "^1.0.1",
41+
"@types/papaparse": "^5.2.5",
42+
"@types/prismjs": "^1.16.5",
43+
"@types/react": "^17.0.9",
44+
"@types/react-dom": "^17.0.6",
45+
"electron": "^13.1.2",
46+
"electron-packager": "^15.2.0",
47+
"electron-winstaller": "^5.0.0",
48+
"esbuild": "^0.12.7",
49+
"prettier": "^2.3.0",
50+
"typescript": "^4.3.2"
51+
}
4352
}

scripts/build.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,7 @@ def lex_command(command):
5858
return line
5959

6060
for command in script:
61-
if command.strip().startswith('#'):
61+
if command.strip().startswith('#') or not command.strip():
6262
continue
6363

6464
line = lex_command(command)

shared/state.ts

+16-4
Original file line numberDiff line numberDiff line change
@@ -147,11 +147,22 @@ export class GraphPanelInfo extends PanelInfo {
147147
}
148148

149149
export class SQLPanelInfo extends PanelInfo {
150-
sql: SQLConnectorInfo;
150+
sql: {
151+
type: SQLConnectorInfoType;
152+
connectorIndex: number;
153+
};
151154

152-
constructor(name?: string, sql?: SQLConnectorInfo, content?: string) {
155+
constructor(
156+
name?: string,
157+
type?: SQLConnectorInfoType,
158+
connectorIndex?: number,
159+
content?: string
160+
) {
153161
super('sql', name, content);
154-
this.sql = new SQLConnectorInfo();
162+
this.sql = {
163+
type: type || 'in-memory',
164+
connectorIndex: connectorIndex || 0,
165+
};
155166
}
156167
}
157168

@@ -269,7 +280,8 @@ export const DEFAULT_PROJECT: ProjectState = new ProjectState(
269280
),
270281
new SQLPanelInfo(
271282
'Transform with SQL',
272-
new SQLConnectorInfo('In Memory', 'in-memory'),
283+
'in-memory',
284+
0,
273285
'SELECT name, age+5 AS age FROM DM_getPanel(0);'
274286
),
275287
(() => {

tsconfig.json

+11-14
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,13 @@
11
{
2-
"compilerOptions": {
3-
"noImplicitAny": true,
4-
"noEmit": true,
5-
"jsx": "react",
6-
"target": "esnext",
7-
"module": "commonjs",
8-
"skipLibCheck": true,
9-
"esModuleInterop": true,
10-
"typeRoots": [
11-
"./node_modules/@types",
12-
"./type-overrides"
13-
]
14-
},
15-
"exclude": ["node_modules", "*.js", "build"]
2+
"compilerOptions": {
3+
"noImplicitAny": true,
4+
"noEmit": true,
5+
"jsx": "react",
6+
"target": "esnext",
7+
"module": "commonjs",
8+
"skipLibCheck": true,
9+
"esModuleInterop": true,
10+
"typeRoots": ["./node_modules/@types", "./type-overrides"]
11+
},
12+
"exclude": ["node_modules", "*.js", "build"]
1613
}

ui/Page.tsx

+4-2
Original file line numberDiff line numberDiff line change
@@ -1,25 +1,27 @@
11
import * as React from 'react';
22

3-
import { ProjectPage } from './../shared/state';
3+
import { ConnectorInfo, ProjectPage } from './../shared/state';
44

55
import { Panels } from './Panels';
66
import { evalPanel } from './Panel';
77
import { PanelResult } from './ProjectStore';
88

99
export function Page({
1010
page,
11+
connectors,
1112
updatePage,
1213
panelResults,
1314
setPanelResults,
1415
}: {
1516
page: ProjectPage;
1617
updatePage: (page: ProjectPage) => void;
18+
connectors: Array<ConnectorInfo>;
1719
panelResults: Array<PanelResult>;
1820
setPanelResults: (panelIndex: number, results: PanelResult) => void;
1921
}) {
2022
async function reevalPanel(panelIndex: number) {
2123
try {
22-
const r = await evalPanel(page, panelIndex, panelResults);
24+
const r = await evalPanel(page, panelIndex, panelResults, connectors);
2325
setPanelResults(panelIndex, { lastRun: new Date(), value: r });
2426
} catch (e) {
2527
setPanelResults(panelIndex, {

ui/Pages.tsx

+1
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,7 @@ export function Pages({
7777

7878
<Page
7979
page={page}
80+
connectors={state.connectors}
8081
updatePage={updatePage}
8182
panelResults={panelResultsByPage[page.id]}
8283
setPanelResults={setPanelResults}

ui/Panel.tsx

+4-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import * as React from 'react';
22

33
import {
4+
ConnectorInfo,
45
ProjectPage,
56
PanelInfo,
67
GraphPanelInfo,
@@ -38,7 +39,8 @@ export const PANEL_TYPE_ICON = {
3839
export async function evalPanel(
3940
page: ProjectPage,
4041
panelId: number,
41-
panelResults: Array<PanelResult>
42+
panelResults: Array<PanelResult>,
43+
connectors: Array<ConnectorInfo>
4244
) {
4345
const panel = page.panels[panelId];
4446
switch (panel.type) {
@@ -47,7 +49,7 @@ export async function evalPanel(
4749
case 'literal':
4850
return await evalLiteralPanel(panel as LiteralPanelInfo);
4951
case 'sql':
50-
return evalSQLPanel(panel as SQLPanelInfo, panelResults);
52+
return evalSQLPanel(panel as SQLPanelInfo, panelResults, connectors);
5153
case 'graph':
5254
return (panelResults[(panel as GraphPanelInfo).graph.panelSource] || {})
5355
.value;

ui/SQLPanel.tsx

+12-17
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,8 @@ import { Select } from './component-library/Select';
1414

1515
export async function evalSQLPanel(
1616
panel: SQLPanelInfo,
17-
panelResults: Array<PanelResult>
17+
panelResults: Array<PanelResult>,
18+
connectors: Array<ConnectorInfo>
1819
) {
1920
if (!panel.content.trim().toLowerCase().startsWith('select ')) {
2021
throw new Error('SQL must be read-only SELECT');
@@ -78,15 +79,15 @@ export async function evalSQLPanel(
7879
throw new Error('Malformed substitution in SQL');
7980
}
8081

81-
if (panel.sql.sql.type === 'postgres') {
82+
if (panel.sql.type === 'postgres') {
8283
return await asyncRPC<SQLConnectorInfo, string, Array<object>>(
8384
'evalSQL',
8485
content,
85-
panel.sql
86+
connectors[+panel.sql.connectorIndex] as SQLConnectorInfo
8687
);
8788
}
8889

89-
if (panel.sql.sql.type === 'in-memory') {
90+
if (panel.sql.type === 'in-memory') {
9091
let sql = (window as any).SQL;
9192
if (!sql) {
9293
sql = (window as any).SQL = await (window as any).initSqlJs({
@@ -96,12 +97,7 @@ export async function evalSQLPanel(
9697

9798
const db = new sql.Database();
9899
let res: any;
99-
try {
100-
res = db.exec(content)[0];
101-
} catch (e) {
102-
console.error(e, content);
103-
throw new Error('Syntax error');
104-
}
100+
res = db.exec(content)[0];
105101

106102
return res.values.map((row: Array<any>) => {
107103
const formattedRow: { [k: string]: any } = {};
@@ -132,32 +128,31 @@ export function SQLPanelDetails({
132128
className="block"
133129
value={panel.sql.type}
134130
onChange={(value: string) => {
135-
panel.sql.sql.type = value as SQLConnectorInfoType;
131+
panel.sql.type = value as SQLConnectorInfoType;
136132
updatePanel(panel);
137133
}}
138134
>
139135
{MODE_FEATURES.sql && <option value="postgres">PostgreSQL</option>}
140136
<option value="in-memory">In-Memory SQL</option>
141137
</Select>
142-
{MODE_FEATURES.sql && (
138+
{MODE_FEATURES.sql && panel.sql.type !== 'in-memory' && (
143139
<Select
144140
label="Connector"
145141
className="block"
146-
value={panel.sql.type}
142+
value={panel.sql.connectorIndex.toString()}
147143
onChange={(connectorIndex: string) => {
148-
panel.sql.sql = (
149-
connectors[+connectorIndex] as SQLConnectorInfo
150-
).sql;
144+
panel.sql.connectorIndex = +connectorIndex;
151145
updatePanel(panel);
152146
}}
147+
nodefault
153148
>
154149
{connectors
155150
.map((c: ConnectorInfo, index: number) => {
156151
if (c.type != 'sql') {
157152
return null;
158153
}
159154

160-
return <option value={index}>c.name</option>;
155+
return <option value={index}>{c.name}</option>;
161156
})
162157
.filter(Boolean)}
163158
</Select>

ui/app.tsx

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import * as pako from 'pako';
22
import * as React from 'react';
33
import * as ReactDOM from 'react-dom';
4+
import debounce from 'lodash.debounce';
45

56
import { APP_NAME, MODE, MODE_FEATURES } from '../shared/constants';
67
import {
@@ -66,7 +67,8 @@ function useProjectState(
6667
const [state, setProjectState] = React.useState<ProjectState>(null);
6768

6869
function setState(newState: ProjectState) {
69-
store.update(projectId, newState);
70+
const syncMillis = 5000;
71+
debounce(() => store.update(projectId, newState), syncMillis);
7072
setProjectState(newState);
7173
}
7274

ui/component-library/Select.tsx

+6-2
Original file line numberDiff line numberDiff line change
@@ -7,13 +7,15 @@ export function Select({
77
disabled,
88
label,
99
className,
10+
nodefault,
1011
}: {
1112
value: string;
1213
onChange: (value: string) => void;
1314
children: React.ReactNode;
1415
disabled?: boolean;
1516
label?: string;
1617
className?: string;
18+
nodefault?: boolean;
1719
}) {
1820
let selectClass = 'select';
1921
if (className) {
@@ -27,9 +29,11 @@ export function Select({
2729
onChange={(e: React.ChangeEvent<HTMLSelectElement>) =>
2830
onChange(e.target.value)
2931
}
30-
children={children}
3132
disabled={disabled}
32-
/>
33+
>
34+
{nodefault && <option label=" "></option>}
35+
{children}
36+
</select>
3337
);
3438

3539
if (label) {

0 commit comments

Comments
 (0)