Compare commits

...

378 Commits

Author SHA1 Message Date
Jan Prochazka
f43ffd9af4 mongo form view WIP (blind branch?) 2021-04-05 08:52:24 +02:00
Jan Prochazka
573e404612 json row view 2021-04-05 08:51:12 +02:00
Jan Prochazka
91c88bd92d handle no data, clear filter 2021-04-04 21:38:53 +02:00
Jan Prochazka
475f82a35e mongo conditions support 2021-04-04 10:37:24 +02:00
Jan Prochazka
0548bae7af simplified flow of selectedCellsPublished 2021-04-04 08:16:51 +02:00
Jan Prochazka
0413f4b4d9 refactor: joined FreeTableGrid and DataGrid into one component 2021-04-04 08:05:29 +02:00
Jan Prochazka
9e9991c675 fixed expandable column order 2021-04-03 21:53:13 +02:00
Jan Prochazka
21502bda65 fixed expandable column order 2021-04-03 21:11:04 +02:00
Jan Prochazka
69e1c6c625 nosql: view expandable data (arrays, objects) 2021-04-03 20:45:57 +02:00
Jan Prochazka
fcedeb2316 Merge branch 'master' into mongo 2021-04-03 19:53:14 +02:00
Jan Prochazka
a23ff752a3 upgraded dependencies 2021-04-03 19:53:00 +02:00
Jan Prochazka
138b0414f2 packages-tools v4.0.3-rc.1 2021-04-03 18:54:58 +02:00
Jan Prochazka
87988d5c3a db analyser fix 2021-04-03 18:53:46 +02:00
Jan Prochazka
41cf7009b3 json view 2021-04-03 18:52:11 +02:00
Jan Prochazka
18860c823d mongo data grid works 2021-04-02 19:54:09 +02:00
Jan Prochazka
55cc51d24a data grid row height meter 2021-04-02 19:31:49 +02:00
Jan Prochazka
2a49eaab12 WIP 2021-04-02 19:12:06 +02:00
Jan Prochazka
394c6028c9 mong support WIP 2021-04-02 18:40:07 +02:00
Jan Prochazka
d4bd6e03c9 fix 2021-04-02 13:35:54 +02:00
Jan Prochazka
1a94222262 v4.0.3-beta.1 2021-04-02 09:06:22 +02:00
Jan Prochazka
94b41fecbc open tabs on startup fix 2021-04-02 09:04:06 +02:00
Jan Prochazka
9ed6932c1e fixes for firefox 2021-04-02 08:50:56 +02:00
Jan Prochazka
e748591c10 use resize observer 2021-04-01 15:59:15 +02:00
Jan Prochazka
fca6b87cb9 fixed dev 2021-04-01 15:17:06 +02:00
Jan Prochazka
c23ecfff47 v4.0.2 2021-04-01 14:06:12 +02:00
Jan Prochazka
8738665dcf web build fix 2021-04-01 14:05:57 +02:00
Jan Prochazka
a05fc90579 v4.0.1 2021-04-01 13:52:54 +02:00
Jan Prochazka
71f7a705c4 fixed dbgate npm edition 2021-04-01 13:52:28 +02:00
Jan Prochazka
9cb2d397ad dbgate removed dependencies 2021-04-01 13:37:48 +02:00
Jan Prochazka
484e7c27a2 open favorites on startup 2021-04-01 13:32:44 +02:00
Jan Prochazka
acdba0c52c fixed docker app 2021-04-01 13:13:41 +02:00
Jan Prochazka
943544958a set dependency version to 4.0.0 2021-04-01 12:49:57 +02:00
Jan Prochazka
e78420b1b0 v4.0.0 2021-04-01 12:35:48 +02:00
Jan Prochazka
69d639b9ea changelog 2021-04-01 12:35:14 +02:00
Jan Prochazka
a540220c05 v4.0.0-beta.5 2021-04-01 11:08:48 +02:00
Jan Prochazka
b1ddcbfbfc upgraded plugin dependencies 2021-04-01 11:08:21 +02:00
Jan Prochazka
87aaa281e4 packages-tools v4.0.0-rc-2 2021-04-01 11:00:22 +02:00
Jan Prochazka
704733d80d ts fix 2021-04-01 10:59:46 +02:00
Jan Prochazka
a50458494e sql generator fix 2021-04-01 10:56:36 +02:00
Jan Prochazka
2a980a7892 DB ping can recover DB connection 2021-04-01 10:48:42 +02:00
Jan Prochazka
a3762c6caa sql generator 2021-04-01 10:48:20 +02:00
Jan Prochazka
d6ba822338 correct handle big data in sql preview 2021-04-01 10:28:54 +02:00
Jan Prochazka
f146d70e2b sql generator 2021-04-01 08:27:58 +02:00
Jan Prochazka
d62177d996 sql generator 2021-04-01 08:09:22 +02:00
Jan Prochazka
a1993214e2 v4.0.0-beta.4 2021-03-29 21:46:40 +02:00
Jan Prochazka
70411b764b dependency fix 2021-03-29 21:45:50 +02:00
Jan Prochazka
cd0fb0fdf2 v4.0.0-beta.3 2021-03-28 20:14:20 +02:00
Jan Prochazka
9713c9b88e sqlgen design 2021-03-28 20:11:58 +02:00
Jan Prochazka
d5118909d1 sql generator 2021-03-28 18:38:45 +02:00
Jan Prochazka
bb41236a5f sql generator - generates SQL inserts 2021-03-27 20:39:27 +01:00
Jan Prochazka
9d84c0f213 sql generator in connection process 2021-03-27 20:14:39 +01:00
Jan Prochazka
d45fbcb8c8 sqlgen 2021-03-27 19:57:03 +01:00
Jan Prochazka
4762597741 sql generator - basic concept 2021-03-27 19:40:56 +01:00
Jan Prochazka
9c27c224ec fullscreen modal template, sql generator modal 2021-03-27 16:45:51 +01:00
Jan Prochazka
2268d6126b packages-tools v4.0.0-rc.1 2021-03-27 16:11:37 +01:00
Jan Prochazka
bbc50ea3fb fixes 2021-03-27 16:04:12 +01:00
Jan Prochazka
11985004b5 timer label 2021-03-27 11:36:37 +01:00
Jan Prochazka
4f58d2ff80 fixed editor 2021-03-27 10:38:49 +01:00
Jan Prochazka
987fe6095a fix 2021-03-27 10:20:40 +01:00
Jan Prochazka
d8fcbc8c17 database context menu 2021-03-27 10:14:00 +01:00
Jan Prochazka
833610c88b error handler improved, fixes 2021-03-27 10:07:36 +01:00
Jan Prochazka
218478c128 fixes 2021-03-27 08:48:59 +01:00
Jan Prochazka
3f8ff91e2c handle app crash 2021-03-27 08:30:46 +01:00
Jan Prochazka
11436c065c handle app crash 2021-03-27 08:28:21 +01:00
Jan Prochazka
f1c70f6f82 v4.0.0-beta.2 2021-03-25 20:37:10 +01:00
Jan Prochazka
c66ff5820c select template 2021-03-25 20:33:24 +01:00
Jan Prochazka
4c87ad50b1 text cell view 2021-03-25 20:17:25 +01:00
Jan Prochazka
6c5f5a7cfb delete connection command 2021-03-25 19:27:09 +01:00
Jan Prochazka
9876a76836 additional macros 2021-03-25 19:19:14 +01:00
Jan Prochazka
952bdc4baa run macros in table data editor #68 2021-03-25 19:07:51 +01:00
Jan Prochazka
2afa7a5f58 fix 2021-03-25 13:50:23 +01:00
Jan Prochazka
bc9e8a2ea6 change theme menu command 2021-03-25 13:38:06 +01:00
Jan Prochazka
e2dcfe9940 change recent database 2021-03-25 13:31:24 +01:00
Jan Prochazka
638b04877d change to recent database 2021-03-25 13:20:42 +01:00
Jan Prochazka
2a9c67d2f6 v4.0.0-beta.1 2021-03-25 12:52:34 +01:00
Jan Prochazka
2c9d424fc8 improved colors 2021-03-25 12:47:20 +01:00
Jan Prochazka
4e76f10175 themes 2021-03-25 12:39:38 +01:00
Jan Prochazka
5f2afc037e favorites & about button in toolbar 2021-03-25 10:53:35 +01:00
Jan Prochazka
50e61cdce1 filter focus- fix deadlock (using ref) 2021-03-25 10:20:25 +01:00
Jan Prochazka
586f2fed21 filterfocus 2021-03-25 10:05:04 +01:00
Jan Prochazka
020b4163d7 connection list refresh 2021-03-25 09:30:23 +01:00
Jan Prochazka
6b0e1e322a data grid - handle edge cases (error, structure not loaded) 2021-03-25 08:46:04 +01:00
Jan Prochazka
0d0bd29812 fixes from testing 2021-03-25 08:39:07 +01:00
Jan Prochazka
4e4447de8a cell data view 2021-03-22 21:18:44 +01:00
Jan Prochazka
15c9e93e8a css for command palette 2021-03-22 19:01:57 +01:00
Jan Prochazka
5531705433 data grid context menu extended 2021-03-22 18:57:11 +01:00
Jan Prochazka
dfadf0653d readme 2021-03-22 18:33:53 +01:00
Jan Prochazka
742b68453a favorite modal 2021-03-22 18:31:33 +01:00
Jan Prochazka
155406827e Merge branch 'develop' 2021-03-22 17:56:40 +01:00
Jan Prochazka
a97e6dbcab insert SQL join 2021-03-21 20:48:04 +01:00
Jan Prochazka
d4989c75ca code completion 2021-03-21 20:03:46 +01:00
Jan Prochazka
b7b9dde5ae wokring electron app build + simplified web build 2021-03-21 19:20:42 +01:00
Jan Prochazka
34f2fb2a0a form view 2021-03-21 17:55:25 +01:00
Jan Prochazka
965a967450 formview commands 2021-03-21 10:49:46 +01:00
Jan Prochazka
40872699c6 upgrade mysql dependency 2021-03-21 10:27:23 +01:00
Jan Prochazka
ec90a8b952 formview save 2021-03-21 10:20:57 +01:00
Jan Prochazka
90c4b44fdb datagrid commands 2021-03-21 09:50:44 +01:00
Jan Prochazka
df5062c9a5 resize observer polyfill 2021-03-21 08:54:49 +01:00
Jan Prochazka
437155e4c5 use resize observer in formview 2021-03-21 08:51:18 +01:00
Jan Prochazka
5ebee161ae form view 2021-03-21 08:16:15 +01:00
Jan Prochazka
10c77ad153 form view 2021-03-20 20:22:49 +01:00
Jan Prochazka
5e59926556 markdown view tab 2021-03-20 10:59:14 +01:00
Jan Prochazka
4b1b61328a favorite editor json tab 2021-03-20 10:44:05 +01:00
Jan Prochazka
4c6d9f0660 favorites list 2021-03-20 10:30:34 +01:00
Jan Prochazka
a5a7447cec local storage garbage collector 2021-03-20 10:08:27 +01:00
Jan Prochazka
dd373f9db9 closed tabs 2021-03-20 10:05:36 +01:00
Jan Prochazka
bb0f5e4404 plugin tab 2021-03-20 09:40:30 +01:00
Jan Prochazka
c77bc820d4 plugins widget 2021-03-20 09:01:26 +01:00
Jan Prochazka
a1ab47a6f9 chart popup menu 2021-03-20 08:35:18 +01:00
Jan Prochazka
efc07280a6 context menu 2021-03-20 08:16:32 +01:00
Jan Prochazka
dcb4c5071a solved configStore problem 2021-03-19 20:57:09 +01:00
Jan Prochazka
7b625c6073 working charts 2021-03-19 20:31:36 +01:00
Jan Prochazka
489c9a905c chart initial import 2021-03-19 20:12:13 +01:00
Jan Prochazka
75c578de47 query designer 2021-03-19 17:12:05 +01:00
Jan Prochazka
f7c4bbc708 query design columns 2021-03-19 16:57:00 +01:00
Jan Prochazka
9c1227273c query designer context menu 2021-03-18 20:21:07 +01:00
Jan Prochazka
47a045fc24 drag & drop references 2021-03-18 18:34:36 +01:00
Jan Prochazka
9e9df60d37 designer - draw references 2021-03-18 17:10:51 +01:00
Jan Prochazka
93b7a9a674 designer 2021-03-18 16:16:21 +01:00
Jan Prochazka
b96576ba6f query designer 2021-03-18 13:57:50 +01:00
Jan Prochazka
0524b4c5b6 table structure tab 2021-03-18 11:51:19 +01:00
Jan Prochazka
b7663e2e06 view data tab 2021-03-18 10:28:41 +01:00
Jan Prochazka
24f4e1d898 widget column bar - respects sizes 2021-03-18 09:58:25 +01:00
Jan Prochazka
9089f78593 preview data grid, fix multiple D & D to electron 2021-03-18 09:02:48 +01:00
Jan Prochazka
8f90dad303 import files controls 2021-03-18 07:57:21 +01:00
Jan Prochazka
c823e18699 open file command 2021-03-17 19:12:15 +01:00
Jan Prochazka
73bfac2bfb electron menu, about dialog 2021-03-17 19:04:28 +01:00
Jan Prochazka
08b5bce03c changed ts=>js 2021-03-17 18:34:32 +01:00
Jan Prochazka
ddf8a5806c simplified runner output files 2021-03-17 18:33:38 +01:00
Jan Prochazka
eceab2dde9 using table control with slots 2021-03-17 18:30:36 +01:00
Jan Prochazka
9c7df42948 use of createRef instead of not working { corrent: xxx } 2021-03-17 18:20:26 +01:00
Jan Prochazka
321d5d71de group commands 2021-03-15 21:06:15 +01:00
Jan Prochazka
d4a35fb414 commands running from electron 2021-03-15 20:48:43 +01:00
Jan Prochazka
21feb3a042 fmt 2021-03-15 19:41:42 +01:00
Jan Prochazka
9adb4b41c6 redo in toolbar 2021-03-15 19:38:56 +01:00
Jan Prochazka
3b3e81e3f7 command enabling refactor 2021-03-15 19:33:37 +01:00
Jan Prochazka
dfa8ca6797 free table editor - working macros 2021-03-14 21:12:29 +01:00
Jan Prochazka
0af207d330 macro parameters 2021-03-14 20:44:19 +01:00
Jan Prochazka
49337a4112 free table - save to archive 2021-03-14 13:18:29 +01:00
Jan Prochazka
7cd26c4fe4 free table editor 2021-03-14 10:40:38 +01:00
Jan Prochazka
159ba72129 archive tab 2021-03-14 08:39:02 +01:00
Jan Prochazka
cfb772c717 archive widget 2021-03-14 08:30:06 +01:00
Jan Prochazka
500c1c76ba D & D CSV files 2021-03-13 20:28:06 +01:00
Jan Prochazka
834eeabd3f D&D screen 2021-03-13 12:28:35 +01:00
Jan Prochazka
8770034bf5 export working 2021-03-13 12:04:33 +01:00
Jan Prochazka
423f876d68 export - tables select 2021-03-13 09:57:40 +01:00
Jan Prochazka
af54c958ba database object context menu 2021-03-13 09:45:43 +01:00
Jan Prochazka
4e16119835 selectfield using svelte select 2021-03-13 09:20:27 +01:00
Jan Prochazka
d6f9db48aa use svelte select 2021-03-11 21:00:25 +01:00
Jan Prochazka
2063005d5c import export 2021-03-11 20:37:05 +01:00
Jan Prochazka
c2c54856ff markdown editor 2021-03-11 14:57:12 +01:00
Jan Prochazka
cedb740fb0 shell tab 2021-03-11 14:37:40 +01:00
Jan Prochazka
913f89e970 shell tab 2021-03-11 13:40:19 +01:00
Jan Prochazka
7d6bf90a0a search/replace in ctx menu 2021-03-11 11:19:15 +01:00
Jan Prochazka
8a4275fb09 save tab 2021-03-11 10:52:51 +01:00
Jan Prochazka
d5ebea3764 files widget 2021-03-11 09:42:14 +01:00
Jan Prochazka
a93aff1bb7 ace editor 2021-03-11 08:17:06 +01:00
Jan Prochazka
c193955fbe editor context menu, focus fix 2021-03-11 07:53:37 +01:00
Jan Prochazka
5f97f7d922 grid focus fix 2021-03-10 22:02:54 +01:00
Jan Prochazka
54d17a67d4 datagrid, query - solved focus, active tab 2021-03-10 18:50:02 +01:00
Jan Prochazka
0c94d7fcac TabContent component split 2021-03-10 18:23:02 +01:00
Jan Prochazka
fca23f65de load query data fix 2021-03-10 18:12:30 +01:00
Jan Prochazka
a35d5525a3 query grid 2021-03-08 20:00:51 +01:00
Jan Prochazka
904d35d26a execute query 2021-03-08 18:23:41 +01:00
Jan Prochazka
929c08e094 simplified ace editor 2021-03-07 20:28:07 +01:00
Jan Prochazka
97a27381f2 execute query 2021-03-07 20:25:04 +01:00
Jan Prochazka
f4fe5b9b53 use editor data 2021-03-07 11:38:02 +01:00
Jan Prochazka
00d5b25baa ace editor 2021-03-07 10:17:05 +01:00
Jan Prochazka
c6e95dbb6a try to fix 2021-03-07 09:53:33 +01:00
Jan Prochazka
da6bd9b475 sql editor - not working 2021-03-07 09:50:20 +01:00
Jan Prochazka
2eebe44f87 tabs tools 2021-03-07 07:57:17 +01:00
Jan Prochazka
3dd99a44cb paste 2021-03-06 20:49:07 +01:00
Jan Prochazka
ec2acebdc9 data grid - copy to clipboard 2021-03-06 20:46:07 +01:00
Jan Prochazka
e49c0169da grid commands + ctx menu 2021-03-06 20:32:02 +01:00
Jan Prochazka
49f22e1a3b save commands 2021-03-06 18:50:58 +01:00
Jan Prochazka
423644e9d9 save dialog 2021-03-06 18:01:39 +01:00
Jan Prochazka
b64b6be68a filter colors 2021-03-06 15:14:07 +01:00
Jan Prochazka
d3c4c18b62 upgraded svelte + used svelte:fragment 2021-03-06 13:45:35 +01:00
Jan Prochazka
fe5826bc8e group by in datagrid 2021-03-06 13:02:01 +01:00
Jan Prochazka
7709e24ccd resize datagrid columns 2021-03-06 12:07:28 +01:00
Jan Prochazka
b91cf18aee master/detail working 2021-03-06 11:35:15 +01:00
Jan Prochazka
ae9c4b4f98 reference manager 2021-03-06 09:59:56 +01:00
Jan Prochazka
3d25a51cf9 fix 2021-03-06 09:32:55 +01:00
Jan Prochazka
18e7171038 column manager 2021-03-06 09:31:02 +01:00
Jan Prochazka
8cf014efa4 splitters 2021-03-06 08:52:35 +01:00
Jan Prochazka
78d71602bf moved some widgets to elements 2021-03-06 08:09:29 +01:00
Jan Prochazka
dcfd6ee1dc index.html 2021-03-05 20:59:00 +01:00
Jan Prochazka
eb4ecb4cf8 filter multiple values dialog 2021-03-05 20:49:42 +01:00
Jan Prochazka
bc54564d64 set filter modal 2021-03-05 20:29:10 +01:00
Jan Prochazka
1c7052810a fix 2021-03-05 17:48:49 +01:00
Jan Prochazka
4bfba2ec02 datagrid 2021-03-05 17:47:07 +01:00
Jan Prochazka
c2dc4d76ba connection modal converted 2021-03-04 17:55:45 +01:00
Jan Prochazka
ce44e271ae input styles 2021-03-04 15:52:12 +01:00
Jan Prochazka
ef5bfb5a89 forms 2021-03-04 15:20:08 +01:00
Jan Prochazka
7acea0f4ac form provider 2021-03-04 10:56:58 +01:00
Jan Prochazka
593e61abb9 connection modal 2021-03-04 10:04:34 +01:00
Jan Prochazka
a0aa508e8d statusbar 2021-03-01 21:09:03 +01:00
Jan Prochazka
ca517f9c73 sql object list 2021-03-01 20:47:38 +01:00
Jan Prochazka
ad0e02de5d columns in tables list 2021-03-01 20:27:09 +01:00
Jan Prochazka
ea05ae0079 fix 2021-03-01 18:39:54 +01:00
Jan Prochazka
689eb7baa2 change current db by tab 2021-03-01 18:37:15 +01:00
Jan Prochazka
a4387155e7 expand icon simplified 2021-03-01 18:26:01 +01:00
Jan Prochazka
565a60e638 app object list 2021-02-28 21:16:08 +01:00
Jan Prochazka
5dba5a6dfd app object module methods, search in connection list 2021-02-28 18:43:16 +01:00
Jan Prochazka
c497c1ceca connection context menu 2021-02-28 18:12:13 +01:00
Jan Prochazka
5929a01010 fix 2021-02-28 10:37:07 +01:00
Jan Prochazka
182b0e0a75 tabs drop down menu 2021-02-28 10:32:14 +01:00
Jan Prochazka
2cc74b594e dropdown menu implementation 2021-02-28 10:17:52 +01:00
Jan Prochazka
e9430988f4 fix 2021-02-28 09:28:47 +01:00
Jan Prochazka
f30bd0a894 draggable splitter on mainscreen 2021-02-28 09:27:57 +01:00
Jan Prochazka
1c0a8cad56 use get_current_component from svelte/internal 2021-02-28 08:36:02 +01:00
Jan Prochazka
8a4fd302d2 Merge pull request #65 from janproch/master
Svelte proof of concept
2021-02-26 20:29:32 +01:00
Jan Prochazka
db7f8d6a74 command text fix 2021-02-26 20:25:54 +01:00
Jan Prochazka
4f1eb4003a toolbar logic 2021-02-26 19:34:29 +01:00
Jan Prochazka
3efaac7d1f toolbar ordered 2021-02-26 19:32:52 +01:00
Jan Prochazka
d9387bef1f toolbar 2021-02-26 19:25:35 +01:00
Jan Prochazka
a101f21483 theme moved to plugin 2021-02-26 16:52:22 +01:00
Jan Prochazka
8a0d10e50d keyboard commands 2021-02-25 21:43:23 +01:00
Jan Prochazka
fe1fc7923f change theme from command palette 2021-02-25 19:30:20 +01:00
Jan Prochazka
f0802dc471 command palette control 2021-02-25 18:05:44 +01:00
Jan Prochazka
30ade5867c datagrid selection 2021-02-25 11:45:06 +01:00
Jan Prochazka
ea54673497 datagrid 2021-02-25 11:02:27 +01:00
Jan Prochazka
2ffd729465 grid scroll bars 2021-02-25 10:25:34 +01:00
Jan Prochazka
ef910f43a6 datagrid 2021-02-25 09:12:01 +01:00
Jan Prochazka
fc333167ac datagrid 2021-02-25 09:02:08 +01:00
Jan Prochazka
390447c948 datagrid 2021-02-22 21:09:06 +01:00
Jan Prochazka
1bb5f4974d conection - extinfo 2021-02-22 18:58:07 +01:00
Jan Prochazka
48e3cf1be5 load extensions 2021-02-22 18:50:54 +01:00
Jan Prochazka
1e540b3fe9 table data grid 2021-02-22 17:34:24 +01:00
Jan Prochazka
60c1090d6c tabs panel 2021-02-21 11:35:50 +01:00
Jan Prochazka
71bea87a7a view table list in database widget 2021-02-21 09:57:26 +01:00
Jan Prochazka
a03261bfd4 reactive variables - better approach 2021-02-21 08:43:36 +01:00
Jan Prochazka
704a04e9bb fix 2021-02-21 08:40:27 +01:00
Jan Prochazka
28c1421294 connections pinger 2021-02-20 21:35:24 +01:00
Jan Prochazka
7a5bcc62c8 color icon classes 2021-02-20 19:37:19 +01:00
Jan Prochazka
321eedefea web-svelte => web 2021-02-20 19:16:05 +01:00
Jan Prochazka
daf9e9d18b remove web 2021-02-20 19:15:11 +01:00
Jan Prochazka
dd7db5904c connection app object 2021-02-20 19:14:22 +01:00
Jan Prochazka
6bddf3aa83 lang ts 2021-02-20 17:57:53 +01:00
Jan Prochazka
9743569ca7 metadata loaders as svelte stream 2021-02-20 17:47:48 +01:00
Jan Prochazka
40ed020c0a left panel 2021-02-20 11:53:00 +01:00
Jan Prochazka
63ac08cc27 dim left panel width 2021-02-20 10:43:43 +01:00
Jan Prochazka
e16b0ef61f themes simplified 2021-02-20 10:40:29 +01:00
Jan Prochazka
14f9a40851 left panel logic 2021-02-17 20:08:16 +01:00
Jan Prochazka
ba6abd1e64 theme logic reused 2021-02-17 19:26:48 +01:00
Jan Prochazka
4ffc5842bb left iconbar 2021-02-17 18:46:27 +01:00
Jan Prochazka
4cb304a41c v3.9.6 2021-02-15 18:38:49 +01:00
Jan Prochazka
abdbe4122f changelog 2021-02-15 18:37:50 +01:00
Jan Prochazka
b2e37e88ea fix 2021-02-15 18:37:13 +01:00
Jan Prochazka
6a28ceaa51 v3.9.6-beta.7 2021-02-15 06:41:03 +01:00
Jan Prochazka
c15f859eae #63 - clean solution - added ...process.env to fork(env) 2021-02-15 06:40:41 +01:00
Jan Prochazka
f717dfa3b5 readme 2021-02-14 21:18:35 +01:00
Jan Prochazka
ab7510c8e8 readme 2021-02-14 21:15:59 +01:00
Jan Prochazka
3601ac21b4 changelog 2021-02-14 21:09:08 +01:00
Jan Prochazka
65bd5a60ef v3.9.6-beta.6 2021-02-14 19:26:49 +01:00
Jan Prochazka
6a9bd8248c #63 - workaround export from SNAP 2021-02-14 19:26:00 +01:00
Jan Prochazka
98ff6db701 runners cwd simplified 2021-02-14 19:09:52 +01:00
Jan Prochazka
7b67576131 Revert "electron changed to version 9"
This reverts commit 1f057fb0a9.
2021-02-14 18:09:36 +01:00
Jan Prochazka
d0d6d86bb9 v3.9.6-beta.5 2021-02-14 11:51:00 +01:00
Jan Prochazka
1f057fb0a9 electron changed to version 9 2021-02-14 11:50:29 +01:00
Jan Prochazka
75a429b74f build npm only on final release 2021-02-14 11:46:29 +01:00
Jan Prochazka
d34524c3d0 v3.9.6-alpha.14 2021-02-14 11:37:57 +01:00
Jan Prochazka
bdfa66d37d v3.9.6-beta.4 2021-02-14 11:37:31 +01:00
Jan Prochazka
1d5d87e26a try to upgrade electron 2021-02-14 11:37:06 +01:00
Jan Prochazka
9f3aadc17d fixed exports in dbgate NPM package 2021-02-14 11:33:53 +01:00
Jan Prochazka
58f213d042 v3.9.6-beta.3 2021-02-14 11:10:27 +01:00
Jan Prochazka
89dbf38962 try to fix export in electron 2021-02-14 11:09:49 +01:00
Jan Prochazka
670e3d127e sql editor context menu 2021-02-14 10:50:55 +01:00
Jan Prochazka
72181e70a1 #63 2021-02-14 09:41:32 +01:00
Jan Prochazka
c3ac836fa9 upgrade plugin dependencies 2021-02-14 09:37:59 +01:00
Jan Prochazka
b0deba4bae style 2021-02-14 09:26:24 +01:00
Jan Prochazka
d08fc85459 SSL support 2021-02-14 09:11:40 +01:00
Jan Prochazka
417ec9fcd2 connection modal style 2021-02-13 19:56:04 +01:00
Jan Prochazka
f9d4a9a3a0 connection modal style 2021-02-13 19:46:15 +01:00
Jan Prochazka
eab870c237 connection form style 2021-02-13 19:05:37 +01:00
Jan Prochazka
521199ee1a handle ssh tunnel error 2021-02-13 12:46:37 +01:00
Jan Prochazka
0d1a6e96f3 code format 2021-02-13 12:14:23 +01:00
Jan Prochazka
1076fb8391 ssh tunnel - alternative modes 2021-02-13 12:13:10 +01:00
Jan Prochazka
114dc0b543 ssh tunnel - reuse SSH connection + local port for multiple DB connections 2021-02-13 07:47:55 +01:00
Jan Prochazka
728ca72cc1 ssh tunnel - wking POC 2021-02-11 11:34:54 +01:00
Jan Prochazka
e243ecd96a fixed repo links 2021-02-11 10:11:34 +01:00
Jan Prochazka
2defdc3f28 issue templates 2021-02-08 19:58:22 +01:00
Jan Prochazka
7aeef55a58 Update issue templates 2021-02-08 19:55:51 +01:00
Jan Prochazka
5e8967da52 issue templates 2021-02-08 19:51:27 +01:00
Jan Prochazka
679145a394 v3.9.6-alpha.13 2021-02-08 19:22:56 +01:00
Jan Prochazka
9b012c187a v3.9.6-beta.2 2021-02-08 19:22:36 +01:00
Jan Prochazka
9ce1fdd59e version in reposirory should be last stable version 2021-02-08 19:22:13 +01:00
Jan Prochazka
c0d0a00615 try to fix electron export 2021-02-08 19:21:35 +01:00
Jan Prochazka
c67c08bd69 v3.9.6-beta.1 2021-02-08 18:48:59 +01:00
Jan Prochazka
6e846797b9 v3.9.6-alpha.12 2021-02-08 18:37:38 +01:00
Jan Prochazka
a3ad98d2a9 missing dependency 2021-02-08 18:37:27 +01:00
Jan Prochazka
c7dbf333c7 v3.9.6-alpha.11 2021-02-08 18:28:46 +01:00
Jan Prochazka
19392e9406 fixes + optimalized web package 2021-02-08 18:27:26 +01:00
Jan Prochazka
818de9b111 v3.9.6-alpha.10 2021-02-08 18:08:24 +01:00
Jan Prochazka
0292a37b16 readme 2021-02-08 18:08:07 +01:00
Jan Prochazka
8b9031b0c2 npm token refactor 2021-02-08 17:59:37 +01:00
Jan Prochazka
d88591032e web can be run from dbgate package 2021-02-08 17:55:09 +01:00
Jan Prochazka
2c6a59638b fix 2021-02-08 17:26:27 +01:00
Jan Prochazka
8312415430 v3.9.6-alpha.9 2021-02-08 17:15:33 +01:00
Jan Prochazka
fdb14d687b setCurrentVersion script 2021-02-08 17:15:08 +01:00
Jan Prochazka
a20b351938 v3.9.6-alpha.9 2021-02-07 17:53:19 +01:00
Jan Prochazka
f8016d26ec fix 2021-02-07 17:52:51 +01:00
Jan Prochazka
ad186f5efb v3.9.6-alpha.8 2021-02-07 17:49:02 +01:00
Jan Prochazka
8e26918975 fixes 2021-02-07 17:48:46 +01:00
Jan Prochazka
e0303aa77e v3.9.6-alpha.7 2021-02-07 17:37:24 +01:00
Jan Prochazka
6a02c4ebaa fix 2021-02-07 17:37:06 +01:00
Jan Prochazka
2c73ab6bc1 v3.9.6-alpha.6 2021-02-07 17:25:05 +01:00
Jan Prochazka
b60714f30c npm build dbgate 2021-02-07 17:24:45 +01:00
Jan Prochazka
f3163617e0 v3.9.6-alpha.5 2021-02-07 10:52:33 +01:00
Jan Prochazka
834be32676 npmrc file 2021-02-07 10:52:20 +01:00
Jan Prochazka
0f6637188b v3.9.6-alpha.4 2021-02-07 10:43:47 +01:00
Jan Prochazka
ef0921ecf5 try to fix 2021-02-07 10:43:36 +01:00
Jan Prochazka
1ffa613e09 v3.9.6-alpha.3 2021-02-07 10:33:13 +01:00
Jan Prochazka
c0c8cd88e3 npm token 2021-02-07 10:32:58 +01:00
Jan Prochazka
2666717c3a v3.9.6-alpha.2 2021-02-07 10:19:15 +01:00
Jan Prochazka
777abbc097 v3.9.6-aplha.2 2021-02-07 10:18:21 +01:00
Jan Prochazka
e4db985ef9 fix 2021-02-07 10:17:49 +01:00
Jan Prochazka
6afaa6f856 v3.9.6-alpha.1 2021-02-07 10:13:24 +01:00
Jan Prochazka
1325851bcf npm build 2021-02-07 10:13:07 +01:00
Jan Prochazka
fb7da60127 fix 2021-02-07 10:01:15 +01:00
Jan Prochazka
ecde9cb6bd fix 2021-02-07 09:58:57 +01:00
Jan Prochazka
6b06ed5baf set current version 2021-02-07 09:56:50 +01:00
Jan Prochazka
2aa965cf3b license, API - prepare to run from dbgate npm package 2021-02-07 09:44:42 +01:00
Jan Prochazka
fc11fe1e8d readme 2021-02-04 16:23:11 +01:00
Jan Prochazka
e33e14bd5f v3.9.5 2021-02-01 19:49:00 +01:00
Jan Prochazka
0c7fc0b7b6 v3.9.5-beta.2 2021-02-01 19:35:06 +01:00
Jan Prochazka
5904d45c44 Revert "upgraded electron - fixed problem with deleted localstorage"
This reverts commit 84e475192e.
2021-02-01 18:28:38 +01:00
Jan Prochazka
5d997fc1c9 revert 2021-02-01 18:28:16 +01:00
Jan Prochazka
4693564ffa v3.9.5-beta.1 2021-02-01 19:08:12 +01:00
Jan Prochazka
3377929b30 v0.0.0 2021-02-01 19:08:01 +01:00
Jan Prochazka
a35f6f2629 reverted - private package 2021-02-01 19:07:46 +01:00
Jan Prochazka
cb264ac6cc v3.9.4 2021-02-01 18:41:31 +01:00
Jan Prochazka
9236e1a6c2 Merge branch 'develop' 2021-02-01 18:40:20 +01:00
Jan Prochazka
df359aea58 reverted try to dbgate global package 2021-02-01 18:39:55 +01:00
Jan Prochazka
fdf60b5267 dbgate package 2021-02-01 18:09:57 +01:00
Jan Prochazka
bd3c18d883 v3.9.4-beta.5 2021-01-31 09:24:27 +01:00
Jan Prochazka
18bf6e5979 open data files using open dialog in electron + drag & drop in electron without uploading 2021-01-31 09:21:54 +01:00
Jan Prochazka
edaf9676e4 readme 2021-01-31 08:00:37 +01:00
Jan Prochazka
bd524d345a readme 2021-01-31 07:51:20 +01:00
Jan Prochazka
0a39a6829c screenshot 2021-01-31 07:46:50 +01:00
Jan Prochazka
5c7a011efb report problem menu 2021-01-31 07:08:15 +01:00
Jan Prochazka
4e350e99c4 v3.9.4-beta.4 2021-01-30 20:07:28 +01:00
Jan Prochazka
a714f7ae54 small refactor 2021-01-30 20:05:50 +01:00
Jan Prochazka
a17b76c570 save from electron menu 2021-01-30 20:01:54 +01:00
Jan Prochazka
54d476a972 open sql file with drag & drop 2021-01-30 19:06:30 +01:00
Jan Prochazka
255c3e5ef4 improved save file experience 2021-01-30 18:23:05 +01:00
Jan Prochazka
059eabf2fa rename 2021-01-30 12:49:02 +01:00
Jan Prochazka
79fdde73ae v3.9.4-beta.3 2021-01-30 11:01:03 +01:00
Jan Prochazka
84e475192e upgraded electron - fixed problem with deleted localstorage 2021-01-30 10:56:22 +01:00
Jan Prochazka
3907b1ae8b v3.9.4-beta.2 2021-01-30 10:43:07 +01:00
Jan Prochazka
dcfefc78a2 fixed save generated content in useEditorData 2021-01-30 10:37:28 +01:00
Jan Prochazka
d3039a9248 useStorage improved - setter never changes (behaves more like useState) 2021-01-30 10:29:56 +01:00
Jan Prochazka
31dd80b79a fixed qorking with tabs 2021-01-30 09:41:50 +01:00
Jan Prochazka
8d6d1d979e v3.9.4-beta.1 2021-01-28 18:54:15 +01:00
Jan Prochazka
fe1c5f5801 electron: save file to custom location 2021-01-28 18:52:59 +01:00
Jan Prochazka
df976a84d2 electron menu sinplified 2021-01-28 16:56:47 +01:00
Jan Prochazka
420e94600e closed tab - show more info 2021-01-28 16:41:30 +01:00
Jan Prochazka
9940bd5177 upgrade mysql plugin dependency 2021-01-28 15:29:45 +01:00
Jan Prochazka
45d99a4126 timer labels in query design tab and shell tab 2021-01-28 13:00:24 +01:00
Jan Prochazka
c2b7c775c0 code cleanup 2021-01-28 12:56:06 +01:00
Jan Prochazka
51ba9d3b5a statusbar - show query execution duration 2021-01-28 12:49:00 +01:00
Jan Prochazka
8396e726ec numbering tabs 2021-01-28 10:10:41 +01:00
Jan Prochazka
cb67b57faf numbering tabs fix 2021-01-28 10:07:02 +01:00
Jan Prochazka
99381536d7 numbering tabs 2021-01-28 10:05:27 +01:00
Jan Prochazka
a9cb9f1874 style fix 2021-01-28 09:48:33 +01:00
Jan Prochazka
420a58380a upgraded required plugin-postgres version 2021-01-28 09:44:48 +01:00
Jan Prochazka
75ca3cbb11 packages-tools v1.0.8 2021-01-28 09:12:38 +01:00
Jan Prochazka
a5c1966a94 makeUniqueColumnNames function 2021-01-28 09:12:10 +01:00
Jan Prochazka
ca4ff95316 optimalized connection ping 2021-01-28 08:16:31 +01:00
Jan Prochazka
a3294950a4 v3.9.3 2021-01-26 20:40:15 +01:00
Jan Prochazka
29355a6d3e v3.9.3-beta.1 2021-01-26 20:31:10 +01:00
Jan Prochazka
add0ba09c3 fix 2021-01-26 20:30:38 +01:00
Jan Prochazka
005ae87309 v3.9.2 2021-01-25 17:46:53 +01:00
Jan Prochazka
5f372a1d0f v3.9.2-beta.1 2021-01-25 17:33:42 +01:00
Jan Prochazka
ecce75960a fix in open ref table 2021-01-25 17:33:16 +01:00
Jan Prochazka
72cc510c64 improved error boundary 2021-01-25 17:20:17 +01:00
Jan Prochazka
7e39b8c2a0 v3.9.1 2021-01-25 06:52:06 +01:00
Jan Prochazka
ed4ef4d999 v3.9.1-beta.2 2021-01-24 17:26:28 +01:00
546 changed files with 22318 additions and 24197 deletions

33
.github/ISSUE_TEMPLATE/bug_report.md vendored Normal file
View File

@@ -0,0 +1,33 @@
---
name: Bug report
about: Create a report to help us improve DbGate
title: 'BUG: Say something here'
labels: ''
assignees: ''
---
**Describe the bug**
A clear and concise description of what the bug is.
**To Reproduce**
Steps to reproduce the behavior:
1. Go to '...'
2. Click on '....'
3. See error
**Expected behavior**
A clear and concise description of what you expected to happen.
**Screenshots**
If applicable, add screenshots to help explain your problem.
**Version Information (please complete the following information):**
- OS: [e.g. Windows/Mac/Ubuntu]
- App Version [see Help -> About]
- Install source [e.g. installer/SNAP/Docker/NPM]
- Type - Web/Application
- Database engine: [e.g. MySQL/PostgreSQL/SQL Server]
**Additional context**
Anything else you think might be helpful

View File

@@ -0,0 +1,20 @@
---
name: Feature request
about: Suggest an idea for DbGate
title: 'FEAT: '
labels: ''
assignees: ''
---
**Is your feature request related to a problem? Please describe.**
A clear and concise description of what the problem is. Ex. I'm always frustrated when [...]
**Describe the solution you'd like**
A clear and concise description of what you want to happen.
**Describe alternatives you've considered**
A clear and concise description of any alternative solutions or features you've considered.
**Additional context**
Add any other context or screenshots about the feature request here.

18
.github/ISSUE_TEMPLATE/question.md vendored Normal file
View File

@@ -0,0 +1,18 @@
---
name: Question
about: Ask a question about how to do something
title: 'QUESTION: Summary of your question'
labels: ''
assignees: ''
---
**Details:**
Details about your question
**Version Information [might be relevant to your issue]**
Operating System:
App Version [help -> about]:
**Screenshot [if appropriate]**:
A screenshot of the app if that helps

91
.github/workflows/build-npm.yaml vendored Normal file
View File

@@ -0,0 +1,91 @@
name: NPM packages
# on: [push]
on:
push:
tags:
- 'v[0-9]+.[0-9]+.[0-9]+'
# - 'v[0-9]+.[0-9]+.[0-9]+-alpha.[0-9]+'
# on:
# push:
# branches:
# - production
jobs:
build:
runs-on: ${{ matrix.os }}
strategy:
matrix:
os: [ubuntu-18.04]
# os: [macOS-10.14, windows-2016, ubuntu-18.04]
steps:
- name: Context
env:
GITHUB_CONTEXT: ${{ toJson(github) }}
run: echo "$GITHUB_CONTEXT"
- uses: actions/checkout@v1
with:
fetch-depth: 1
- name: Use Node.js 10.x
uses: actions/setup-node@v1
with:
node-version: 10.x
- name: Configure NPM token
env:
NPM_TOKEN: ${{ secrets.NPM_TOKEN }}
run: |
npm config set '//registry.npmjs.org/:_authToken' "${NPM_TOKEN}"
- name: yarn install
run: |
yarn install
- name: setCurrentVersion
run: |
yarn setCurrentVersion
- name: Publish types
working-directory: packages/types
run: |
npm publish
- name: Publish tools
working-directory: packages/tools
run: |
npm publish
- name: Publish sqltree
working-directory: packages/sqltree
run: |
npm publish
- name: Publish api
working-directory: packages/api
run: |
npm publish
- name: Publish datalib
working-directory: packages/datalib
run: |
npm publish
- name: Publish filterparser
working-directory: packages/filterparser
run: |
npm publish
- name: Publish web
working-directory: packages/web
run: |
npm publish
- name: Publish dbgate
working-directory: packages/dbgate
run: |
npm publish

5
.gitignore vendored
View File

@@ -12,6 +12,10 @@ node_modules
build
dist
app/packages/web/public
docker/public
docker/bundle.js
# misc
.DS_Store
.env.local
@@ -24,3 +28,4 @@ yarn-debug.log*
yarn-error.log*
app/src/nativeModulesContent.js
packages/api/src/nativeModulesContent.js
.VSCodeCounter

28
CHANGELOG.md Normal file
View File

@@ -0,0 +1,28 @@
# ChangeLog
### 4.0.3
- FIX: fixes for FireFox (mainly incorrent handle of bind:clientHeight, replaces with resizeobserver)
### 4.0.2
- FIX: fixed docker and NPM build
### 4.0.0
- CHANGED: Excahnged React with Svelte. Changed theme colors. Huge speed and memory optimalization
- ADDED: SQL Generator (CREATE, INSERT, DROP)
- ADDED: Command palette (F1). Introduced commands, extended some context menus
- ADDED: New keyboard shortcuts
- ADDED: Switch to recent database feature
- ADDED: Macros from free table editor are available also in table data editor
- CHANGED: Cell data preview is now in left widgets panel
- CHANGED: Toolbar refactor
- FIX: Solved reconnecting expired connection
### 3.9.6
- ADDED: Connect using SSH Tunnel
- ADDED: Connect using SSL
- ADDED: Database connection dialog redesigned
- ADDED: #63 Ctrl+Enter runs query
- ADDED: Published dbgate NPM package
- ADDED: SQL editor context menu
- FIX: #62 - import, export executed from SNAP installs didn't work
### 3.9.5
- Start point of changelog

View File

@@ -1,37 +1,46 @@
[![NPM version](https://img.shields.io/npm/v/dbgate.svg)](https://www.npmjs.com/package/dbgate)
[![dbgate](https://snapcraft.io/dbgate/badge.svg)](https://snapcraft.io/dbgate)
[![dbgate](https://snapcraft.io/dbgate/trending.svg?name=0)](https://snapcraft.io/dbgate)
[![styled with prettier](https://img.shields.io/badge/styled_with-prettier-ff69b4.svg)](https://github.com/prettier/prettier)
[![Donate](https://img.shields.io/badge/donate-paypal-blue.svg)](https://paypal.me/JanProchazkaCz/30eur)
[![NPM version](https://img.shields.io/npm/v/dbgate-api.svg)](https://www.npmjs.com/package/dbgate-api)
# DbGate - database administration tool
DbGate is fast and efficient database administration tool. It is focused to work with data (filtering, editing, master/detail views etc.)
DbGate is fast and easy to use database manager. Works with MySQL, PostgreSQL and SQL Server.
**Try it online** - https://demo.dbgate.org - online demo application
* Try it online - [demo.dbgate.org](https://demo.dbgate.org) - online demo application
* Download application for Windows, Linux or Mac from [dbgate.org](https://dbgate.org/download/)
* Run web version as [NPM package](https://www.npmjs.com/package/dbgate) or as [docker image](https://hub.docker.com/r/dbgate/dbgate)
![Screenshot](https://raw.githubusercontent.com/dbgate/dbgate/master/screenshot.png)
## Features
* Support for Microsoft SQL Server, Postgre SQL, MySQL
* Table data browsing - filtering, sorting, related columns using foreign keys
* Connect to Microsoft SQL Server, Postgre SQL, MySQL
* Table data editing, with SQL change script preview
* Master/detail views
* Query designer
* Form view for comfortable work with tables with many columns
* Charts
* Browsing objects - tables, views, procedures, functions
* Table data editing, with SQL change script preview
* Explore tables, views, procedures, functions
* SQL editor, execute SQL script, SQL code formatter, SQL code completion, SQL join wizard
* Runs as application for Windows, Linux and Mac. Or in Docker container on server and in web Browser on client.
* Import, export from/to CSV, Excel, JSON
* Free table editor - quick table data editing (cleanup data after import/before export, prototype tables etc.)
* Archives - backup your data in JSON files on local filesystem (or on DbGate server, when using web application)
* Light and dark theme
* Charts
* For detailed info, how to run DbGate in docker container, visit [docker hub](https://hub.docker.com/r/dbgate/dbgate)
* Extensible plugin architecture
![Screenshot](https://raw.githubusercontent.com/dbshell/dbgate/master/screenshot.png)
## Why is DbGate different
There are many database managers now, so why DbGate?
* Works everywhere - Windows, Linux, Mac, Web browser (+mobile web is planned), without compromises in features
* Based on standalone NPM packages, scripts can be run without DbGate (example - [CSV export](https://www.npmjs.com/package/dbgate-plugin-csv) )
* Many data browsing functions based using foreign keys - master/detail, expand columns, expandable form view (on screenshot above)
## Design goals
* Application simplicity - DbGate takes the best and only the best from old [DbGate](http://www.jenasoft.com/dbgate), [DatAdmin](http://www.jenasoft.com/datadmin) and [DbMouse](http://www.jenasoft.com/dbmouse) .
* Minimal dependencies
* Frontend - React, styled-components, socket.io
* Frontend - Svelte, socket.io
* Backend - NodeJs, ExpressJs, socket.io, database connection drivers
* JavaScript + TypeScript
* App - electron
@@ -41,36 +50,12 @@ DbGate is fast and efficient database administration tool. It is focused to work
## Plugins
Plugins are standard NPM packages published on [npmjs.com](https://www.npmjs.com).
See all [existing DbGate plugins](https://www.npmjs.com/search?q=keywords:dbgateplugin).
Visit [dbgate generator homepage](https://github.com/dbshell/generator-dbgate) to see, how to create your own plugin.
Visit [dbgate generator homepage](https://github.com/dbgate/generator-dbgate) to see, how to create your own plugin.
Currently following extensions can be implemented using plugins:
- File format parsers/writers
- Database engine connectors
## How Can I Contribute?
You're welcome to contribute to this project! Below are some ideas, how to contribute:
* Create plugins for new import/export formats or database engines
* Bug fixing
* Test Mac edition
* Create unit tests
* Whatever else
Any help is appreciated!
Feel free to report issues and open merge requests.
## Roadmap
| Feature | Complexity | Schedule |
|---|---|---|
| Table designer (structure editor) | big | february 2021 |
| Support for SQLite | big | 2021 |
| Filtering, sorting in free table editor | small | ??? |
| Query designer | medium | december 2020 - done |
| Using tedious driver instead of mssql | small | january 2021 - done |
| Filter SQL result sets | small | november 2020 - done |
## How to run development environment
```sh
@@ -85,7 +70,7 @@ yarn lib
Open http://localhost:5000 in your browser
You could run electron app, using this server:
You could run electron app (requires running localhost:5000):
```sh
cd app
yarn
@@ -93,7 +78,7 @@ yarn start
```
## How to run built electron app locally
This mode is very similar to production run of electron app. Electron app forks process with API on dynamically allocated port, works with compiled javascript files.
This mode is very similar to production run of electron app. Electron app forks process with API on dynamically allocated port, works with compiled javascript files (doesn't use localhost:5000)
```sh
cd app
@@ -109,13 +94,12 @@ yarn start:app:local
## Packages
Some dbgate packages can be used also without DbGate. You can find them on [NPM repository](https://www.npmjs.com/search?q=keywords:dbgate)
* [api](https://github.com/dbshell/dbgate/tree/master/packages/api) - backend, Javascript, ExpressJS [![NPM version](https://img.shields.io/npm/v/dbgate-api.svg)](https://www.npmjs.com/package/dbgate-api)
* [datalib](https://github.com/dbshell/dbgate/tree/master/packages/datalib) - TypeScript library for utility classes
* [app](https://github.com/dbshell/dbgate/tree/master/app) - application (JavaScript)
structure, creating specific queries (JavaScript) [![NPM version](https://img.shields.io/npm/v/dbgate-engines.svg)](https://www.npmjs.com/package/dbgate-engines)
* [filterparser](https://github.com/dbshell/dbgate/tree/master/packages/filterparser) - TypeScript library for parsing data filter expressions using parsimmon
* [sqltree](https://github.com/dbshell/dbgate/tree/master/packages/sqltree) - JSON representation of SQL query, functions converting to SQL (TypeScript) [![NPM version](https://img.shields.io/npm/v/dbgate-sqltree.svg)](https://www.npmjs.com/package/dbgate-sqltree)
* [types](https://github.com/dbshell/dbgate/tree/master/packages/types) - common TypeScript definitions [![NPM version](https://img.shields.io/npm/v/dbgate-types.svg)](https://www.npmjs.com/package/dbgate-types)
* [web](https://github.com/dbshell/dbgate/tree/master/packages/web) - frontend in React (JavaScript)
* [tools](https://github.com/dbshell/dbgate/tree/master/packages/tools) - various tools [![NPM version](https://img.shields.io/npm/v/dbgate-tools.svg)](https://www.npmjs.com/package/dbgate-tools)
* [api](https://github.com/dbgate/dbgate/tree/master/packages/api) - backend, Javascript, ExpressJS [![NPM version](https://img.shields.io/npm/v/dbgate-api.svg)](https://www.npmjs.com/package/dbgate-api)
* [datalib](https://github.com/dbgate/dbgate/tree/master/packages/datalib) - TypeScript library for utility classes [![NPM version](https://img.shields.io/npm/v/dbgate-datalib.svg)](https://www.npmjs.com/package/dbgate-datalib)
* [app](https://github.com/dbgate/dbgate/tree/master/app) - application (JavaScript) structure, creating specific queries (JavaScript)
* [filterparser](https://github.com/dbgate/dbgate/tree/master/packages/filterparser) - TypeScript library for parsing data filter expressions using parsimmon [![NPM version](https://img.shields.io/npm/v/dbgate-filterparser.svg)](https://www.npmjs.com/package/dbgate-filterparser)
* [sqltree](https://github.com/dbgate/dbgate/tree/master/packages/sqltree) - JSON representation of SQL query, functions converting to SQL (TypeScript) [![NPM version](https://img.shields.io/npm/v/dbgate-sqltree.svg)](https://www.npmjs.com/package/dbgate-sqltree)
* [types](https://github.com/dbgate/dbgate/tree/master/packages/types) - common TypeScript definitions [![NPM version](https://img.shields.io/npm/v/dbgate-types.svg)](https://www.npmjs.com/package/dbgate-types)
* [web](https://github.com/dbgate/dbgate/tree/master/packages/web) - frontend in Svelte (JavaScript) [![NPM version](https://img.shields.io/npm/v/dbgate-web.svg)](https://www.npmjs.com/package/dbgate-web)
* [tools](https://github.com/dbgate/dbgate/tree/master/packages/tools) - various tools [![NPM version](https://img.shields.io/npm/v/dbgate-tools.svg)](https://www.npmjs.com/package/dbgate-tools)

View File

@@ -1,6 +1,6 @@
{
"name": "dbgate",
"version": "3.9.1-beta.1",
"version": "4.0.0",
"private": true,
"author": "Jan Prochazka <jenasoft.database@gmail.com>",
"description": "Opensource database administration tool",
@@ -11,7 +11,7 @@
},
"repository": {
"type": "git",
"url": "https://github.com/dbshell/dbgate.git"
"url": "https://github.com/dbgate/dbgate.git"
},
"build": {
"appId": "org.dbgate",
@@ -37,6 +37,15 @@
"github"
]
},
"snap": {
"publish": [
"github",
"snapStore"
],
"environment": {
"ELECTRON_SNAP": "true"
}
},
"win": {
"target": [
"nsis"
@@ -59,19 +68,19 @@
"start": "cross-env ELECTRON_START_URL=http://localhost:5000 electron .",
"start:local": "cross-env electron .",
"dist": "electron-builder",
"build": "cd ../packages/api && yarn build && cd ../web && yarn build:app && cd ../../app && yarn dist",
"build:local": "cd ../packages/api && yarn build && cd ../web && yarn build:app && cd ../../app && yarn predist",
"build": "cd ../packages/api && yarn build && cd ../web && yarn build && cd ../../app && yarn dist",
"build:local": "cd ../packages/api && yarn build && cd ../web && yarn build && cd ../../app && yarn predist",
"postinstall": "electron-builder install-app-deps",
"predist": "copyfiles ../packages/api/dist/* packages && copyfiles \"../packages/web/build/*\" packages && copyfiles \"../packages/web/build/**/*\" packages"
"predist": "copyfiles ../packages/api/dist/* packages && copyfiles \"../packages/web/public/*\" packages && copyfiles \"../packages/web/public/**/*\" packages"
},
"main": "src/electron.js",
"devDependencies": {
"copyfiles": "^2.2.0",
"cross-env": "^6.0.3",
"electron": "11.1.1",
"electron": "11.2.3",
"electron-builder": "22.9.1"
},
"optionalDependencies": {
"msnodesqlv8": "^2.0.10"
}
}
}

View File

@@ -1,6 +1,6 @@
const electron = require('electron');
const os = require('os');
const { Menu } = require('electron');
const { Menu, ipcMain } = require('electron');
const { fork } = require('child_process');
const { autoUpdater } = require('electron-updater');
const Store = require('electron-store');
@@ -20,12 +20,15 @@ const store = new Store();
// be closed automatically when the JavaScript object is garbage collected.
let mainWindow;
let splashWindow;
let mainMenu;
log.transports.file.level = 'debug';
autoUpdater.logger = log;
// TODO - create settings for this
// appUpdater.channel = 'beta';
let commands = {};
function hideSplash() {
if (splashWindow) {
splashWindow.destroy();
@@ -34,29 +37,48 @@ function hideSplash() {
mainWindow.show();
}
function commandItem(id) {
const command = commands[id];
return {
id,
label: command ? command.menuName || command.toolbarName || command.name : id,
accelerator: command ? command.keyText : undefined,
enabled: command ? command.enabled : false,
click() {
mainWindow.webContents.executeJavaScript(`dbgate_runCommand('${id}')`);
},
};
}
function buildMenu() {
const template = [
{
label: 'File',
submenu: [
{
label: 'Connect to database',
click() {
mainWindow.webContents.executeJavaScript(`dbgate_createNewConnection()`);
},
},
{
label: 'New query',
click() {
mainWindow.webContents.executeJavaScript(`dbgate_newQuery()`);
},
},
commandItem('new.connection'),
commandItem('file.open'),
commandItem('group.save'),
commandItem('group.saveAs'),
{ type: 'separator' },
{ role: 'close' },
],
},
{
label: 'Edit',
submenu: [{ role: 'copy' }, { role: 'paste' }],
label: 'Window',
submenu: [commandItem('new.query'), { type: 'separator' }, commandItem('tabs.closeAll'), { role: 'minimize' }],
},
// {
// label: 'Edit',
// submenu: [
// { role: 'undo' },
// { role: 'redo' },
// { type: 'separator' },
// { role: 'cut' },
// { role: 'copy' },
// { role: 'paste' },
// ],
// },
{
label: 'View',
submenu: [
@@ -69,20 +91,7 @@ function buildMenu() {
{ role: 'zoomout' },
{ type: 'separator' },
{ role: 'togglefullscreen' },
],
},
{
role: 'window',
submenu: [
{
label: 'Close all tabs',
click() {
mainWindow.webContents.executeJavaScript('dbgate_closeAll()');
},
},
{ type: 'separator' },
{ role: 'minimize' },
{ role: 'close' },
commandItem('theme.changeTheme'),
],
},
{
@@ -97,7 +106,7 @@ function buildMenu() {
{
label: 'DbGate on GitHub',
click() {
require('electron').shell.openExternal('https://github.com/dbshell/dbgate');
require('electron').shell.openExternal('https://github.com/dbgate/dbgate');
},
},
{
@@ -107,11 +116,12 @@ function buildMenu() {
},
},
{
label: 'About',
label: 'Report problem or feature request',
click() {
mainWindow.webContents.executeJavaScript(`dbgate_showAbout()`);
require('electron').shell.openExternal('https://github.com/dbgate/dbgate/issues/new');
},
},
commandItem('about.show'),
],
},
];
@@ -119,6 +129,24 @@ function buildMenu() {
return Menu.buildFromTemplate(template);
}
ipcMain.on('update-commands', async (event, arg) => {
commands = JSON.parse(arg);
for (const key of Object.keys(commands)) {
const menu = mainMenu.getMenuItemById(key);
if (!menu) continue;
const command = commands[key];
// rebuild menu
if (menu.label != command.text || menu.accelerator != command.keyText) {
mainMenu = buildMenu();
mainWindow.setMenu(mainMenu);
return;
}
menu.enabled = command.enabled;
}
});
function createWindow() {
const bounds = store.get('winBounds');
@@ -135,13 +163,14 @@ function createWindow() {
},
});
mainWindow.setMenu(buildMenu());
mainMenu = buildMenu();
mainWindow.setMenu(mainMenu);
function loadMainWindow() {
const startUrl =
process.env.ELECTRON_START_URL ||
url.format({
pathname: path.join(__dirname, '../packages/web/build/index.html'),
pathname: path.join(__dirname, '../packages/web/public/index.html'),
protocol: 'file:',
slashes: true,
});

View File

@@ -717,10 +717,10 @@ electron-updater@^4.3.5:
lodash.isequal "^4.5.0"
semver "^7.3.2"
electron@11.1.1:
version "11.1.1"
resolved "https://registry.yarnpkg.com/electron/-/electron-11.1.1.tgz#188f036f8282798398dca9513e9bb3b10213e3aa"
integrity sha512-tlbex3xosJgfileN6BAQRotevPRXB/wQIq48QeQ08tUJJrXwE72c8smsM/hbHx5eDgnbfJ2G3a60PmRjHU2NhA==
electron@11.2.3:
version "11.2.3"
resolved "https://registry.yarnpkg.com/electron/-/electron-11.2.3.tgz#8ad1d9858436cfca0e2e5ea7fea326794ae58ebb"
integrity sha512-6yxOc42nDAptHKNlUG/vcOh2GI9x2fqp2nQbZO0/3sz2CrwsJkwR3i3oMN9XhVJaqI7GK1vSCJz0verOkWlXcQ==
dependencies:
"@electron/get" "^1.0.1"
"@types/node" "^12.0.12"

View File

@@ -1,5 +1,6 @@
{
"private": true,
"version": "4.0.3-beta.1",
"name": "dbgate-all",
"workspaces": [
"packages/*"
@@ -8,7 +9,7 @@
"start:api": "yarn workspace dbgate-api start",
"start:api:portal": "yarn workspace dbgate-api start:portal",
"start:api:covid": "yarn workspace dbgate-api start:covid",
"start:web": "yarn workspace dbgate-web start",
"start:web": "yarn workspace dbgate-web dev",
"start:sqltree": "yarn workspace dbgate-sqltree start",
"start:tools": "yarn workspace dbgate-tools start",
"start:datalib": "yarn workspace dbgate-datalib start",
@@ -20,7 +21,7 @@
"build:lib": "yarn build:tools && yarn build:sqltree && yarn build:filterparser && yarn build:datalib",
"build:app": "cd app && yarn install && yarn build",
"build:api": "yarn workspace dbgate-api build",
"build:web:docker": "yarn workspace dbgate-web build:docker",
"build:web:docker": "yarn workspace dbgate-web build",
"build:app:local": "cd app && yarn build:local",
"start:app:local": "cd app && yarn start:local",
"setCurrentVersion": "node setCurrentVersion",
@@ -28,7 +29,7 @@
"fillNativeModules": "node fillNativeModules",
"fillNativeModulesElectron": "node fillNativeModules --eletron",
"prettier": "prettier --write packages/api/src && prettier --write packages/datalib/src && prettier --write packages/filterparser/src && prettier --write packages/sqltree/src && prettier --write packages/tools/src && prettier --write packages/types && prettier --write packages/web/src && prettier --write app/src",
"copy:docker:build": "copyfiles packages/api/dist/* docker -f && copyfiles packages/web/build/* docker -u 2 && copyfiles \"packages/web/build/**/*\" docker -u 2",
"copy:docker:build": "copyfiles packages/api/dist/* docker -f && copyfiles packages/web/public/* docker -u 2 && copyfiles \"packages/web/public/**/*\" docker -u 2",
"prepare:docker": "yarn build:web:docker && yarn build:api && yarn copy:docker:build",
"prepare": "yarn build:lib",
"start": "concurrently --kill-others-on-fail \"yarn start:api\" \"yarn start:web\"",

View File

@@ -5,19 +5,11 @@ SERVER_mysql=localhost
USER_mysql=root
PASSWORD_mysql=test
PORT_mysql=3307
ENGINE_mysql=mysql
ENGINE_mysql=mysql@dbgate-plugin-mysql
LABEL_postgres=Postgres localhost
SERVER_postgres=localhost
USER_postgres=postgres
PASSWORD_postgres=test
PORT_postgres=5433
ENGINE_postgres=postgres
TOOLBAR=home
ICON_home=mdi mdi-home
TITLE_home=Home
PAGE_home=home.html
STARTUP_PAGES=home
PAGES_DIRECTORY=/home/jena/jenasoft/dbgate-web/pages
ENGINE_postgres=postgres@dbgate-plugin-postgres

View File

@@ -32,7 +32,7 @@ dbgateApi.runScript(run);
```
Silly example, runs without any dependencies. Copy [fakeObjectReader](https://github.com/dbshell/dbgate/blob/master/packages/api/src/shell/fakeObjectReader.js) to [consoleObjectWriter](https://github.com/dbshell/dbgate/blob/master/packages/api/src/shell/consoleObjectWriter.js) .
Silly example, runs without any dependencies. Copy [fakeObjectReader](https://github.com/dbgate/dbgate/blob/master/packages/api/src/shell/fakeObjectReader.js) to [consoleObjectWriter](https://github.com/dbgate/dbgate/blob/master/packages/api/src/shell/consoleObjectWriter.js) .
```javascript

View File

@@ -1,15 +1,15 @@
{
"name": "dbgate-api",
"main": "src/index.js",
"version": "1.0.7",
"version": "4.0.0",
"homepage": "https://dbgate.org/",
"repository": {
"type": "git",
"url": "https://github.com/dbshell/dbgate.git"
"url": "https://github.com/dbgate/dbgate.git"
},
"funding": "https://www.paypal.com/paypalme/JanProchazkaCz/30eur",
"author": "Jan Prochazka",
"license": "GPL",
"license": "MIT",
"keywords": [
"sql",
"json",
@@ -26,8 +26,8 @@
"compare-versions": "^3.6.0",
"cors": "^2.8.5",
"cross-env": "^6.0.3",
"dbgate-sqltree": "^1.0.0",
"dbgate-tools": "^1.0.0",
"dbgate-sqltree": "^4.0.0",
"dbgate-tools": "^4.0.0",
"eslint": "^6.8.0",
"express": "^4.17.1",
"express-basic-auth": "^1.2.0",
@@ -35,17 +35,21 @@
"find-free-port": "^2.0.0",
"fs-extra": "^8.1.0",
"http": "^0.0.0",
"json-stable-stringify": "^1.0.1",
"line-reader": "^0.4.0",
"lodash": "^4.17.15",
"ncp": "^2.0.0",
"nedb-promises": "^4.0.1",
"node-cron": "^2.0.3",
"node-ssh-forward": "^0.7.2",
"portfinder": "^1.0.28",
"simple-encryptor": "^4.0.0",
"socket.io": "^2.3.0",
"tar": "^6.0.5",
"uuid": "^3.4.0"
},
"scripts": {
"start": "nodemon src/index.js",
"start": "node src/index.js",
"start:portal": "env-cmd nodemon src/index.js",
"start:covid": "env-cmd -f .covid-env nodemon src/index.js",
"ts": "tsc",
@@ -53,7 +57,7 @@
},
"devDependencies": {
"@types/lodash": "^4.14.149",
"dbgate-types": "^1.0.0",
"dbgate-types": "^4.0.0",
"env-cmd": "^10.1.0",
"node-loader": "^1.0.2",
"nodemon": "^2.0.2",
@@ -64,4 +68,4 @@
"optionalDependencies": {
"msnodesqlv8": "^2.0.10"
}
}
}

View File

@@ -1,4 +1,5 @@
const currentVersion = require('../currentVersion');
const platformInfo = require('../utility/platformInfo');
module.exports = {
get_meta: 'get',
@@ -31,4 +32,10 @@ module.exports = {
...currentVersion,
};
},
platformInfo_meta: 'get',
async platformInfo() {
return platformInfo;
},
};

View File

@@ -6,6 +6,7 @@ const nedb = require('nedb-promises');
const { datadir } = require('../utility/directories');
const socket = require('../utility/socket');
const { encryptConnection } = require('../utility/crypting');
const { handleProcessCommunication } = require('../utility/processComm');
function getPortalCollections() {
if (process.env.CONNECTIONS) {
@@ -47,6 +48,7 @@ module.exports = {
test(req, res) {
const subprocess = fork(process.argv[1], ['connectProcess', ...process.argv.slice(3)]);
subprocess.on('message', resp => {
if (handleProcessCommunication(resp, subprocess)) return;
// @ts-ignore
const { msgtype } = resp;
if (msgtype == 'connected' || msgtype == 'error') {

View File

@@ -3,6 +3,7 @@ const connections = require('./connections');
const socket = require('../utility/socket');
const { fork } = require('child_process');
const { DatabaseAnalyser } = require('dbgate-tools');
const { handleProcessCommunication } = require('../utility/processComm');
module.exports = {
/** @type {import('dbgate-types').OpenedDatabaseConnection[]} */
@@ -50,8 +51,10 @@ module.exports = {
status: { name: 'pending' },
};
this.opened.push(newOpened);
// @ts-ignore
subprocess.on('message', ({ msgtype, ...message }) => {
subprocess.on('message', message => {
// @ts-ignore
const { msgtype } = message;
if (handleProcessCommunication(message, subprocess)) return;
if (newOpened.disconnected) return;
this[`handle_${msgtype}`](conid, database, message);
});
@@ -89,6 +92,13 @@ module.exports = {
return res;
},
collectionData_meta: 'post',
async collectionData({ conid, database, options }) {
const opened = await this.ensureOpened(conid, database);
const res = await this.sendRequest(opened, { msgtype: 'collectionData', options });
return res.result;
},
status_meta: 'get',
async status({ conid, database }) {
const existing = this.opened.find(x => x.conid == conid && x.database == database);
@@ -103,10 +113,14 @@ module.exports = {
ping_meta: 'post',
async ping({ conid, database }) {
const existing = this.opened.find(x => x.conid == conid && x.database == database);
let existing = this.opened.find(x => x.conid == conid && x.database == database);
if (existing) {
existing.subprocess.send({ msgtype: 'ping' });
} else {
existing = await this.ensureOpened(conid, database);
}
return {
status: 'ok',
connectionStatus: existing ? existing.status : null,
@@ -150,6 +164,16 @@ module.exports = {
// };
},
sqlPreview_meta: 'post',
async sqlPreview({ conid, database, objects, options }) {
// wait for structure
await this.structure({ conid, database });
const opened = await this.ensureOpened(conid, database);
const res = await this.sendRequest(opened, { msgtype: 'sqlPreview', objects, options });
return res;
},
// runCommand_meta: 'post',
// async runCommand({ conid, database, sql }) {
// console.log(`Running SQL command , conid=${conid}, database=${database}, sql=${sql}`);

View File

@@ -78,6 +78,11 @@ module.exports = {
}
},
saveAs_meta: 'post',
async saveAs({ filePath, data, format }) {
await fs.writeFile(filePath, serialize(format, data));
},
favorites_meta: 'get',
async favorites() {
if (!hasPermission(`files/favorites/read`)) return [];

View File

@@ -121,7 +121,13 @@ module.exports = {
getStats_meta: 'get',
getStats({ jslid }) {
const file = `${getJslFileName(jslid)}.stats`;
if (fs.existsSync(file)) return JSON.parse(fs.readFileSync(file, 'utf-8'));
if (fs.existsSync(file)) {
try {
return JSON.parse(fs.readFileSync(file, 'utf-8'));
} catch (e) {
return {};
}
}
return {};
},

View File

@@ -17,7 +17,7 @@ module.exports = {
listObjects_meta: 'get',
async listObjects({ conid, database }) {
const opened = await databaseConnections.ensureOpened(conid, database);
const types = ['tables', 'views', 'procedures', 'functions', 'triggers'];
const types = ['tables', 'collections', 'views', 'procedures', 'functions', 'triggers'];
return types.reduce(
(res, type) => ({
...res,

View File

@@ -28,9 +28,9 @@ const hasPermission = require('../utility/hasPermission');
// }
const preinstallPluginMinimalVersions = {
'dbgate-plugin-mssql': '1.0.10',
'dbgate-plugin-mysql': '1.0.3',
'dbgate-plugin-postgres': '1.0.2',
'dbgate-plugin-mssql': '1.2.1',
'dbgate-plugin-mysql': '1.2.1',
'dbgate-plugin-postgres': '1.2.1',
'dbgate-plugin-csv': '1.0.8',
'dbgate-plugin-excel': '1.0.6',
};
@@ -149,7 +149,7 @@ module.exports = {
return content.commands[command](args);
},
authTypes_meta: 'post',
authTypes_meta: 'get',
async authTypes({ engine }) {
const packageName = extractPackageName(engine);
const content = requirePlugin(packageName);

View File

@@ -7,6 +7,7 @@ const socket = require('../utility/socket');
const { fork } = require('child_process');
const { rundir, uploadsdir, pluginsdir } = require('../utility/directories');
const { extractShellApiPlugins, extractShellApiFunctionName } = require('dbgate-tools');
const { handleProcessCommunication } = require('../utility/processComm');
function extractPlugins(script) {
const requireRegex = /\s*\/\/\s*@require\s+([^\s]+)\s*\n/g;
@@ -23,6 +24,7 @@ const requirePluginsTemplate = plugins =>
const scriptTemplate = script => `
const dbgateApi = require(process.env.DBGATE_API);
dbgateApi.initializeApiEnvironment();
${requirePluginsTemplate(extractPlugins(script))}
require=null;
async function run() {
@@ -35,6 +37,7 @@ dbgateApi.runScript(run);
const loaderScriptTemplate = (functionName, props, runid) => `
const dbgateApi = require(process.env.DBGATE_API);
dbgateApi.initializeApiEnvironment();
${requirePluginsTemplate(extractShellApiPlugins(functionName, props))}
require=null;
async function run() {
@@ -96,7 +99,8 @@ module.exports = {
cwd: directory,
stdio: ['ignore', 'pipe', 'pipe', 'ipc'],
env: {
DBGATE_API: process.argv[1],
...process.env,
DBGATE_API: global['dbgateApiModulePath'] || process.argv[1],
..._.fromPairs(pluginNames.map(name => [`PLUGIN_${_.camelCase(name)}`, path.join(pluginsdir(), name)])),
},
});
@@ -123,8 +127,10 @@ module.exports = {
subprocess,
};
this.opened.push(newOpened);
// @ts-ignore
subprocess.on('message', ({ msgtype, ...message }) => {
subprocess.on('message', message => {
// @ts-ignore
const { msgtype } = message;
if (handleProcessCommunication(message, subprocess)) return;
this[`handle_${msgtype}`](runid, message);
});
return newOpened;

View File

@@ -3,11 +3,13 @@ const socket = require('../utility/socket');
const { fork } = require('child_process');
const _ = require('lodash');
const AsyncLock = require('async-lock');
const { handleProcessCommunication } = require('../utility/processComm');
const lock = new AsyncLock();
module.exports = {
opened: [],
closed: {},
lastPinged: {},
handle_databases(conid, { databases }) {
const existing = this.opened.find(x => x.conid == conid);
@@ -42,8 +44,10 @@ module.exports = {
this.opened.push(newOpened);
delete this.closed[conid];
socket.emitChanged(`server-status-changed`);
// @ts-ignore
subprocess.on('message', ({ msgtype, ...message }) => {
subprocess.on('message', message => {
// @ts-ignore
const { msgtype } = message;
if (handleProcessCommunication(message, subprocess)) return;
if (newOpened.disconnected) return;
this[`handle_${msgtype}`](conid, message);
});
@@ -88,7 +92,12 @@ module.exports = {
ping_meta: 'post',
async ping({ connections }) {
await Promise.all(
connections.map(async conid => {
_.uniq(connections).map(async conid => {
const last = this.lastPinged[conid];
if (last && new Date().getTime() - last < 30 * 1000) {
return Promise.resolve();
}
this.lastPinged[conid] = new Date().getTime();
const opened = await this.ensureOpened(conid);
opened.subprocess.send({ msgtype: 'ping' });
})

View File

@@ -4,6 +4,7 @@ const connections = require('./connections');
const socket = require('../utility/socket');
const { fork } = require('child_process');
const jsldata = require('./jsldata');
const { handleProcessCommunication } = require('../utility/processComm');
module.exports = {
/** @type {import('dbgate-types').OpenedSession[]} */
@@ -73,8 +74,10 @@ module.exports = {
sesid,
};
this.opened.push(newOpened);
// @ts-ignore
subprocess.on('message', ({ msgtype, ...message }) => {
subprocess.on('message', message => {
// @ts-ignore
const { msgtype } = message;
if (handleProcessCommunication(message, subprocess)) return;
this[`handle_${msgtype}`](sesid, message);
});
subprocess.send({ msgtype: 'connect', ...connection, database });

View File

@@ -1,4 +1,5 @@
module.exports = {
version: '3.8.6',
buildTime: '2020-12-10T11:14:01.053Z',
module.exports = {
version: '4.0.0',
buildTime: '2021-04-01T10:48:22.253Z'
};

View File

@@ -1,10 +1,4 @@
const shell = require('./shell');
// require('socket.io-client');
// "socket.io-client": "^2.3.0",
// "utf-8-validate": "^5.0.2",
// "uuid": "^3.4.0",
// "uws": "10.148.1"
const argument = process.argv[2];
if (argument && argument.endsWith('Process')) {
@@ -18,4 +12,7 @@ if (argument && argument.endsWith('Process')) {
main.start(argument);
}
module.exports = shell;
module.exports = {
...shell,
getMainModule: () => require('./main'),
};

View File

@@ -8,6 +8,7 @@ const io = require('socket.io');
const fs = require('fs');
const findFreePort = require('find-free-port');
const childProcessChecker = require('./utility/childProcessChecker');
const path = require('path');
const useController = require('./utility/useController');
const socket = require('./utility/socket');
@@ -78,13 +79,15 @@ function start(argument = null) {
app.use('/runners/data', express.static(rundir()));
if (fs.existsSync('/home/dbgate-docker/build')) {
if (fs.existsSync('/home/dbgate-docker/public')) {
// server static files inside docker container
app.use(express.static('/home/dbgate-docker/build'));
app.use(express.static('/home/dbgate-docker/public'));
} else {
app.get('/', (req, res) => {
res.send('DbGate API');
});
if (argument != 'startNodeWeb') {
app.get('/', (req, res) => {
res.send('DbGate API');
});
}
}
if (argument == '--dynport') {
@@ -96,6 +99,13 @@ function start(argument = null) {
process.send({ msgtype: 'listening', port });
});
});
} else if (argument == 'startNodeWeb') {
app.use(express.static(path.join(__dirname, '../../dbgate-web/public')));
findFreePort(5000, function (err, port) {
server.listen(port, () => {
console.log(`DbGate API listening on port ${port}`);
});
});
} else {
server.listen(3000);
}

View File

@@ -1,13 +1,15 @@
const childProcessChecker = require('../utility/childProcessChecker');
const requireEngineDriver = require('../utility/requireEngineDriver');
const { decryptConnection } = require('../utility/crypting');
const connectUtility = require('../utility/connectUtility');
const { handleProcessCommunication } = require('../utility/processComm');
function start() {
childProcessChecker();
process.on('message', async connection => {
if (handleProcessCommunication(connection)) return;
try {
const driver = requireEngineDriver(connection);
const conn = await driver.connect(decryptConnection(connection));
const conn = await connectUtility(driver, connection);
const res = await driver.getVersion(conn);
process.send({ msgtype: 'connected', ...res });
} catch (e) {

View File

@@ -1,7 +1,9 @@
const stableStringify = require('json-stable-stringify');
const childProcessChecker = require('../utility/childProcessChecker');
const requireEngineDriver = require('../utility/requireEngineDriver');
const { decryptConnection } = require('../utility/crypting');
const connectUtility = require('../utility/connectUtility');
const { handleProcessCommunication } = require('../utility/processComm');
const { SqlGenerator } = require('dbgate-tools');
let systemConnection;
let storedConnection;
@@ -60,7 +62,7 @@ async function handleConnect({ connection, structure }) {
if (!structure) setStatusName('pending');
const driver = requireEngineDriver(storedConnection);
systemConnection = await checkedAsyncCall(driver.connect(decryptConnection(storedConnection)));
systemConnection = await checkedAsyncCall(connectUtility(driver, storedConnection));
if (structure) {
analysedStructure = structure;
handleIncrementalRefresh();
@@ -92,6 +94,38 @@ async function handleQueryData({ msgid, sql }) {
}
}
async function handleCollectionData({ msgid, options }) {
await waitConnected();
const driver = requireEngineDriver(storedConnection);
try {
const result = await driver.readCollection(systemConnection, options);
process.send({ msgtype: 'response', msgid, result });
} catch (err) {
process.send({ msgtype: 'response', msgid, errorMessage: err.message });
}
}
async function handleSqlPreview({ msgid, objects, options }) {
await waitConnected();
const driver = requireEngineDriver(storedConnection);
try {
const dmp = driver.createDumper();
const generator = new SqlGenerator(analysedStructure, options, objects, dmp, driver, systemConnection);
await generator.dump();
process.send({ msgtype: 'response', msgid, sql: dmp.s, isTruncated: generator.isTruncated });
if (generator.isUnhandledException) {
setTimeout(() => {
console.log('Exiting because of unhandled exception');
process.exit(0);
}, 500);
}
} catch (err) {
process.send({ msgtype: 'response', msgid, isError: true, errorMessage: err.message });
}
}
// async function handleRunCommand({ msgid, sql }) {
// await waitConnected();
// const driver = engines(storedConnection);
@@ -106,6 +140,8 @@ function handlePing() {
const messageHandlers = {
connect: handleConnect,
queryData: handleQueryData,
collectionData: handleCollectionData,
sqlPreview: handleSqlPreview,
ping: handlePing,
// runCommand: handleRunCommand,
};
@@ -127,6 +163,7 @@ function start() {
}, 60 * 1000);
process.on('message', async message => {
if (handleProcessCommunication(message)) return;
try {
await handleMessage(message);
} catch (e) {

View File

@@ -1,5 +1,6 @@
const childProcessChecker = require('../utility/childProcessChecker');
const JsonLinesDatastore = require('../utility/JsonLinesDatastore');
const { handleProcessCommunication } = require('../utility/processComm');
let lastPing = null;
let datastore = new JsonLinesDatastore();
@@ -47,6 +48,7 @@ function start() {
}, 60 * 1000);
process.on('message', async message => {
if (handleProcessCommunication(message)) return;
try {
await handleMessage(message);
} catch (e) {

View File

@@ -2,6 +2,8 @@ const stableStringify = require('json-stable-stringify');
const childProcessChecker = require('../utility/childProcessChecker');
const requireEngineDriver = require('../utility/requireEngineDriver');
const { decryptConnection } = require('../utility/crypting');
const connectUtility = require('../utility/connectUtility');
const { handleProcessCommunication } = require('../utility/processComm');
let systemConnection;
let storedConnection;
@@ -48,7 +50,7 @@ async function handleConnect(connection) {
const driver = requireEngineDriver(storedConnection);
try {
systemConnection = await driver.connect(decryptConnection(storedConnection));
systemConnection = await connectUtility(driver, storedConnection);
handleRefresh();
setInterval(handleRefresh, 30 * 1000);
} catch (err) {
@@ -67,7 +69,7 @@ function handlePing() {
async function handleCreateDatabase({ name }) {
const driver = requireEngineDriver(storedConnection);
systemConnection = await driver.connect(decryptConnection(storedConnection));
systemConnection = await connectUtility(driver, storedConnection);
console.log(`RUNNING SCRIPT: CREATE DATABASE ${driver.dialect.quoteIdentifier(name)}`);
await driver.query(systemConnection, `CREATE DATABASE ${driver.dialect.quoteIdentifier(name)}`);
await handleRefresh();
@@ -96,6 +98,7 @@ function start() {
}, 60 * 1000);
process.on('message', async message => {
if (handleProcessCommunication(message)) return;
try {
await handleMessage(message);
} catch (err) {

View File

@@ -8,6 +8,8 @@ const goSplit = require('../utility/goSplit');
const { jsldir } = require('../utility/directories');
const requireEngineDriver = require('../utility/requireEngineDriver');
const { decryptConnection } = require('../utility/crypting');
const connectUtility = require('../utility/connectUtility');
const { handleProcessCommunication } = require('../utility/processComm');
let systemConnection;
let storedConnection;
@@ -131,7 +133,7 @@ async function handleConnect(connection) {
storedConnection = connection;
const driver = requireEngineDriver(storedConnection);
systemConnection = await driver.connect(decryptConnection(storedConnection));
systemConnection = await connectUtility(driver, storedConnection);
for (const [resolve] of afterConnectCallbacks) {
resolve();
}
@@ -182,6 +184,7 @@ async function handleMessage({ msgtype, ...other }) {
function start() {
childProcessChecker();
process.on('message', async message => {
if (handleProcessCommunication(message)) return;
try {
await handleMessage(message);
} catch (e) {

View File

@@ -1,12 +1,13 @@
const goSplit = require('../utility/goSplit');
const requireEngineDriver = require('../utility/requireEngineDriver');
const { decryptConnection } = require('../utility/crypting');
const connectUtility = require('../utility/connectUtility');
async function executeQuery({ connection, sql }) {
console.log(`Execute query ${sql}`);
const driver = requireEngineDriver(connection);
const pool = await driver.connect(decryptConnection(connection));
const pool = await connectUtility(driver, connection);
console.log(`Connected.`);
for (const sqlItem of goSplit(sql)) {

View File

@@ -17,6 +17,7 @@ const requirePlugin = require('./requirePlugin');
const download = require('./download');
const executeQuery = require('./executeQuery');
const loadFile = require('./loadFile');
const initializeApiEnvironment = require('./initializeApiEnvironment');
const dbgateApi = {
queryReader,
@@ -37,6 +38,7 @@ const dbgateApi = {
registerPlugins,
executeQuery,
loadFile,
initializeApiEnvironment,
};
requirePlugin.initializeDbgateApi(dbgateApi);

View File

@@ -0,0 +1,9 @@
const { handleProcessCommunication } = require('../utility/processComm');
async function initializeApiEnvironment() {
process.on('message', async message => {
handleProcessCommunication(message);
});
}
module.exports = initializeApiEnvironment;

View File

@@ -1,11 +1,12 @@
const requireEngineDriver = require('../utility/requireEngineDriver');
const { decryptConnection } = require('../utility/crypting');
const connectUtility = require('../utility/connectUtility');
async function queryReader({ connection, sql }) {
console.log(`Reading query ${sql}`);
const driver = requireEngineDriver(connection);
const pool = await driver.connect(decryptConnection(connection));
const pool = await connectUtility(driver, connection);
console.log(`Connected.`);
return await driver.readQuery(pool, sql);
}

View File

@@ -1,10 +1,11 @@
const { quoteFullName, fullNameToString } = require('dbgate-tools');
const requireEngineDriver = require('../utility/requireEngineDriver');
const { decryptConnection } = require('../utility/crypting');
const connectUtility = require('../utility/connectUtility');
async function tableReader({ connection, pureName, schemaName }) {
const driver = requireEngineDriver(connection);
const pool = await driver.connect(decryptConnection(connection));
const pool = await connectUtility(driver, connection);
console.log(`Connected.`);
const fullName = { pureName, schemaName };

View File

@@ -1,12 +1,13 @@
const { fullNameToString } = require('dbgate-tools');
const requireEngineDriver = require('../utility/requireEngineDriver');
const { decryptConnection } = require('../utility/crypting');
const connectUtility = require('../utility/connectUtility');
async function tableWriter({ connection, schemaName, pureName, ...options }) {
console.log(`Writing table ${fullNameToString({ schemaName, pureName })}`);
const driver = requireEngineDriver(connection);
const pool = await driver.connect(decryptConnection(connection));
const pool = await connectUtility(driver, connection);
console.log(`Connected.`);
return await driver.writeTable(pool, { schemaName, pureName }, options);
}

View File

@@ -1,5 +1,6 @@
const { fork } = require('child_process');
const uuidv1 = require('uuid/v1');
const { handleProcessCommunication } = require('./processComm');
class DatastoreProxy {
constructor(file) {
@@ -30,8 +31,11 @@ class DatastoreProxy {
if (!this.subprocess) {
this.subprocess = fork(process.argv[1], ['jslDatastoreProcess', ...process.argv.slice(3)]);
// @ts-ignore
this.subprocess.on('message', ({ msgtype, ...message }) => {
this.subprocess.on('message', message => {
// @ts-ignore
const { msgtype } = message;
if (handleProcessCommunication(message, this.subprocess)) return;
// if (this.disconnected) return;
this[`handle_${msgtype}`](message);
});

View File

@@ -0,0 +1,59 @@
const { SSHConnection } = require('node-ssh-forward');
const portfinder = require('portfinder');
const fs = require('fs-extra');
const { decryptConnection } = require('./crypting');
const { getSshTunnel } = require('./sshTunnel');
const { getSshTunnelProxy } = require('./sshTunnelProxy');
async function connectUtility(driver, storedConnection) {
const connection = {
...decryptConnection(storedConnection),
};
if (!connection.port && driver.defaultPort) connection.port = driver.defaultPort.toString();
if (connection.useSshTunnel) {
const tunnel = await getSshTunnelProxy(connection);
if (tunnel.state == 'error') {
throw new Error(tunnel.message);
}
connection.server = '127.0.0.1';
connection.port = tunnel.localPort;
}
// SSL functionality - copied from https://github.com/beekeeper-studio/beekeeper-studio
if (connection.useSsl) {
connection.ssl = {};
if (connection.sslCaFile) {
connection.ssl.ca = await fs.readFile(connection.sslCaFile);
}
if (connection.sslCertFile) {
connection.ssl.cert = await fs.readFile(connection.sslCertFile);
}
if (connection.sslKeyFile) {
connection.ssl.key = await fs.readFile(connection.sslKeyFile);
}
if (!connection.ssl.key && !connection.ssl.ca && !connection.ssl.cert) {
// TODO: provide this as an option in settings
// or per-connection as 'reject self-signed certs'
// How it works:
// if false, cert can be self-signed
// if true, has to be from a public CA
// Heroku certs are self-signed.
// if you provide ca/cert/key files, it overrides this
connection.ssl.rejectUnauthorized = false;
} else {
connection.ssl.rejectUnauthorized = connection.sslRejectUnauthorized;
}
}
const conn = await driver.connect(connection);
return conn;
}
module.exports = connectUtility;

View File

@@ -42,28 +42,42 @@ function getEncryptor() {
return _encryptor;
}
function encryptConnection(connection) {
function encryptPasswordField(connection, field) {
if (
connection &&
connection.password &&
!connection.password.startsWith('crypt:') &&
connection[field] &&
!connection[field].startsWith('crypt:') &&
connection.passwordMode != 'saveRaw'
) {
return {
...connection,
password: 'crypt:' + getEncryptor().encrypt(connection.password),
[field]: 'crypt:' + getEncryptor().encrypt(connection[field]),
};
}
return connection;
}
function decryptPasswordField(connection, field) {
if (connection && connection[field] && connection[field].startsWith('crypt:')) {
return {
...connection,
[field]: getEncryptor().decrypt(connection[field].substring('crypt:'.length)),
};
}
return connection;
}
function encryptConnection(connection) {
connection = encryptPasswordField(connection, 'password');
connection = encryptPasswordField(connection, 'sshPassword');
connection = encryptPasswordField(connection, 'sshKeyFilePassword');
return connection;
}
function decryptConnection(connection) {
if (connection && connection.password && connection.password.startsWith('crypt:')) {
return {
...connection,
password: getEncryptor().decrypt(connection.password.substring('crypt:'.length)),
};
}
connection = decryptPasswordField(connection, 'password');
connection = decryptPasswordField(connection, 'sshPassword');
connection = decryptPasswordField(connection, 'sshKeyFilePassword');
return connection;
}

View File

@@ -0,0 +1,27 @@
const fs = require('fs');
const os = require('os');
const path = require('path');
const p = process;
const platform = p.env.OS_OVERRIDE ? p.env.OS_OVERRIDE : p.platform;
const isWindows = platform === 'win32';
const isMac = platform === 'darwin';
const isLinux = platform === 'linux';
const isDocker = fs.existsSync('/home/dbgate-docker/build');
const platformInfo = {
isWindows,
isMac,
isLinux,
isDocker,
isSnap: p.env.ELECTRON_SNAP == 'true',
isPortable: isWindows && p.env.PORTABLE_EXECUTABLE_DIR,
isAppImage: p.env.DESKTOPINTEGRATION === 'AppImageLauncher',
sshAuthSock: p.env.SSH_AUTH_SOCK,
environment: process.env.NODE_ENV,
platform,
runningInWebpack: !!p.env.WEBPACK_DEV_SERVER_URL,
defaultKeyFile: path.join(os.homedir(), '.ssh/id_rsa'),
};
module.exports = platformInfo;

View File

@@ -0,0 +1,18 @@
const { handleGetSshTunnelRequest, handleGetSshTunnelResponse } = require('./sshTunnelProxy');
function handleProcessCommunication(message, subprocess) {
const { msgtype } = message;
if (msgtype == 'getsshtunnel-request') {
handleGetSshTunnelRequest(message, subprocess);
return true;
}
if (msgtype == 'getsshtunnel-response') {
handleGetSshTunnelResponse(message, subprocess);
return true;
}
return false;
}
module.exports = {
handleProcessCommunication,
};

View File

@@ -0,0 +1,82 @@
const { SSHConnection } = require('node-ssh-forward');
const fs = require('fs-extra');
const portfinder = require('portfinder');
const stableStringify = require('json-stable-stringify');
const _ = require('lodash');
const platformInfo = require('./platformInfo');
const sshConnectionCache = {};
const sshTunnelCache = {};
const CONNECTION_FIELDS = [
'sshHost',
'sshPort',
'sshLogin',
'sshPassword',
'sshMode',
'sshKeyFile',
'sshBastionHost',
'sshKeyFilePassword',
];
const TUNNEL_FIELDS = [...CONNECTION_FIELDS, 'server', 'port'];
async function getSshConnection(connection) {
const connectionCacheKey = stableStringify(_.pick(connection, CONNECTION_FIELDS));
if (sshConnectionCache[connectionCacheKey]) return sshConnectionCache[connectionCacheKey];
const sshConfig = {
endHost: connection.sshHost || '',
endPort: connection.sshPort || 22,
bastionHost: connection.sshBastionHost || '',
agentForward: connection.sshMode == 'agent',
passphrase: connection.sshMode == 'keyFile' ? connection.sshKeyFilePassword : undefined,
username: connection.sshLogin,
password: connection.sshMode == 'userPassword' ? connection.sshPassword : undefined,
agentSocket: connection.sshMode == 'agent' ? platformInfo.sshAuthSock : undefined,
privateKey:
connection.sshMode == 'keyFile' && connection.sshKeyFile ? await fs.readFile(connection.sshKeyFile) : undefined,
skipAutoPrivateKey: true,
noReadline: true,
};
const sshConn = new SSHConnection(sshConfig);
sshConnectionCache[connectionCacheKey] = sshConn;
return sshConn;
}
async function getSshTunnel(connection) {
const sshConn = await getSshConnection(connection);
const tunnelCacheKey = stableStringify(_.pick(connection, TUNNEL_FIELDS));
if (sshTunnelCache[tunnelCacheKey]) return sshTunnelCache[tunnelCacheKey];
const localPort = await portfinder.getPortPromise({ port: 10000, stopPort: 60000 });
// workaround for `getPortPromise` not releasing the port quickly enough
await new Promise(resolve => setTimeout(resolve, 500));
const tunnelConfig = {
fromPort: localPort,
toPort: connection.port,
toHost: connection.server,
};
try {
const tunnel = await sshConn.forward(tunnelConfig);
console.log(
`Created SSH tunnel to ${connection.sshHost}-${connection.server}:${connection.port}, using local port ${localPort}`
);
sshTunnelCache[tunnelCacheKey] = {
state: 'ok',
localPort,
};
return sshTunnelCache[tunnelCacheKey];
} catch (err) {
// error is not cached
return {
state: 'error',
message: err.message,
};
}
}
module.exports = {
getSshTunnel,
};

View File

@@ -0,0 +1,30 @@
const uuidv1 = require('uuid/v1');
const { getSshTunnel } = require('./sshTunnel');
const dispatchedMessages = {};
async function handleGetSshTunnelRequest({ msgid, connection }, subprocess) {
const response = await getSshTunnel(connection);
subprocess.send({ msgtype: 'getsshtunnel-response', msgid, response });
}
function handleGetSshTunnelResponse({ msgid, response }, subprocess) {
const { resolve } = dispatchedMessages[msgid];
delete dispatchedMessages[msgid];
resolve(response);
}
async function getSshTunnelProxy(connection) {
if (!process.send) return getSshTunnel(connection);
const msgid = uuidv1();
process.send({ msgtype: 'getsshtunnel-request', msgid, connection });
return new Promise((resolve, reject) => {
dispatchedMessages[msgid] = { resolve, reject };
});
}
module.exports = {
handleGetSshTunnelRequest,
handleGetSshTunnelResponse,
getSshTunnelProxy,
};

View File

@@ -1,5 +1,5 @@
{
"version": "1.0.0",
"version": "4.0.0",
"name": "dbgate-datalib",
"main": "lib/index.js",
"typings": "lib/index.d.ts",
@@ -12,11 +12,11 @@
"lib"
],
"dependencies": {
"dbgate-sqltree": "^1.0.0",
"dbgate-filterparser": "^1.0.0"
"dbgate-sqltree": "^4.0.0",
"dbgate-filterparser": "^4.0.0"
},
"devDependencies": {
"dbgate-types": "^1.0.0",
"dbgate-types": "^4.0.0",
"@types/node": "^13.7.0",
"typescript": "^3.7.5"
}

View File

@@ -347,5 +347,6 @@ export function changeSetInsertNewRow(changeSet: ChangeSet, name?: NamedObjectIn
}
export function changeSetContainsChanges(changeSet: ChangeSet) {
if (!changeSet) return false;
return changeSet.deletes.length > 0 || changeSet.updates.length > 0 || changeSet.inserts.length > 0;
}

View File

@@ -0,0 +1,127 @@
import { FormViewDisplay } from './FormViewDisplay';
import _ from 'lodash';
import { GridDisplay, ChangeCacheFunc, DisplayColumn, DisplayedColumnInfo, ChangeConfigFunc } from './GridDisplay';
import { TableInfo, EngineDriver, ViewInfo, ColumnInfo, NamedObjectInfo, DatabaseInfo } from 'dbgate-types';
import { GridConfig, GridCache, createGridCache } from './GridConfig';
import {
Expression,
Select,
treeToSql,
dumpSqlSelect,
mergeConditions,
Condition,
OrderByExpression,
} from 'dbgate-sqltree';
import { filterName } from './filterName';
import { TableGridDisplay } from './TableGridDisplay';
import stableStringify from 'json-stable-stringify';
import { ChangeSetFieldDefinition, ChangeSetRowDefinition } from './ChangeSet';
import { CollectionGridDisplay } from './CollectionGridDisplay';
export class CollectionFormViewDisplay extends FormViewDisplay {
// use utility functions from GridDisplay and publish result in FromViewDisplay interface
private gridDisplay: CollectionGridDisplay;
constructor(
public collectionName: NamedObjectInfo,
driver: EngineDriver,
config: GridConfig,
setConfig: ChangeConfigFunc,
cache: GridCache,
setCache: ChangeCacheFunc,
loadedRow: any
) {
super(config, setConfig, cache, setCache, driver);
this.gridDisplay = new CollectionGridDisplay(collectionName, driver, config, setConfig, cache, setCache, [loadedRow]);
this.isLoadedCorrectly = this.gridDisplay.isLoadedCorrectly && !!this.driver;
this.columns = [];
this.addDisplayColumns(this.gridDisplay.columns);
}
addDisplayColumns(columns: DisplayColumn[]) {
for (const col of columns) {
this.columns.push(col);
}
}
navigate(row) {
const formViewKey = this.extractKey(row);
this.setConfig(cfg => ({
...cfg,
formViewKey,
}));
}
isLoadedCurrentRow(row) {
if (!row) return false;
const formViewKey = this.extractKey(row);
return stableStringify(formViewKey) == stableStringify(this.config.formViewKey);
}
navigateRowQuery(commmand: 'begin' | 'previous' | 'next' | 'end') {
if (!this.driver) return null;
const select = this.gridDisplay.createSelect();
if (!select) return null;
const { primaryKey } = this.gridDisplay.baseTable;
function getOrderBy(direction): OrderByExpression[] {
return primaryKey.columns.map(({ columnName }) => ({
exprType: 'column',
columnName,
direction,
}));
}
select.topRecords = 1;
switch (commmand) {
case 'begin':
select.orderBy = getOrderBy('ASC');
break;
case 'end':
select.orderBy = getOrderBy('DESC');
break;
case 'previous':
select.orderBy = getOrderBy('DESC');
select.where = mergeConditions(select.where, this.getPrimaryKeyOperatorCondition('<'));
break;
case 'next':
select.orderBy = getOrderBy('ASC');
select.where = mergeConditions(select.where, this.getPrimaryKeyOperatorCondition('>'));
break;
}
const sql = treeToSql(this.driver, select, dumpSqlSelect);
return sql;
}
getChangeSetRow(row): ChangeSetRowDefinition {
if (!this.baseTable) return null;
return {
pureName: this.baseTable.pureName,
schemaName: this.baseTable.schemaName,
condition: this.extractKey(row),
};
}
getChangeSetField(row, uniqueName): ChangeSetFieldDefinition {
const col = this.columns.find(x => x.uniqueName == uniqueName);
if (!col) return null;
if (!this.baseTable) return null;
if (this.baseTable.pureName != col.pureName || this.baseTable.schemaName != col.schemaName) return null;
return {
...this.getChangeSetRow(row),
uniqueName: uniqueName,
columnName: col.columnName,
};
}
toggleExpandedColumn(uniqueName: string) {
this.gridDisplay.toggleExpandedColumn(uniqueName);
this.gridDisplay.reload();
}
isExpandedColumn(uniqueName: string) {
return this.gridDisplay.isExpandedColumn(uniqueName);
}
}

View File

@@ -0,0 +1,100 @@
import _ from 'lodash';
import { GridDisplay, ChangeCacheFunc, ChangeConfigFunc, DisplayColumn } from './GridDisplay';
import { EngineDriver, ViewInfo, ColumnInfo, CollectionInfo } from 'dbgate-types';
import { GridConfig, GridCache } from './GridConfig';
function getObjectKeys(obj) {
if (_.isArray(obj)) {
return Object.keys(obj)
.slice(0, 10)
.map(x => parseInt(x));
}
if (_.isPlainObject(obj)) {
return Object.keys(obj);
}
return [];
}
function createHeaderText(path) {
let res = `${path[0]}`;
for (let i = 1; i < path.length; i++) {
const name = path[i];
if (_.isNumber(name)) res += `[${name}]`;
else res += `.${name}`;
}
return res;
}
export class CollectionGridDisplay extends GridDisplay {
constructor(
public collection: CollectionInfo,
driver: EngineDriver,
config: GridConfig,
setConfig: ChangeConfigFunc,
cache: GridCache,
setCache: ChangeCacheFunc,
loadedRows
) {
super(config, setConfig, cache, setCache, driver);
this.columns = this.getDisplayColumns(loadedRows || []);
this.filterable = true;
this.sortable = true;
this.editable = false;
this.supportsReload = true;
this.isDynamicStructure = true;
}
getDisplayColumns(rows) {
const res = [];
for (const row of rows) {
this.getColumnsForObject([], row, res);
}
return (
res.map(col => ({
...col,
isChecked: this.isColumnChecked(col),
})) || []
);
}
getColumnsForObject(basePath, obj, res: any[]) {
for (const name of getObjectKeys(obj)) {
const uniqueName = [...basePath, name].join('.');
let column = res.find(x => x.uniqueName == uniqueName);
if (!column) {
column = this.getDisplayColumn(basePath, name);
if (basePath.length > 0) {
const lastIndex1 = _.findLastIndex(res, x => x.parentHeaderText.startsWith(column.parentHeaderText));
const lastIndex2 = _.findLastIndex(res, x => x.headerText == column.parentHeaderText);
// console.log(uniqueName, lastIndex1, lastIndex2);
if (lastIndex1 >= 0) res.splice(lastIndex1 + 1, 0, column);
else if (lastIndex2 >= 0) res.splice(lastIndex2 + 1, 0, column);
else res.push(column);
} else {
res.push(column);
}
}
if (_.isPlainObject(obj[name]) || _.isArray(obj[name])) {
column.isExpandable = true;
}
if (this.isExpandedColumn(column.uniqueName)) {
this.getColumnsForObject([...basePath, name], obj[name], res);
}
}
}
getDisplayColumn(basePath, columnName) {
const uniquePath = [...basePath, columnName];
const uniqueName = uniquePath.join('.');
return {
columnName,
headerText: createHeaderText(uniquePath),
uniqueName,
uniquePath,
isStructured: true,
parentHeaderText: createHeaderText(basePath),
filterType: 'mongo',
};
}
}

View File

@@ -18,9 +18,12 @@ export interface DisplayColumn {
autoIncrement?: boolean;
isPrimaryKey?: boolean;
foreignKey?: ForeignKeyInfo;
isExpandable?: boolean;
isChecked?: boolean;
hintColumnName?: string;
dataType?: string;
filterType?: boolean;
isStructured?: boolean;
}
export interface DisplayedColumnEx extends DisplayColumn {
@@ -59,6 +62,7 @@ export abstract class GridDisplay {
editable = false;
isLoadedCorrectly = true;
supportsReload = false;
isDynamicStructure = false;
setColumnVisibility(uniquePath: string[], isVisible: boolean) {
const uniqueName = uniquePath.join('.');
@@ -66,7 +70,7 @@ export abstract class GridDisplay {
this.includeInColumnSet('hiddenColumns', uniqueName, !isVisible);
} else {
this.includeInColumnSet('addedColumns', uniqueName, isVisible);
this.reload();
if (!this.isDynamicStructure) this.reload();
}
}

View File

@@ -19,4 +19,6 @@ export interface MacroDefinition {
export interface MacroSelectedCell {
column: string;
row: number;
rowData: any;
value: any;
}

View File

@@ -18,7 +18,7 @@ import stableStringify from 'json-stable-stringify';
import { ChangeSetFieldDefinition, ChangeSetRowDefinition } from './ChangeSet';
export class TableFormViewDisplay extends FormViewDisplay {
// use utility functions from GridDisplay and publish result in FromViewDisplat interface
// use utility functions from GridDisplay and publish result in FromViewDisplay interface
private gridDisplay: TableGridDisplay;
constructor(

View File

@@ -59,6 +59,7 @@ export class TableGridDisplay extends GridDisplay {
...col,
isChecked: this.isColumnChecked(col),
hintColumnName: col.foreignKey ? `hint_${col.uniqueName}` : null,
isExpandable: !!col.foreignKey,
})) || []
);
}

View File

@@ -11,3 +11,4 @@ export * from './MacroDefinition';
export * from './runMacro';
export * from './FormViewDisplay';
export * from './TableFormViewDisplay';
export * from './CollectionGridDisplay';

View File

@@ -4,6 +4,8 @@ import uuidv1 from 'uuid/v1';
import uuidv4 from 'uuid/v4';
import moment from 'moment';
import { MacroDefinition, MacroSelectedCell } from './MacroDefinition';
import { ChangeSet, setChangeSetValue } from './ChangeSet';
import { GridDisplay } from './GridDisplay';
const getMacroFunction = {
transformValue: code => `
@@ -183,3 +185,55 @@ export function runMacro(
}
return data;
}
export function compileMacroFunction(macro: MacroDefinition, errors = []) {
if (!macro) return null;
let func;
try {
func = eval(getMacroFunction[macro.type](macro.code));
return func;
} catch (err) {
errors.push(`Error compiling macro ${macro.name}: ${err.message}`);
return null;
}
}
export function runMacroOnValue(compiledFunc, macroArgs, value, rowIndex, row, column, errors = []) {
if (!compiledFunc) return value;
try {
const res = compiledFunc(value, macroArgs, modules, rowIndex, row, column);
return res;
} catch (err) {
errors.push(`Error processing column ${column} on row ${rowIndex}: ${err.message}`);
return value;
}
}
export function runMacroOnChangeSet(
macro: MacroDefinition,
macroArgs: {},
selectedCells: MacroSelectedCell[],
changeSet: ChangeSet,
display: GridDisplay
): ChangeSet {
const errors = [];
const compiledMacroFunc = compileMacroFunction(macro, errors);
if (!compiledMacroFunc) return null;
let res = changeSet;
for (const cell of selectedCells) {
const definition = display.getChangeSetField(cell.rowData, cell.column, undefined);
const macroResult = runMacroOnValue(
compiledMacroFunc,
macroArgs,
cell.value,
cell.row,
cell.rowData,
cell.column,
errors
);
res = setChangeSetValue(res, definition, macroResult);
}
return res;
}

28
packages/dbgate/README.md Normal file
View File

@@ -0,0 +1,28 @@
[![styled with prettier](https://img.shields.io/badge/styled_with-prettier-ff69b4.svg)](https://github.com/prettier/prettier)
[![Donate](https://img.shields.io/badge/donate-paypal-blue.svg)](https://paypal.me/JanProchazkaCz/30eur)
[![NPM version](https://img.shields.io/npm/v/dbgate.svg)](https://www.npmjs.com/package/dbgate)
# DbGate - database administration tool
DbGate is fast and easy to use database administration tool for MySQL, PostgreSQL, SQL Server.
## Install using npm
```sh
npm install -g dbgate
```
After installing, you can run dbgate with command:
```sh
dbgate
```
Then open http://localhost:5000 in your browser
## Download electron app
You can also download binary packages from https://dbgate.org . Or run from source code, as described on [github](https://github.com/dbgate/dbgate)
## Other dbgate packages
You can use some functionality of dbgate from your JavaScript code. See [dbgate-api](https://npmjs.com/dbgate-api) package.
## Screenshot
![Screenshot](https://raw.githubusercontent.com/dbgate/dbgate/master/screenshot.png)

7
packages/dbgate/bin/dbgate.js Executable file
View File

@@ -0,0 +1,7 @@
#!/usr/bin/env node
const dbgateApi = require('dbgate-api');
global.dbgateApiModulePath = require.resolve('dbgate-api');
dbgateApi.getMainModule().start('startNodeWeb');

View File

@@ -0,0 +1,25 @@
{
"name": "dbgate",
"version": "4.0.0",
"homepage": "https://dbgate.org/",
"repository": {
"type": "git",
"url": "https://github.com/dbgate/dbgate.git"
},
"funding": "https://www.paypal.com/paypalme/JanProchazkaCz/30eur",
"description": "Opensource database administration tool - web interface",
"author": "Jan Prochazka",
"license": "MIT",
"bin": {
"dbgate": "./bin/dbgate.js"
},
"keywords": [
"sql",
"dbgate",
"web"
],
"dependencies": {
"dbgate-api": "^4.0.0",
"dbgate-web": "^4.0.0"
}
}

View File

@@ -1,5 +1,5 @@
{
"version": "1.0.0",
"version": "4.0.0",
"name": "dbgate-filterparser",
"main": "lib/index.js",
"typings": "lib/index.d.ts",
@@ -13,7 +13,7 @@
"lib"
],
"devDependencies": {
"dbgate-types": "^1.0.0",
"dbgate-types": "^4.0.0",
"@types/jest": "^25.1.4",
"@types/node": "^13.7.0",
"jest": "^24.9.0",
@@ -22,9 +22,9 @@
},
"dependencies": {
"@types/parsimmon": "^1.10.1",
"dbgate-tools": "^1.0.0",
"dbgate-tools": "^4.0.0",
"lodash": "^4.17.15",
"moment": "^2.24.0",
"parsimmon": "^1.13.0"
}
}
}

View File

@@ -0,0 +1,32 @@
import P from 'parsimmon';
export const whitespace = P.regexp(/\s*/m);
export function token(parser) {
return parser.skip(whitespace);
}
export function word(str) {
return P.string(str).thru(token);
}
export function interpretEscapes(str) {
let escapes = {
b: '\b',
f: '\f',
n: '\n',
r: '\r',
t: '\t',
};
return str.replace(/\\(u[0-9a-fA-F]{4}|[^u])/, (_, escape) => {
let type = escape.charAt(0);
let hex = escape.slice(1);
if (type === 'u') {
return String.fromCharCode(parseInt(hex, 16));
}
if (escapes.hasOwnProperty(type)) {
return escapes[type];
}
return type;
});
}

View File

@@ -0,0 +1,121 @@
import P from 'parsimmon';
import { interpretEscapes, token, word, whitespace } from './common';
const operatorCondition = operator => value => ({
__placeholder__: {
[operator]: value,
},
});
const regexCondition = regexString => value => ({
__placeholder__: {
$regex: regexString.replace('#VALUE#', value),
$options: 'i',
},
});
const numberTestCondition = () => value => ({
$or: [
{
__placeholder__: {
$regex: `.*${value}.*`,
$options: 'i',
},
},
{
__placeholder__: value,
},
],
});
const testCondition = (operator, value) => () => ({
__placeholder__: {
[operator]: value,
},
});
const compoudCondition = conditionType => conditions => {
if (conditions.length == 1) return conditions[0];
return {
[conditionType]: conditions,
};
};
const negateCondition = condition => ({
__placeholder__: {
$not: condition.__placeholder__,
},
});
const createParser = () => {
const langDef = {
string1: () =>
token(P.regexp(/"((?:\\.|.)*?)"/, 1))
.map(interpretEscapes)
.desc('string quoted'),
string2: () =>
token(P.regexp(/'((?:\\.|.)*?)'/, 1))
.map(interpretEscapes)
.desc('string quoted'),
number: () =>
token(P.regexp(/-?(0|[1-9][0-9]*)([.][0-9]+)?([eE][+-]?[0-9]+)?/))
.map(Number)
.desc('number'),
noQuotedString: () => P.regexp(/[^\s^,^'^"]+/).desc('string unquoted'),
value: r => P.alt(r.string1, r.string2, r.number, r.noQuotedString),
valueTestNum: r => r.number.map(numberTestCondition()),
valueTest: r => r.value.map(regexCondition('.*#VALUE#.*')),
comma: () => word(','),
not: () => word('NOT'),
notExists: r => r.not.then(r.exists).map(testCondition('$exists', false)),
exists: () => word('EXISTS').map(testCondition('$exists', true)),
true: () => word('TRUE').map(testCondition('$eq', true)),
false: () => word('FALSE').map(testCondition('$eq', false)),
eq: r => word('=').then(r.value).map(operatorCondition('$eq')),
ne: r => word('!=').then(r.value).map(operatorCondition('$ne')),
lt: r => word('<').then(r.value).map(operatorCondition('$lt')),
gt: r => word('>').then(r.value).map(operatorCondition('$gt')),
le: r => word('<=').then(r.value).map(operatorCondition('$lte')),
ge: r => word('>=').then(r.value).map(operatorCondition('$gte')),
startsWith: r => word('^').then(r.value).map(regexCondition('#VALUE#.*')),
endsWith: r => word('$').then(r.value).map(regexCondition('.*#VALUE#')),
contains: r => word('+').then(r.value).map(regexCondition('.*#VALUE#.*')),
startsWithNot: r => word('!^').then(r.value).map(regexCondition('#VALUE#.*')).map(negateCondition),
endsWithNot: r => word('!$').then(r.value).map(regexCondition('.*#VALUE#')).map(negateCondition),
containsNot: r => word('~').then(r.value).map(regexCondition('.*#VALUE#.*')).map(negateCondition),
element: r =>
P.alt(
r.exists,
r.notExists,
r.true,
r.false,
r.eq,
r.ne,
r.lt,
r.gt,
r.le,
r.ge,
r.startsWith,
r.endsWith,
r.contains,
r.startsWithNot,
r.endsWithNot,
r.containsNot,
r.valueTestNum,
r.valueTest
).trim(whitespace),
factor: r => r.element.sepBy(whitespace).map(compoudCondition('$and')),
list: r => r.factor.sepBy(r.comma).map(compoudCondition('$or')),
};
return P.createLanguage(langDef);
};
export const mongoParser = createParser();

View File

@@ -3,37 +3,8 @@ import moment from 'moment';
import { FilterType } from './types';
import { Condition } from 'dbgate-sqltree';
import { TransformType } from 'dbgate-types';
const whitespace = P.regexp(/\s*/m);
function token(parser) {
return parser.skip(whitespace);
}
function word(str) {
return P.string(str).thru(token);
}
function interpretEscapes(str) {
let escapes = {
b: '\b',
f: '\f',
n: '\n',
r: '\r',
t: '\t',
};
return str.replace(/\\(u[0-9a-fA-F]{4}|[^u])/, (_, escape) => {
let type = escape.charAt(0);
let hex = escape.slice(1);
if (type === 'u') {
return String.fromCharCode(parseInt(hex, 16));
}
if (escapes.hasOwnProperty(type)) {
return escapes[type];
}
return type;
});
}
import { interpretEscapes, token, word, whitespace } from './common';
import { mongoParser } from './mongoParser';
const binaryCondition = operator => value => ({
conditionType: 'binary',
@@ -239,7 +210,8 @@ const createParser = (filterType: FilterType) => {
yearMonthNum: () => P.regexp(/\d\d\d\d-\d\d?/).map(yearMonthCondition()),
yearMonthDayNum: () => P.regexp(/\d\d\d\d-\d\d?-\d\d?/).map(yearMonthDayCondition()),
yearMonthDayMinute: () => P.regexp(/\d\d\d\d-\d\d?-\d\d?\s+\d\d?:\d\d?/).map(yearMonthDayMinuteCondition()),
yearMonthDaySecond: () => P.regexp(/\d\d\d\d-\d\d?-\d\d?(\s+|T)\d\d?:\d\d?:\d\d?/).map(yearMonthDaySecondCondition()),
yearMonthDaySecond: () =>
P.regexp(/\d\d\d\d-\d\d?-\d\d?(\s+|T)\d\d?:\d\d?:\d\d?/).map(yearMonthDaySecondCondition()),
value: r => P.alt(...allowedValues.map(x => r[x])),
valueTestEq: r => r.value.map(binaryCondition('=')),
@@ -348,9 +320,11 @@ const parsers = {
string: createParser('string'),
datetime: createParser('datetime'),
logical: createParser('logical'),
mongo: mongoParser,
};
export function parseFilter(value: string, filterType: FilterType): Condition {
// console.log('PARSING', value, 'WITH', filterType);
const ast = parsers[filterType].list.tryParse(value);
// console.log('AST', ast);
return ast;

View File

@@ -1,3 +1,3 @@
// import types from 'dbgate-types';
export type FilterType = 'number' | 'string' | 'datetime' | 'logical';
export type FilterType = 'number' | 'string' | 'datetime' | 'logical' | 'mongo';

View File

@@ -37,7 +37,7 @@ console.log("Generated query:", sql);
```
See [TypeScript definitions](https://github.com/dbshell/dbgate/blob/master/packages/sqltree/src/types.ts) for complete list of available SQL command options.
See [TypeScript definitions](https://github.com/dbgate/dbgate/blob/master/packages/sqltree/src/types.ts) for complete list of available SQL command options.
## Installation

View File

@@ -1,16 +1,16 @@
{
"version": "1.0.4",
"version": "4.0.0",
"name": "dbgate-sqltree",
"main": "lib/index.js",
"typings": "lib/index.d.ts",
"homepage": "https://dbgate.org/",
"repository": {
"type": "git",
"url": "https://github.com/dbshell/dbgate.git"
"url": "https://github.com/dbgate/dbgate.git"
},
"funding": "https://www.paypal.com/paypalme/JanProchazkaCz/30eur",
"author": "Jan Prochazka",
"license": "GPL",
"license": "MIT",
"keywords": [
"sql",
"mssql",
@@ -29,10 +29,10 @@
],
"devDependencies": {
"@types/node": "^13.7.0",
"dbgate-types": "^1.0.0",
"dbgate-types": "^4.0.0",
"typescript": "^3.7.5"
},
"dependencies": {
"lodash": "^4.17.15"
}
}
}

View File

@@ -1,16 +1,16 @@
{
"version": "1.0.7",
"version": "4.0.3-rc.1",
"name": "dbgate-tools",
"main": "lib/index.js",
"typings": "lib/index.d.ts",
"homepage": "https://dbgate.org/",
"repository": {
"type": "git",
"url": "https://github.com/dbshell/dbgate.git"
"url": "https://github.com/dbgate/dbgate.git"
},
"funding": "https://www.paypal.com/paypalme/JanProchazkaCz/30eur",
"author": "Jan Prochazka",
"license": "GPL",
"license": "MIT",
"keywords": [
"sql",
"dbgate"
@@ -27,7 +27,7 @@
],
"devDependencies": {
"@types/node": "^13.7.0",
"dbgate-types": "^1.0.0",
"dbgate-types": "^4.0.0",
"jest": "^24.9.0",
"ts-jest": "^25.2.1",
"typescript": "^3.7.5"

View File

@@ -40,7 +40,7 @@ export class DatabaseAnalyser {
return this._runAnalysis();
}
mergeAnalyseResult(newlyAnalysed, extractObjectId) {
mergeAnalyseResult(newlyAnalysed) {
if (this.structure == null) {
return {
...DatabaseAnalyser.createEmptyStructure(),
@@ -49,15 +49,15 @@ export class DatabaseAnalyser {
}
const res = {};
for (const field of ['tables', 'views', 'functions', 'procedures', 'triggers']) {
for (const field of ['tables', 'collections', 'views', 'functions', 'procedures', 'triggers']) {
const removedIds = this.modifications
.filter(x => x.action == 'remove' && x.objectTypeField == field)
.map(x => extractObjectId(x));
.map(x => x.objectId);
const newArray = newlyAnalysed[field] || [];
const addedChangedIds = newArray.map(x => extractObjectId(x));
const addedChangedIds = newArray.map(x => x.objectId);
const removeAllIds = [...removedIds, ...addedChangedIds];
res[field] = _sortBy(
[...this.structure[field].filter(x => !removeAllIds.includes(extractObjectId(x))), ...newArray],
[...this.structure[field].filter(x => !removeAllIds.includes(x.objectId)), ...newArray],
x => x.pureName
);
}
@@ -78,6 +78,7 @@ export class DatabaseAnalyser {
static createEmptyStructure(): DatabaseInfo {
return {
tables: [],
collections: [],
views: [],
functions: [],
procedures: [],

View File

@@ -1,11 +1,20 @@
import {
ColumnInfo,
ConstraintInfo,
EngineDriver,
ForeignKeyInfo,
FunctionInfo,
NamedObjectInfo,
PrimaryKeyInfo,
ProcedureInfo,
SqlDialect,
TableInfo,
TransformType,
TriggerInfo,
ViewInfo,
IndexInfo,
UniqueInfo,
CheckInfo,
} from 'dbgate-types';
import _isString from 'lodash/isString';
import _isNumber from 'lodash/isNumber';
@@ -46,11 +55,12 @@ export class SqlDumper {
}
putValue(value) {
if (value === null) this.putRaw('NULL');
if (value === true) this.putRaw('1');
if (value === false) this.putRaw('0');
else if (value === true) this.putRaw('1');
else if (value === false) this.putRaw('0');
else if (_isString(value)) this.putStringValue(value);
else if (_isNumber(value)) this.putRaw(value.toString());
else if (_isDate(value)) this.putStringValue(new Date(value).toISOString());
else this.putRaw('NULL');
}
putCmd(format, ...args) {
this.put(format, ...args);
@@ -260,4 +270,189 @@ export class SqlDumper {
}
allowIdentityInsert(table: NamedObjectInfo, allow: boolean) {}
enableConstraints(table: NamedObjectInfo, enabled: boolean) {}
comment(value: string) {
if (!value) return;
for (const line of value.split('\n')) {
this.put(' -- %s', line.trimRight());
}
}
createView(obj: ViewInfo) {
this.putRaw(obj.createSql);
this.endCommand();
}
dropView(obj: ViewInfo, { testIfExists = false }) {
this.putCmd('^drop ^view %f', obj);
}
alterView(obj: ViewInfo) {
this.putRaw(obj.createSql.replace(/create\s+view/i, 'ALTER VIEW'));
this.endCommand();
}
changeViewSchema(obj: ViewInfo, newSchema: string) {}
renameView(obj: ViewInfo, newSchema: string) {}
createProcedure(obj: ProcedureInfo) {
this.putRaw(obj.createSql);
this.endCommand();
}
dropProcedure(obj: ProcedureInfo, { testIfExists = false }) {
this.putCmd('^drop ^procedure %f', obj);
}
alterProcedure(obj: ProcedureInfo) {
this.putRaw(obj.createSql.replace(/create\s+procedure/i, 'ALTER PROCEDURE'));
this.endCommand();
}
changeProcedureSchema(obj: ProcedureInfo, newSchema: string) {}
renameProcedure(obj: ProcedureInfo, newSchema: string) {}
createFunction(obj: FunctionInfo) {
this.putRaw(obj.createSql);
this.endCommand();
}
dropFunction(obj: FunctionInfo, { testIfExists = false }) {
this.putCmd('^drop ^function %f', obj);
}
alterFunction(obj: FunctionInfo) {
this.putRaw(obj.createSql.replace(/create\s+function/i, 'ALTER FUNCTION'));
this.endCommand();
}
changeFunctionSchema(obj: FunctionInfo, newSchema: string) {}
renameFunction(obj: FunctionInfo, newSchema: string) {}
createTrigger(obj: TriggerInfo) {
this.putRaw(obj.createSql);
this.endCommand();
}
dropTrigger(obj: TriggerInfo, { testIfExists = false }) {
this.putCmd('^drop ^trigger %f', obj);
}
alterTrigger(obj: TriggerInfo) {
this.putRaw(obj.createSql.replace(/create\s+trigger/i, 'ALTER TRIGGER'));
this.endCommand();
}
changeTriggerSchema(obj: TriggerInfo, newSchema: string) {}
renameTrigger(obj: TriggerInfo, newSchema: string) {}
dropConstraint(cnt: ConstraintInfo) {
this.putCmd('^alter ^table %f ^drop ^constraint %i', cnt, cnt.constraintName);
}
dropForeignKey(fk: ForeignKeyInfo) {
if (this.dialect.explicitDropConstraint) {
this.putCmd('^alter ^table %f ^drop ^foreign ^key %i', fk, fk.constraintName);
} else {
this.dropConstraint(fk);
}
}
createForeignKey(fk: ForeignKeyInfo) {
this.put('^alter ^table %f ^add ', fk);
this.createForeignKeyFore(fk);
this.endCommand();
}
dropPrimaryKey(pk: PrimaryKeyInfo) {
if (this.dialect.explicitDropConstraint) {
this.putCmd('^alter ^table %f ^drop ^primary ^key', pk);
} else {
this.dropConstraint(pk);
}
}
createPrimaryKey(pk: PrimaryKeyInfo) {
this.putCmd(
'^alter ^table %f ^add ^constraint %i ^primary ^key (%,i)',
pk,
pk.constraintName,
pk.columns.map(x => x.columnName)
);
}
dropIndex(ix: IndexInfo) {}
createIndex(ix: IndexInfo) {}
dropUnique(uq: UniqueInfo) {
this.dropConstraint(uq);
}
createUniqueCore(uq: UniqueInfo) {
this.put(
'^constraint %i ^unique (%,i)',
uq.constraintName,
uq.columns.map(x => x.columnName)
);
}
createUnique(uq: UniqueInfo) {
this.put('^alter ^table %f ^add ', uq);
this.createUniqueCore(uq);
this.endCommand();
}
dropCheck(ch: CheckInfo) {
this.dropConstraint(ch);
}
createCheckCore(ch: CheckInfo) {
this.put('^constraint %i ^check (%s)', ch.constraintName, ch.definition);
}
createCheck(ch: CheckInfo) {
this.put('^alter ^table %f ^add ', ch);
this.createCheckCore(ch);
this.endCommand();
}
renameConstraint(constraint: ConstraintInfo, newName: string) {}
createColumn(table: TableInfo, column: ColumnInfo, constraints: ConstraintInfo[]) {
this.put('^alter ^table %f ^add %i ', table, column.columnName);
this.columnDefinition(column);
this.inlineConstraints(constraints);
this.endCommand();
}
inlineConstraints(constrains: ConstraintInfo[]) {
if (constrains == null) return;
for (const cnt of constrains) {
if (cnt.constraintType == 'primaryKey') {
if (cnt.constraintName != null && !this.dialect.anonymousPrimaryKey) {
this.put(' ^constraint %i', cnt.constraintName);
}
this.put(' ^primary ^key ');
}
}
}
dropColumn(column: ColumnInfo) {
this.putCmd('^alter ^table %f ^drop ^column %i', column, column.columnName);
}
renameColumn(column: ColumnInfo, newName: string) {}
changeColumn(oldcol: ColumnInfo, newcol: ColumnInfo, constraints: ConstraintInfo[]) {}
dropTable(obj: TableInfo, { testIfExists = false }) {
this.putCmd('^drop ^table %f', obj);
}
changeTableSchema(obj: TableInfo, schema: string) {}
renameTable(obj: TableInfo, newname: string) {}
beginTransaction() {
this.putCmd('^begin ^transaction');
}
commitTransaction() {
this.putCmd('^commit');
}
alterProlog() {}
alterEpilog() {}
selectTableIntoNewTable(sourceName: NamedObjectInfo, targetName: NamedObjectInfo) {
this.putCmd('^select * ^into %f ^from %f', targetName, sourceName);
}
truncateTable(name: NamedObjectInfo) {
this.putCmd('^delete ^from %f', name);
}
}

View File

@@ -0,0 +1,292 @@
import {
DatabaseInfo,
EngineDriver,
FunctionInfo,
ProcedureInfo,
TableInfo,
TriggerInfo,
ViewInfo,
} from 'dbgate-types';
import _ from 'lodash';
import { SqlDumper } from './SqlDumper';
import { extendDatabaseInfo } from './structureTools';
interface SqlGeneratorOptions {
dropTables: boolean;
checkIfTableExists: boolean;
dropReferences: boolean;
createTables: boolean;
createReferences: boolean;
createForeignKeys: boolean;
createIndexes: boolean;
insert: boolean;
skipAutoincrementColumn: boolean;
disableConstraints: boolean;
omitNulls: boolean;
truncate: boolean;
dropViews: boolean;
checkIfViewExists: boolean;
createViews: boolean;
dropProcedures: boolean;
checkIfProcedureExists: boolean;
createProcedures: boolean;
dropFunctions: boolean;
checkIfFunctionExists: boolean;
createFunctions: boolean;
dropTriggers: boolean;
checkIfTriggerExists: boolean;
createTriggers: boolean;
}
interface SqlGeneratorObject {
schemaName: string;
pureName: string;
objectTypeField: 'tables' | 'views' | 'procedures' | 'functions';
}
export class SqlGenerator {
private tables: TableInfo[];
private views: ViewInfo[];
private procedures: ProcedureInfo[];
private functions: FunctionInfo[];
private triggers: TriggerInfo[];
public dbinfo: DatabaseInfo;
public isTruncated = false;
public isUnhandledException = false;
constructor(
dbinfo: DatabaseInfo,
public options: SqlGeneratorOptions,
public objects: SqlGeneratorObject[],
public dmp: SqlDumper,
public driver: EngineDriver,
public pool
) {
this.dbinfo = extendDatabaseInfo(dbinfo);
this.tables = this.extract('tables');
this.views = this.extract('views');
this.procedures = this.extract('procedures');
this.functions = this.extract('functions');
this.triggers = this.extract('triggers');
}
private handleException = error => {
console.log('Unhandled error', error);
this.isUnhandledException = true;
};
async dump() {
try {
process.on('uncaughtException', this.handleException);
this.dropObjects(this.procedures, 'Procedure');
if (this.checkDumper()) return;
this.dropObjects(this.functions, 'Function');
if (this.checkDumper()) return;
this.dropObjects(this.views, 'View');
if (this.checkDumper()) return;
this.dropObjects(this.triggers, 'Trigger');
if (this.checkDumper()) return;
this.dropTables();
if (this.checkDumper()) return;
this.createTables();
if (this.checkDumper()) return;
this.truncateTables();
if (this.checkDumper()) return;
await this.insertData();
if (this.checkDumper()) return;
this.createForeignKeys();
if (this.checkDumper()) return;
this.createObjects(this.procedures, 'Procedure');
if (this.checkDumper()) return;
this.createObjects(this.functions, 'Function');
if (this.checkDumper()) return;
this.createObjects(this.views, 'View');
if (this.checkDumper()) return;
this.createObjects(this.triggers, 'Trigger');
if (this.checkDumper()) return;
} finally {
process.off('uncaughtException', this.handleException);
}
}
createForeignKeys() {
const fks = [];
if (this.options.createForeignKeys) fks.push(..._.flatten(this.tables.map(x => x.foreignKeys || [])));
if (this.options.createReferences) fks.push(..._.flatten(this.tables.map(x => x.dependencies || [])));
for (const fk of _.uniqBy(fks, 'constraintName')) {
this.dmp.createForeignKey(fk);
if (this.checkDumper()) return;
}
}
truncateTables() {
if (this.options.truncate) {
for (const table of this.tables) {
this.dmp.truncateTable(table);
if (this.checkDumper()) return;
}
}
}
createTables() {
if (this.options.createTables) {
for (const table of this.tables) {
this.dmp.createTable({
...table,
foreignKeys: [],
dependencies: [],
indexes: [],
});
if (this.checkDumper()) return;
}
}
if (this.options.createIndexes) {
for (const index of _.flatten(this.tables.map(x => x.indexes || []))) {
this.dmp.createIndex(index);
}
}
}
async insertData() {
if (!this.options.insert) return;
this.enableConstraints(false);
for (const table of this.tables) {
await this.insertTableData(table);
if (this.checkDumper()) return;
}
this.enableConstraints(true);
}
checkDumper() {
if (this.dmp.s.length > 4000000) {
if (!this.isTruncated) {
this.dmp.putRaw('\n');
this.dmp.comment(' *************** SQL is truncated ******************');
this.dmp.putRaw('\n');
}
this.isTruncated = true;
return true;
}
return false;
}
dropObjects(list, name) {
if (this.options[`drop${name}s`]) {
for (const item of list) {
this.dmp[`drop${name}`](item, { testIfExists: this.options[`checkIf${name}Exists`] });
if (this.checkDumper()) return;
}
}
}
createObjects(list, name) {
if (this.options[`create${name}s`]) {
for (const item of list) {
this.dmp[`create${name}`](item);
if (this.checkDumper()) return;
}
}
}
dropTables() {
if (this.options.dropReferences) {
for (const fk of _.flatten(this.tables.map(x => x.dependencies || []))) {
this.dmp.dropForeignKey(fk);
}
}
if (this.options.dropTables) {
for (const table of this.tables) {
this.dmp.dropTable(table, { testIfExists: this.options.checkIfTableExists });
}
}
}
async insertTableData(table: TableInfo) {
const dmpLocal = this.driver.createDumper();
dmpLocal.put('^select * ^from %f', table);
const autoinc = table.columns.find(x => x.autoIncrement);
if (autoinc && !this.options.skipAutoincrementColumn) {
this.dmp.allowIdentityInsert(table, true);
}
const readable = await this.driver.readQuery(this.pool, dmpLocal.s, table);
await this.processReadable(table, readable);
if (autoinc && !this.options.skipAutoincrementColumn) {
this.dmp.allowIdentityInsert(table, false);
}
}
processReadable(table: TableInfo, readable) {
const columnsFiltered = this.options.skipAutoincrementColumn
? table.columns.filter(x => !x.autoIncrement)
: table.columns;
const columnNames = columnsFiltered.map(x => x.columnName);
let isClosed = false;
let isHeaderRead = false;
return new Promise(resolve => {
readable.on('data', chunk => {
if (isClosed) return;
if (!isHeaderRead) {
isHeaderRead = true;
return;
}
if (this.checkDumper()) {
isClosed = true;
resolve(undefined);
readable.destroy();
return;
}
const columnNamesCopy = this.options.omitNulls ? columnNames.filter(col => chunk[col] != null) : columnNames;
this.dmp.put(
'^insert ^into %f (%,i) ^values (%,v);&n',
table,
columnNamesCopy,
columnNamesCopy.map(col => chunk[col])
);
});
readable.on('end', () => {
resolve(undefined);
});
});
}
extract(objectTypeField) {
return this.dbinfo[objectTypeField].filter(x =>
this.objects.find(
y => x.pureName == y.pureName && x.schemaName == y.schemaName && y.objectTypeField == objectTypeField
)
);
}
enableConstraints(enabled) {
if (this.options.disableConstraints) {
if (this.driver.dialect.enableConstraintsPerTable) {
for (const table of this.tables) {
this.dmp.enableConstraints(table, enabled);
}
} else {
this.dmp.enableConstraints(null, enabled);
}
}
}
}

View File

@@ -69,6 +69,7 @@ export function createBulkInsertStreamBase(driver, stream, pool, name, options):
}
dmp.putRaw(';');
// require('fs').writeFileSync('/home/jena/test.sql', dmp.s);
// console.log(dmp.s);
await driver.query(pool, dmp.s);
};

View File

@@ -8,3 +8,5 @@ export * from './driverBase';
export * from './SqlDumper';
export * from './testPermission';
export * from './splitPostgresQuery';
export * from './SqlGenerator';
export * from './structureTools';

View File

@@ -38,14 +38,26 @@ export function findObjectLike(
if (!dbinfo) return null;
if (schemaName) {
// @ts-ignore
return dbinfo[objectTypeField].find(
return dbinfo[objectTypeField]?.find(
x => equalStringLike(x.pureName, pureName) && equalStringLike(x.schemaName, schemaName)
);
}
// @ts-ignore
return dbinfo[objectTypeField].find(x => equalStringLike(x.pureName, pureName));
return dbinfo[objectTypeField]?.find(x => equalStringLike(x.pureName, pureName));
}
export function findForeignKeyForColumn(table: TableInfo, column: ColumnInfo) {
return (table.foreignKeys || []).find(fk => fk.columns.find(col => col.columnName == column.columnName));
}
export function makeUniqueColumnNames(res: ColumnInfo[]) {
const usedNames = new Set();
for (let i = 0; i < res.length; i++) {
if (usedNames.has(res[i].columnName)) {
let suffix = 2;
while (usedNames.has(`${res[i].columnName}${suffix}`)) suffix++;
res[i].columnName = `${res[i].columnName}${suffix}`;
}
usedNames.add(res[i].columnName);
}
}

View File

@@ -0,0 +1,84 @@
import { DatabaseInfo } from 'dbgate-types';
import _ from 'lodash';
export function addTableDependencies(db: DatabaseInfo): DatabaseInfo {
const allForeignKeys = _.flatten(db.tables.map(x => x.foreignKeys || []));
return {
...db,
tables: db.tables.map(table => ({
...table,
dependencies: allForeignKeys.filter(x => x.refSchemaName == table.schemaName && x.refTableName == table.pureName),
})),
};
}
function fillTableExtendedInfo(db: DatabaseInfo): DatabaseInfo {
return {
...db,
tables: (db.tables || []).map(obj => ({
...obj,
objectTypeField: 'tables',
columns: (obj.columns || []).map(column => ({
pureName: obj.pureName,
schemaName: obj.schemaName,
...column,
})),
primaryKey: obj.primaryKey
? {
...obj.primaryKey,
pureName: obj.pureName,
schemaName: obj.schemaName,
constraintType: 'primaryKey',
}
: undefined,
foreignKeys: (obj.foreignKeys || []).map(cnt => ({
...cnt,
pureName: obj.pureName,
schemaName: obj.schemaName,
constraintType: 'foreignKey',
})),
indexes: (obj.indexes || []).map(cnt => ({
...cnt,
pureName: obj.pureName,
schemaName: obj.schemaName,
constraintType: 'index',
})),
checks: (obj.checks || []).map(cnt => ({
...cnt,
pureName: obj.pureName,
schemaName: obj.schemaName,
constraintType: 'check',
})),
uniques: (obj.uniques || []).map(cnt => ({
...cnt,
pureName: obj.pureName,
schemaName: obj.schemaName,
constraintType: 'unique',
})),
})),
collections: (db.collections || []).map(obj => ({
...obj,
objectTypeField: 'collections',
})),
views: (db.views || []).map(obj => ({
...obj,
objectTypeField: 'views',
})),
procedures: (db.procedures || []).map(obj => ({
...obj,
objectTypeField: 'procedures',
})),
functions: (db.functions || []).map(obj => ({
...obj,
objectTypeField: 'functions',
})),
triggers: (db.triggers || []).map(obj => ({
...obj,
objectTypeField: 'triggers',
})),
};
}
export function extendDatabaseInfo(db: DatabaseInfo): DatabaseInfo {
return fillTableExtendedInfo(addTableDependencies(db));
}

View File

@@ -10,7 +10,7 @@ export interface ColumnReference {
export interface ConstraintInfo extends NamedObjectInfo {
constraintName: string;
constraintType: string;
constraintType: 'primaryKey' | 'foreignKey' | 'index' | 'check' | 'unique';
}
export interface ColumnsConstraintInfo extends ConstraintInfo {
@@ -26,7 +26,18 @@ export interface ForeignKeyInfo extends ColumnsConstraintInfo {
deleteAction: string;
}
export interface ColumnInfo {
export interface IndexInfo extends ColumnsConstraintInfo {
isUnique: boolean;
indexType: 'normal' | 'clustered' | 'xml' | 'spatial' | 'fulltext';
}
export interface UniqueInfo extends ColumnsConstraintInfo {}
export interface CheckInfo extends ConstraintInfo {
definition: string;
}
export interface ColumnInfo extends NamedObjectInfo {
columnName: string;
notNull: boolean;
autoIncrement: boolean;
@@ -58,8 +69,13 @@ export interface TableInfo extends DatabaseObjectInfo {
primaryKey?: PrimaryKeyInfo;
foreignKeys: ForeignKeyInfo[];
dependencies?: ForeignKeyInfo[];
indexes?: IndexInfo[];
uniques?: UniqueInfo[];
checks?: CheckInfo[];
}
export interface CollectionInfo extends DatabaseObjectInfo {}
export interface ViewInfo extends SqlObjectInfo {
columns: ColumnInfo[];
}
@@ -77,6 +93,7 @@ export interface SchemaInfo {
export interface DatabaseInfoObjects {
tables: TableInfo[];
collections: CollectionInfo[];
views: ViewInfo[];
procedures: ProcedureInfo[];
functions: FunctionInfo[];

View File

@@ -5,4 +5,8 @@ export interface SqlDialect {
offsetFetchRangeSyntax?: boolean;
quoteIdentifier(s: string): string;
fallbackDataType?: string;
explicitDropConstraint?: boolean;
anonymousPrimaryKey?: boolean;
enableConstraintsPerTable?: boolean;
nosql?: boolean; // mongo
}

View File

@@ -24,9 +24,18 @@ export interface EngineAuthType {
disabledFields: string[];
}
export interface ReadCollectionOptions {
pureName: string;
schemaName?: string;
countDocuments?: boolean;
skip?: number;
limit?: number;
}
export interface EngineDriver {
engine: string;
title: string;
defaultPort?: number;
connect({ server, port, user, password, database }): any;
query(pool: any, sql: string): Promise<QueryResult>;
stream(pool: any, sql: string, options: StreamOptions);
@@ -51,6 +60,7 @@ export interface EngineDriver {
dialect: SqlDialect;
createDumper(): SqlDumper;
getAuthTypes(): EngineAuthType[];
readCollection(pool: any, options: ReadCollectionOptions): Promise<any>;
analyserClass?: any;
dumperClass?: any;

View File

@@ -21,6 +21,12 @@ export interface FileFormatDefinition {
getOutputParams?: (sourceName, values) => any;
}
export interface ThemeDefinition {
className: string;
themeName: string;
themeType: 'light' | 'dark';
}
export interface PluginDefinition {
packageName: string;
manifest: any;
@@ -31,4 +37,5 @@ export interface ExtensionsDirectory {
plugins: PluginDefinition[];
fileFormats: FileFormatDefinition[];
drivers: EngineDriver[];
themes: ThemeDefinition[];
}

View File

@@ -1,18 +1,18 @@
{
"version": "1.0.2",
"version": "4.0.0",
"name": "dbgate-types",
"homepage": "https://dbgate.org/",
"repository": {
"type": "git",
"url": "https://github.com/dbshell/dbgate.git"
"url": "https://github.com/dbgate/dbgate.git"
},
"funding": "https://www.paypal.com/paypalme/JanProchazkaCz/30eur",
"author": "Jan Prochazka",
"license": "GPL",
"license": "MIT",
"keywords": [
"dbgate"
],
"types": "index.d.ts",
"main": "",
"typeScriptVersion": "2.8"
}
}

View File

@@ -1,28 +0,0 @@
module.exports = {
"env": {
"browser": true,
"es6": true
},
"extends": [
"eslint:recommended",
"plugin:react/recommended"
],
"globals": {
"Atomics": "readonly",
"SharedArrayBuffer": "readonly"
},
"parserOptions": {
"ecmaFeatures": {
"jsx": true
},
"ecmaVersion": 2018,
"sourceType": "module"
},
"plugins": [
"react"
],
"rules": {
"react/prop-types": "off",
"no-unused-vars": "warn"
}
};

105
packages/web/README.md Normal file
View File

@@ -0,0 +1,105 @@
*Looking for a shareable component template? Go here --> [sveltejs/component-template](https://github.com/sveltejs/component-template)*
---
# svelte app
This is a project template for [Svelte](https://svelte.dev) apps. It lives at https://github.com/sveltejs/template.
To create a new project based on this template using [degit](https://github.com/Rich-Harris/degit):
```bash
npx degit sveltejs/template svelte-app
cd svelte-app
```
*Note that you will need to have [Node.js](https://nodejs.org) installed.*
## Get started
Install the dependencies...
```bash
cd svelte-app
npm install
```
...then start [Rollup](https://rollupjs.org):
```bash
npm run dev
```
Navigate to [localhost:5000](http://localhost:5000). You should see your app running. Edit a component file in `src`, save it, and reload the page to see your changes.
By default, the server will only respond to requests from localhost. To allow connections from other computers, edit the `sirv` commands in package.json to include the option `--host 0.0.0.0`.
If you're using [Visual Studio Code](https://code.visualstudio.com/) we recommend installing the official extension [Svelte for VS Code](https://marketplace.visualstudio.com/items?itemName=svelte.svelte-vscode). If you are using other editors you may need to install a plugin in order to get syntax highlighting and intellisense.
## Building and running in production mode
To create an optimised version of the app:
```bash
npm run build
```
You can run the newly built app with `npm run start`. This uses [sirv](https://github.com/lukeed/sirv), which is included in your package.json's `dependencies` so that the app will work when you deploy to platforms like [Heroku](https://heroku.com).
## Single-page app mode
By default, sirv will only respond to requests that match files in `public`. This is to maximise compatibility with static fileservers, allowing you to deploy your app anywhere.
If you're building a single-page app (SPA) with multiple routes, sirv needs to be able to respond to requests for *any* path. You can make it so by editing the `"start"` command in package.json:
```js
"start": "sirv public --single"
```
## Using TypeScript
This template comes with a script to set up a TypeScript development environment, you can run it immediately after cloning the template with:
```bash
node scripts/setupTypeScript.js
```
Or remove the script via:
```bash
rm scripts/setupTypeScript.js
```
## Deploying to the web
### With [Vercel](https://vercel.com)
Install `vercel` if you haven't already:
```bash
npm install -g vercel
```
Then, from within your project folder:
```bash
cd public
vercel deploy --name my-project
```
### With [surge](https://surge.sh/)
Install `surge` if you haven't already:
```bash
npm install -g surge
```
Then, from within your project folder:
```bash
npm run build
surge public my-project.surge.sh
```

View File

@@ -1,67 +1,54 @@
{
"name": "dbgate-web",
"version": "1.0.0",
"private": true,
"dependencies": {
"version": "4.0.0",
"scripts": {
"build": "rollup -c",
"dev": "cross-env API_URL=http://localhost:3000 rollup -c -w",
"start": "sirv public",
"validate": "svelte-check",
"prepublishOnly": "yarn build"
},
"files": [
"public"
],
"devDependencies": {
"@ant-design/colors": "^5.0.0",
"@mdi/font": "^5.8.55",
"@testing-library/jest-dom": "^4.2.4",
"@testing-library/react": "^9.3.2",
"@testing-library/user-event": "^7.1.2",
"@rollup/plugin-commonjs": "^17.0.0",
"@rollup/plugin-node-resolve": "^11.0.0",
"@rollup/plugin-replace": "^2.4.1",
"@rollup/plugin-typescript": "^6.0.0",
"@tsconfig/svelte": "^1.0.0",
"ace-builds": "^1.4.8",
"axios": "^0.19.0",
"chart.js": "^2.9.4",
"compare-versions": "^3.6.0",
"cross-env": "^6.0.3",
"dbgate-datalib": "^1.0.0",
"dbgate-sqltree": "^1.0.0",
"dbgate-tools": "^1.0.0",
"eslint": "^6.8.0",
"eslint-plugin-react": "^7.17.0",
"cross-env": "^7.0.3",
"dbgate-datalib": "^4.0.0",
"dbgate-sqltree": "^4.0.0",
"dbgate-tools": "^4.0.0",
"dbgate-types": "^4.0.0",
"json-stable-stringify": "^1.0.1",
"localforage": "^1.9.0",
"markdown-to-jsx": "^7.1.0",
"lodash": "^4.17.15",
"randomcolor": "^0.6.2",
"react": "^16.12.0",
"react-ace": "^8.0.0",
"react-chartjs-2": "^2.11.1",
"react-dom": "^16.12.0",
"react-dropzone": "^11.2.3",
"react-helmet": "^6.1.0",
"react-json-view": "^1.19.1",
"react-modal": "^3.11.1",
"react-scripts": "3.3.0",
"react-select": "^3.1.0",
"resize-observer-polyfill": "^1.5.1",
"rollup": "^2.3.4",
"rollup-plugin-copy": "^3.3.0",
"rollup-plugin-css-only": "^3.1.0",
"rollup-plugin-livereload": "^2.0.0",
"rollup-plugin-svelte": "^7.0.0",
"rollup-plugin-terser": "^7.0.0",
"sirv-cli": "^1.0.0",
"socket.io-client": "^2.3.0",
"sql-formatter": "^2.3.3",
"styled-components": "^4.4.1",
"uuid": "^3.4.0"
},
"scripts": {
"start": "cross-env BROWSER=none PORT=5000 react-scripts start",
"build:docker": "cross-env CI=false REACT_APP_API_URL=ORIGIN react-scripts build",
"build:app": "cross-env PUBLIC_URL=. CI=false react-scripts build",
"test": "react-scripts test",
"eject": "react-scripts eject",
"ts": "tsc"
},
"browserslist": {
"production": [
">0.2%",
"not dead",
"not op_mini all"
],
"development": [
"last 1 chrome version",
"last 1 firefox version",
"last 1 safari version"
]
},
"devDependencies": {
"@types/react": "^16.9.17",
"@types/styled-components": "^4.4.2",
"dbgate-types": "^1.0.0",
"typescript": "^3.7.4"
"svelte": "^3.35.0",
"svelte-check": "^1.0.0",
"svelte-preprocess": "^4.0.0",
"tslib": "^2.0.0",
"typescript": "^3.9.3",
"uuid": "^3.4.0",
"@mdi/font": "^5.9.55",
"file-selector": "^0.2.4",
"resize-observer-polyfill": "^1.5.1",
"svelte-json-tree": "^0.1.0",
"svelte-markdown": "^0.1.4",
"svelte-select": "^3.17.0"
}
}
}

419
packages/web/public/bulma.css vendored Normal file
View File

@@ -0,0 +1,419 @@
.m-0 {
margin: 0 !important;
}
.mt-0 {
margin-top: 0 !important;
}
.mr-0 {
margin-right: 0 !important;
}
.mb-0 {
margin-bottom: 0 !important;
}
.ml-0 {
margin-left: 0 !important;
}
.mx-0 {
margin-left: 0 !important;
margin-right: 0 !important;
}
.my-0 {
margin-top: 0 !important;
margin-bottom: 0 !important;
}
.m-1 {
margin: 0.25rem !important;
}
.mt-1 {
margin-top: 0.25rem !important;
}
.mr-1 {
margin-right: 0.25rem !important;
}
.mb-1 {
margin-bottom: 0.25rem !important;
}
.ml-1 {
margin-left: 0.25rem !important;
}
.mx-1 {
margin-left: 0.25rem !important;
margin-right: 0.25rem !important;
}
.my-1 {
margin-top: 0.25rem !important;
margin-bottom: 0.25rem !important;
}
.m-2 {
margin: 0.5rem !important;
}
.mt-2 {
margin-top: 0.5rem !important;
}
.mr-2 {
margin-right: 0.5rem !important;
}
.mb-2 {
margin-bottom: 0.5rem !important;
}
.ml-2 {
margin-left: 0.5rem !important;
}
.mx-2 {
margin-left: 0.5rem !important;
margin-right: 0.5rem !important;
}
.my-2 {
margin-top: 0.5rem !important;
margin-bottom: 0.5rem !important;
}
.m-3 {
margin: 0.75rem !important;
}
.mt-3 {
margin-top: 0.75rem !important;
}
.mr-3 {
margin-right: 0.75rem !important;
}
.mb-3 {
margin-bottom: 0.75rem !important;
}
.ml-3 {
margin-left: 0.75rem !important;
}
.mx-3 {
margin-left: 0.75rem !important;
margin-right: 0.75rem !important;
}
.my-3 {
margin-top: 0.75rem !important;
margin-bottom: 0.75rem !important;
}
.m-4 {
margin: 1rem !important;
}
.mt-4 {
margin-top: 1rem !important;
}
.mr-4 {
margin-right: 1rem !important;
}
.mb-4 {
margin-bottom: 1rem !important;
}
.ml-4 {
margin-left: 1rem !important;
}
.mx-4 {
margin-left: 1rem !important;
margin-right: 1rem !important;
}
.my-4 {
margin-top: 1rem !important;
margin-bottom: 1rem !important;
}
.m-5 {
margin: 1.5rem !important;
}
.mt-5 {
margin-top: 1.5rem !important;
}
.mr-5 {
margin-right: 1.5rem !important;
}
.mb-5 {
margin-bottom: 1.5rem !important;
}
.ml-5 {
margin-left: 1.5rem !important;
}
.mx-5 {
margin-left: 1.5rem !important;
margin-right: 1.5rem !important;
}
.my-5 {
margin-top: 1.5rem !important;
margin-bottom: 1.5rem !important;
}
.m-6 {
margin: 3rem !important;
}
.mt-6 {
margin-top: 3rem !important;
}
.mr-6 {
margin-right: 3rem !important;
}
.mb-6 {
margin-bottom: 3rem !important;
}
.ml-6 {
margin-left: 3rem !important;
}
.mx-6 {
margin-left: 3rem !important;
margin-right: 3rem !important;
}
.my-6 {
margin-top: 3rem !important;
margin-bottom: 3rem !important;
}
.p-0 {
padding: 0 !important;
}
.pt-0 {
padding-top: 0 !important;
}
.pr-0 {
padding-right: 0 !important;
}
.pb-0 {
padding-bottom: 0 !important;
}
.pl-0 {
padding-left: 0 !important;
}
.px-0 {
padding-left: 0 !important;
padding-right: 0 !important;
}
.py-0 {
padding-top: 0 !important;
padding-bottom: 0 !important;
}
.p-1 {
padding: 0.25rem !important;
}
.pt-1 {
padding-top: 0.25rem !important;
}
.pr-1 {
padding-right: 0.25rem !important;
}
.pb-1 {
padding-bottom: 0.25rem !important;
}
.pl-1 {
padding-left: 0.25rem !important;
}
.px-1 {
padding-left: 0.25rem !important;
padding-right: 0.25rem !important;
}
.py-1 {
padding-top: 0.25rem !important;
padding-bottom: 0.25rem !important;
}
.p-2 {
padding: 0.5rem !important;
}
.pt-2 {
padding-top: 0.5rem !important;
}
.pr-2 {
padding-right: 0.5rem !important;
}
.pb-2 {
padding-bottom: 0.5rem !important;
}
.pl-2 {
padding-left: 0.5rem !important;
}
.px-2 {
padding-left: 0.5rem !important;
padding-right: 0.5rem !important;
}
.py-2 {
padding-top: 0.5rem !important;
padding-bottom: 0.5rem !important;
}
.p-3 {
padding: 0.75rem !important;
}
.pt-3 {
padding-top: 0.75rem !important;
}
.pr-3 {
padding-right: 0.75rem !important;
}
.pb-3 {
padding-bottom: 0.75rem !important;
}
.pl-3 {
padding-left: 0.75rem !important;
}
.px-3 {
padding-left: 0.75rem !important;
padding-right: 0.75rem !important;
}
.py-3 {
padding-top: 0.75rem !important;
padding-bottom: 0.75rem !important;
}
.p-4 {
padding: 1rem !important;
}
.pt-4 {
padding-top: 1rem !important;
}
.pr-4 {
padding-right: 1rem !important;
}
.pb-4 {
padding-bottom: 1rem !important;
}
.pl-4 {
padding-left: 1rem !important;
}
.px-4 {
padding-left: 1rem !important;
padding-right: 1rem !important;
}
.py-4 {
padding-top: 1rem !important;
padding-bottom: 1rem !important;
}
.p-5 {
padding: 1.5rem !important;
}
.pt-5 {
padding-top: 1.5rem !important;
}
.pr-5 {
padding-right: 1.5rem !important;
}
.pb-5 {
padding-bottom: 1.5rem !important;
}
.pl-5 {
padding-left: 1.5rem !important;
}
.px-5 {
padding-left: 1.5rem !important;
padding-right: 1.5rem !important;
}
.py-5 {
padding-top: 1.5rem !important;
padding-bottom: 1.5rem !important;
}
.p-6 {
padding: 3rem !important;
}
.pt-6 {
padding-top: 3rem !important;
}
.pr-6 {
padding-right: 3rem !important;
}
.pb-6 {
padding-bottom: 3rem !important;
}
.pl-6 {
padding-left: 3rem !important;
}
.px-6 {
padding-left: 3rem !important;
padding-right: 3rem !important;
}
.py-6 {
padding-top: 3rem !important;
padding-bottom: 3rem !important;
}

View File

@@ -0,0 +1,22 @@
:root {
--dim-widget-icon-size: 60px;
--dim-statusbar-height: 20px;
--dim-left-panel-width: 300px;
--dim-tabs-panel-height: 53px;
--dim-tabs-height: 33px;
--dim-splitter-thickness: 3px;
--dim-visible-left-panel: 1; /* set from JS */
--dim-content-left: calc(
var(--dim-widget-icon-size) + var(--dim-visible-left-panel) *
(var(--dim-left-panel-width) + var(--dim-splitter-thickness))
);
--dim-visible-toolbar: 1; /* set from JS */
--dim-toolbar-height: 30px;
--dim-header-top: calc(var(--dim-toolbar-height) * var(--dim-visible-toolbar));
--dim-content-top: calc(var(--dim-header-top) + var(--dim-tabs-panel-height));
--dim-large-form-margin: 20px;
}

View File

@@ -0,0 +1,149 @@
body {
font-family: -apple-system, BlinkMacSystemFont, Segoe WPC, Segoe UI, HelveticaNeue-Light, Ubuntu, Droid Sans,
sans-serif;
font-size: 14px;
/* font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', 'Roboto', 'Oxygen',
'Ubuntu', 'Cantarell', 'Fira Sans', 'Droid Sans', 'Helvetica Neue',
sans-serif;
*/
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
}
.horizontal-split-handle {
background-color: var(--theme-border);
width: var(--dim-splitter-thickness);
cursor: col-resize;
}
.horizontal-split-handle:hover {
background-color: var(--theme-bg-2);
}
.vertical-split-handle {
background-color: var(--theme-border);
height: var(--dim-splitter-thickness);
cursor: row-resize;
}
.vertical-split-handle:hover {
background-color: var(--theme-bg-2);
}
.icon-invisible {
visibility: hidden;
}
.space-between {
display: flex;
justify-content: space-between;
}
.flex {
display: flex;
}
.flexcol {
display: flex;
flex-direction: column;
}
.nowrap {
white-space: nowrap;
}
.bold {
font-weight: bold;
}
.flex1 {
flex: 1;
}
.relative {
position: relative;
}
.col-9 {
flex-basis: 75%;
max-width: 75%;
}
.col-8 {
flex-basis: 66.6667%;
max-width: 66.6667%;
}
.col-6 {
flex-basis: 50%;
max-width: 50%;
}
.col-4 {
flex-basis: 33.3333%;
max-width: 33.3333%;
}
.col-3 {
flex-basis: 25%;
max-width: 25%;
}
.largeFormMarker input[type='text'] {
width: 100%;
padding: 10px 10px;
font-size: 14px;
box-sizing: border-box;
border-radius: 4px;
}
.largeFormMarker input[type='password'] {
width: 100%;
padding: 10px 10px;
font-size: 14px;
box-sizing: border-box;
border-radius: 4px;
}
.largeFormMarker select {
width: 100%;
padding: 10px 10px;
font-size: 14px;
box-sizing: border-box;
border-radius: 4px;
}
body *::-webkit-scrollbar {
height: 0.8em;
width: 0.8em;
}
body *::-webkit-scrollbar-track {
border-radius: 1px;
background-color: var(--theme-bg-1);
}
body *::-webkit-scrollbar-corner {
border-radius: 1px;
background-color: var(--theme-bg-2);
}
body *::-webkit-scrollbar-thumb {
border-radius: 1px;
background-color: var(--theme-bg-3);
}
body *::-webkit-scrollbar-thumb:hover {
background-color: var(--theme-bg-4);
}
input {
background-color: var(--theme-bg-0);
color: var(--theme-font-1);
border: 1px solid var(--theme-border);
}
input[disabled] {
background-color: var(--theme-bg-1);
}
select {
background-color: var(--theme-bg-0);
color: var(--theme-font-1);
border: 1px solid var(--theme-border);
}
select[disabled] {
background-color: var(--theme-bg-1);
}
textarea {
background-color: var(--theme-bg-0);
color: var(--theme-font-1);
border: 1px solid var(--theme-border);
}

View File

@@ -0,0 +1,32 @@
.color-icon-blue {
color: var(--theme-icon-blue);
}
.color-icon-green {
color: var(--theme-icon-green);
}
.color-icon-red {
color: var(--theme-icon-red);
}
.color-icon-gold {
color: var(--theme-icon-gold);
}
.color-icon-yellow {
color: var(--theme-icon-yellow);
}
.color-icon-magenta {
color: var(--theme-icon-magenta);
}
.color-icon-inv-green {
color: var(--theme-icon-inv-green);
}
.color-icon-inv-red {
color: var(--theme-icon-inv-red);
}

View File

@@ -2,43 +2,28 @@
<html lang="en">
<head>
<meta charset="utf-8" />
<link rel="icon" href="%PUBLIC_URL%/favicon.ico" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<meta name="theme-color" content="#000000" />
<meta name="description"
content="DbGate - web based opensource database administration tool for MS SQL, MySQL, Postgre SQL" />
<link rel="apple-touch-icon" href="%PUBLIC_URL%/logo192.png" />
<!--
manifest.json provides metadata used when your web app is installed on a
user's mobile device or desktop. See https://developers.google.com/web/fundamentals/web-app-manifest/
-->
<link rel="manifest" href="%PUBLIC_URL%/manifest.json" />
<!--
Notice the use of %PUBLIC_URL% in the tags above.
It will be replaced with the URL of the `public` folder during the build.
Only files inside the `public` folder can be referenced from the HTML.
<meta charset='utf-8'>
<meta name='viewport' content='width=device-width,initial-scale=1'>
Unlike "/favicon.ico" or "favicon.ico", "%PUBLIC_URL%/favicon.ico" will
work correctly both with client-side routing and a non-root public URL.
Learn how to configure a non-root public URL by running `npm run build`.
-->
<title>DbGate</title>
<title>DbGate</title>
<meta name="theme-color" content="#000000" />
<meta name="description"
content="DbGate - web based opensource database administration tool for MS SQL, MySQL, Postgre SQL" />
<link rel='icon' type='image/png' href='favicon.ico'>
<link rel="manifest" href="manifest.json" />
<link rel='stylesheet' href='global.css'>
<link rel='stylesheet' href='dimensions.css'>
<link rel='stylesheet' href='bulma.css'>
<link rel='stylesheet' href='icon-colors.css'>
<link rel='stylesheet' href='build/bundle.css'>
<link rel='stylesheet' href='build/fonts/materialdesignicons.css'>
<script defer src='build/bundle.js'></script>
</head>
<body>
<noscript>You need to enable JavaScript to run this app.</noscript>
<div id="root">Loading DbGate...</div>
<!--
This HTML file is a template.
If you open it directly in the browser, you will see an empty page.
You can add webfonts, meta tags, or analytics to this file.
The build step will place the bundled scripts into the <body> tag.
To begin the development, run `npm start` or `yarn start`.
To create a production bundle, use `npm run build` or `yarn build`.
-->
</body>
</html>

View File

@@ -1,2 +0,0 @@
# https://www.robotstxt.org/robotstxt.html
User-agent: *

View File

@@ -1,16 +0,0 @@
body {
background: #666;
}
div {
color: white;
font-size: 25pt;
position: absolute;
left: 0;
top: 0;
right: 0;
bottom: 0;
text-align: center;
font-family: Arial, Helvetica, sans-serif;
margin-top: 40px;
}

View File

@@ -1,12 +0,0 @@
<!DOCTYPE html>
<html>
<head>
<link rel="stylesheet" href="splash.css">
</head>
<body>
<div>Starting DbGate...</div>
</body>
</html>

View File

@@ -0,0 +1,102 @@
import svelte from 'rollup-plugin-svelte';
import commonjs from '@rollup/plugin-commonjs';
import resolve from '@rollup/plugin-node-resolve';
import livereload from 'rollup-plugin-livereload';
import copy from 'rollup-plugin-copy';
import { terser } from 'rollup-plugin-terser';
import sveltePreprocess from 'svelte-preprocess';
import typescript from '@rollup/plugin-typescript';
import replace from '@rollup/plugin-replace';
import css from 'rollup-plugin-css-only';
const production = !process.env.ROLLUP_WATCH;
function serve() {
let server;
function toExit() {
if (server) server.kill(0);
}
return {
writeBundle() {
if (server) return;
server = require('child_process').spawn('npm', ['run', 'start', '--', '--dev'], {
stdio: ['ignore', 'inherit', 'inherit'],
shell: true,
});
process.on('SIGTERM', toExit);
process.on('exit', toExit);
},
};
}
export default {
input: 'src/main.ts',
output: {
sourcemap: true,
format: 'iife',
name: 'app',
file: 'public/build/bundle.js',
},
plugins: [
copy({
targets: [
{
src: '../../node_modules/@mdi/font/css/materialdesignicons.css',
dest: 'public/build/fonts/',
},
{
src: '../../node_modules/@mdi/font/fonts/*',
dest: 'public/build/fonts/',
},
],
}),
replace({
'process.env.API_URL': JSON.stringify(process.env.API_URL),
}),
svelte({
preprocess: sveltePreprocess({ sourceMap: !production }),
compilerOptions: {
// enable run-time checks when not in production
dev: !production,
},
}),
// we'll extract any component CSS out into
// a separate file - better for performance
css({ output: 'bundle.css' }),
// If you have external dependencies installed from
// npm, you'll most likely need these plugins. In
// some cases you'll need additional configuration -
// consult the documentation for details:
// https://github.com/rollup/plugins/tree/master/packages/commonjs
resolve({
browser: true,
dedupe: ['svelte'],
}),
commonjs(),
typescript({
sourceMap: !production,
inlineSources: !production,
}),
// In dev mode, call `npm run start` once
// the bundle has been generated
!production && serve(),
// Watch the `public` directory and refresh the
// browser on changes when not in production
!production && livereload('public'),
// If we're building for production (npm run build
// instead of npm run dev), minify
production && terser(),
],
watch: {
clearScreen: false,
},
};

View File

@@ -1,38 +0,0 @@
.App {
text-align: center;
}
.App-logo {
height: 40vmin;
pointer-events: none;
}
@media (prefers-reduced-motion: no-preference) {
.App-logo {
animation: App-logo-spin infinite 20s linear;
}
}
.App-header {
background-color: #282c34;
min-height: 100vh;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
font-size: calc(10px + 2vmin);
color: white;
}
.App-link {
color: #61dafb;
}
@keyframes App-logo-spin {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
}

View File

@@ -1,57 +0,0 @@
import React from 'react';
import './index.css';
import Screen from './Screen';
import {
CurrentWidgetProvider,
CurrentDatabaseProvider,
OpenedTabsProvider,
OpenedConnectionsProvider,
LeftPanelWidthProvider,
CurrentArchiveProvider,
CurrentThemeProvider,
} from './utility/globalState';
import { SocketProvider } from './utility/SocketProvider';
import ConnectionsPinger from './utility/ConnectionsPinger';
import { ModalLayerProvider } from './modals/showModal';
import UploadsProvider from './utility/UploadsProvider';
import ThemeHelmet from './themes/ThemeHelmet';
import PluginsProvider from './plugins/PluginsProvider';
import { ExtensionsProvider } from './utility/useExtensions';
import { MenuLayerProvider } from './modals/showMenu';
function App() {
return (
<CurrentWidgetProvider>
<CurrentDatabaseProvider>
<SocketProvider>
<OpenedTabsProvider>
<OpenedConnectionsProvider>
<LeftPanelWidthProvider>
<ConnectionsPinger>
<PluginsProvider>
<ExtensionsProvider>
<CurrentArchiveProvider>
<CurrentThemeProvider>
<UploadsProvider>
<ModalLayerProvider>
<MenuLayerProvider>
<ThemeHelmet />
<Screen />
</MenuLayerProvider>
</ModalLayerProvider>
</UploadsProvider>
</CurrentThemeProvider>
</CurrentArchiveProvider>
</ExtensionsProvider>
</PluginsProvider>
</ConnectionsPinger>
</LeftPanelWidthProvider>
</OpenedConnectionsProvider>
</OpenedTabsProvider>
</SocketProvider>
</CurrentDatabaseProvider>
</CurrentWidgetProvider>
);
}
export default App;

View File

@@ -0,0 +1,16 @@
<script lang="ts">
import CommandListener from './commands/CommandListener.svelte';
import DataGridRowHeightMeter from './datagrid/DataGridRowHeightMeter.svelte';
import PluginsProvider from './plugins/PluginsProvider.svelte';
import Screen from './Screen.svelte';
import ErrorHandler from './utility/ErrorHandler.svelte';
import OpenTabsOnStartup from './utility/OpenTabsOnStartup.svelte';
</script>
<DataGridRowHeightMeter />
<ErrorHandler />
<PluginsProvider />
<CommandListener />
<OpenTabsOnStartup />
<Screen />

View File

@@ -1,11 +0,0 @@
// @ts-nocheck
import React from 'react';
import { render } from '@testing-library/react';
import App from './App';
test('renders learn react link', () => {
const { getByText } = render(<App />);
const linkElement = getByText(/learn react/i);
expect(linkElement).toBeInTheDocument();
});

View File

@@ -1,64 +0,0 @@
import React from 'react';
import styled from 'styled-components';
import { FontIcon } from './icons';
import useTheme from './theme/useTheme';
import useExtensions from './utility/useExtensions';
const TargetStyled = styled.div`
position: fixed;
display: flex;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: ${props => props.theme.main_background_blue[3]};
align-items: center;
justify-content: space-around;
z-index: 1000;
`;
const InfoBox = styled.div``;
const IconWrapper = styled.div`
display: flex;
justify-content: space-around;
font-size: 50px;
margin-bottom: 20px;
`;
const InfoWrapper = styled.div`
display: flex;
justify-content: space-around;
margin-top: 10px;
`;
const TitleWrapper = styled.div`
font-size: 30px;
display: flex;
justify-content: space-around;
`;
export default function DragAndDropFileTarget({ isDragActive, inputProps }) {
const theme = useTheme();
const { fileFormats } = useExtensions();
return (
!!isDragActive && (
<TargetStyled theme={theme}>
<InfoBox>
<IconWrapper>
<FontIcon icon="icon cloud-upload" />
</IconWrapper>
<TitleWrapper>Drop the files to upload to DbGate</TitleWrapper>
<InfoWrapper>
Supported file types:{' '}
{fileFormats
.filter(x => x.readerFunc)
.map(x => x.name)
.join(', ')}
</InfoWrapper>
</InfoBox>
<input {...inputProps} />
</TargetStyled>
)
);
}

View File

@@ -0,0 +1,55 @@
<script lang="ts">
import _ from 'lodash';
import FontIcon from './icons/FontIcon.svelte';
import { extensions } from './stores';
import getElectron from './utility/getElectron';
const electron = getElectron();
$: fileTypeNames = _.compact([
...$extensions.fileFormats.filter(x => x.readerFunc).map(x => x.name),
electron ? 'SQL' : null,
]);
</script>
<div class="target">
<div>
<div class="icon">
<FontIcon icon="icon cloud-upload" />
</div>
<div class="title">Drop the files to upload to DbGate</div>
<div class="info">Supported file types: {fileTypeNames.join(', ')}</div>
</div>
</div>
<style>
.target {
position: fixed;
display: flex;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: var(--theme-bg-selected);
align-items: center;
justify-content: space-around;
z-index: 1000;
}
.icon {
display: flex;
justify-content: space-around;
font-size: 50px;
margin-bottom: 20px;
}
.info {
display: flex;
justify-content: space-around;
margin-top: 10px;
}
.title {
font-size: 30px;
display: flex;
justify-content: space-around;
}
</style>

View File

@@ -1,144 +0,0 @@
// @ts-nocheck
import React from 'react';
import dimensions from './theme/dimensions';
import styled from 'styled-components';
import TabsPanel from './TabsPanel';
import TabContent from './TabContent';
import WidgetIconPanel from './widgets/WidgetIconPanel';
import { useCurrentWidget, useLeftPanelWidth, useSetLeftPanelWidth } from './utility/globalState';
import WidgetContainer from './widgets/WidgetContainer';
import ToolBar from './widgets/Toolbar';
import StatusBar from './widgets/StatusBar';
import { useSplitterDrag, HorizontalSplitHandle } from './widgets/Splitter';
import { ModalLayer } from './modals/showModal';
import DragAndDropFileTarget from './DragAndDropFileTarget';
import { useUploadsZone } from './utility/UploadsProvider';
import useTheme from './theme/useTheme';
import { MenuLayer } from './modals/showMenu';
import ErrorBoundary from './utility/ErrorBoundary';
const BodyDiv = styled.div`
position: fixed;
top: ${dimensions.tabsPanel.height + dimensions.toolBar.height}px;
left: ${props => props.contentLeft}px;
bottom: ${dimensions.statusBar.height}px;
right: 0;
background-color: ${props => props.theme.content_background};
`;
const ToolBarDiv = styled.div`
position: fixed;
top: 0;
left: 0;
right: 0;
background-color: ${props => props.theme.toolbar_background};
height: ${dimensions.toolBar.height}px;
`;
const IconBar = styled.div`
position: fixed;
top: ${dimensions.toolBar.height}px;
left: 0;
bottom: ${dimensions.statusBar.height}px;
width: ${dimensions.widgetMenu.iconSize}px;
background-color: ${props => props.theme.widget_background};
`;
const LeftPanel = styled.div`
position: fixed;
top: ${dimensions.toolBar.height}px;
left: ${dimensions.widgetMenu.iconSize}px;
bottom: ${dimensions.statusBar.height}px;
background-color: ${props => props.theme.left_background};
display: flex;
`;
const TabsPanelContainer = styled.div`
display: flex;
position: fixed;
top: ${dimensions.toolBar.height}px;
left: ${props => props.contentLeft}px;
height: ${dimensions.tabsPanel.height}px;
right: 0;
background-color: ${props => props.theme.tabs_background2};
border-top: 1px solid ${props => props.theme.border};
overflow-x: auto;
::-webkit-scrollbar {
height: 7px;
}
}
`;
const StausBarContainer = styled.div`
position: fixed;
height: ${dimensions.statusBar.height}px;
left: 0;
right: 0;
bottom: 0;
background-color: ${props => props.theme.statusbar_background};
`;
const ScreenHorizontalSplitHandle = styled(HorizontalSplitHandle)`
position: absolute;
top: ${dimensions.toolBar.height}px;
bottom: ${dimensions.statusBar.height}px;
`;
// const StyledRoot = styled.div`
// // color: ${(props) => props.theme.fontColor};
// `;
export default function Screen() {
const theme = useTheme();
const currentWidget = useCurrentWidget();
const leftPanelWidth = useLeftPanelWidth();
const setLeftPanelWidth = useSetLeftPanelWidth();
const contentLeft = currentWidget
? dimensions.widgetMenu.iconSize + leftPanelWidth + dimensions.splitter.thickness
: dimensions.widgetMenu.iconSize;
const toolbarPortalRef = React.useRef();
const onSplitDown = useSplitterDrag('clientX', diff => setLeftPanelWidth(v => v + diff));
const { getRootProps, getInputProps, isDragActive } = useUploadsZone();
return (
<div {...getRootProps()}>
<ToolBarDiv theme={theme}>
<ToolBar toolbarPortalRef={toolbarPortalRef} />
</ToolBarDiv>
<IconBar theme={theme}>
<WidgetIconPanel />
</IconBar>
{!!currentWidget && (
<LeftPanel theme={theme}>
<ErrorBoundary>
<WidgetContainer />
</ErrorBoundary>
</LeftPanel>
)}
{!!currentWidget && (
<ScreenHorizontalSplitHandle
onMouseDown={onSplitDown}
theme={theme}
style={{ left: leftPanelWidth + dimensions.widgetMenu.iconSize }}
/>
)}
<TabsPanelContainer contentLeft={contentLeft} theme={theme}>
<TabsPanel></TabsPanel>
</TabsPanelContainer>
<BodyDiv contentLeft={contentLeft} theme={theme}>
<TabContent toolbarPortalRef={toolbarPortalRef} />
</BodyDiv>
<StausBarContainer theme={theme}>
<StatusBar />
</StausBarContainer>
<ModalLayer />
<MenuLayer />
<DragAndDropFileTarget inputProps={getInputProps()} isDragActive={isDragActive} />
</div>
);
}

View File

@@ -0,0 +1,141 @@
<script>
import WidgetContainer from './widgets/WidgetContainer.svelte';
import WidgetIconPanel from './widgets/WidgetIconPanel.svelte';
import {
currentTheme,
currentThemeDefinition,
isFileDragActive,
leftPanelWidth,
selectedWidget,
visibleCommandPalette,
visibleToolbar,
} from './stores';
import TabsPanel from './widgets/TabsPanel.svelte';
import TabRegister from './TabRegister.svelte';
import CommandPalette from './commands/CommandPalette.svelte';
import Toolbar from './widgets/Toolbar.svelte';
import splitterDrag from './utility/splitterDrag';
import CurrentDropDownMenu from './modals/CurrentDropDownMenu.svelte';
import StatusBar from './widgets/StatusBar.svelte';
import ModalLayer from './modals/ModalLayer.svelte';
import DragAndDropFileTarget from './DragAndDropFileTarget.svelte';
import dragDropFileTarget from './utility/dragDropFileTarget';
$: currentThemeType = $currentThemeDefinition?.themeType == 'dark' ? 'theme-type-dark' : 'theme-type-light';
</script>
<div class={`${$currentTheme} ${currentThemeType} root`} use:dragDropFileTarget>
<div class="iconbar">
<WidgetIconPanel />
</div>
<div class="statusbar">
<StatusBar />
</div>
{#if $selectedWidget}
<div class="leftpanel">
<WidgetContainer />
</div>
{/if}
<div class="tabs">
<TabsPanel />
</div>
<div class="content">
<TabRegister />
</div>
{#if $selectedWidget}
<div
class="horizontal-split-handle splitter"
use:splitterDrag={'clientX'}
on:resizeSplitter={e => leftPanelWidth.update(x => x + e.detail)}
/>
{/if}
{#if $visibleCommandPalette}
<div class="commads">
<CommandPalette />
</div>
{/if}
{#if $visibleToolbar}
<div class="toolbar">
<Toolbar />
</div>
{/if}
<CurrentDropDownMenu />
<ModalLayer />
{#if $isFileDragActive}
<DragAndDropFileTarget />
{/if}
</div>
<style>
.root {
color: var(--theme-font-1);
}
.iconbar {
position: fixed;
left: 0;
top: var(--dim-header-top);
bottom: var(--dim-statusbar-height);
width: var(--dim-widget-icon-size);
background: var(--theme-bg-inv-1);
}
.statusbar {
position: fixed;
background: var(--theme-bg-statusbar-inv);
height: var(--dim-statusbar-height);
left: 0;
right: 0;
bottom: 0;
}
.leftpanel {
position: fixed;
top: var(--dim-header-top);
left: var(--dim-widget-icon-size);
bottom: var(--dim-statusbar-height);
width: var(--dim-left-panel-width);
background-color: var(--theme-bg-1);
display: flex;
}
.tabs {
display: flex;
position: fixed;
top: var(--dim-header-top);
left: var(--dim-content-left);
height: var(--dim-tabs-panel-height);
right: 0;
background-color: var(--theme-bg-2);
border-top: 1px solid var(--theme-border);
overflow-x: auto;
}
.tabs::-webkit-scrollbar {
height: 7px;
}
.content {
position: fixed;
top: var(--dim-content-top);
left: var(--dim-content-left);
bottom: var(--dim-statusbar-height);
right: 0;
background-color: var(--theme-bg-1);
}
.commads {
position: fixed;
top: var(--dim-header-top);
left: var(--dim-widget-icon-size);
}
.toolbar {
position: fixed;
top: 0;
height: var(--dim-toolbar-height);
left: 0;
right: 0;
background: var(--theme-bg-1);
}
.splitter {
position: absolute;
top: var(--dim-header-top);
bottom: var(--dim-statusbar-height);
left: calc(var(--dim-widget-icon-size) + var(--dim-left-panel-width));
}
</style>

View File

@@ -1,91 +0,0 @@
import React from 'react';
import _ from 'lodash';
import styled from 'styled-components';
import tabs from './tabs';
import { useOpenedTabs } from './utility/globalState';
import ErrorBoundary from './utility/ErrorBoundary';
const TabContainerStyled = styled.div`
position: absolute;
left: 0;
top: 0;
right: 0;
bottom: 0;
display: flex;
visibility: ${props =>
// @ts-ignore
props.tabVisible ? 'visible' : 'hidden'};
`;
function TabContainer({ TabComponent, ...props }) {
const { tabVisible, tabid, toolbarPortalRef } = props;
return (
// @ts-ignore
<TabContainerStyled tabVisible={tabVisible}>
<ErrorBoundary>
<TabComponent {...props} tabid={tabid} tabVisible={tabVisible} toolbarPortalRef={toolbarPortalRef} />
</ErrorBoundary>
</TabContainerStyled>
);
}
const TabContainerMemo = React.memo(TabContainer);
function createTabComponent(selectedTab) {
const TabComponent = tabs[selectedTab.tabComponent];
if (TabComponent) {
return {
TabComponent,
props: selectedTab.props,
};
}
return null;
}
export default function TabContent({ toolbarPortalRef }) {
const files = useOpenedTabs();
const [mountedTabs, setMountedTabs] = React.useState({});
const selectedTab = files.find(x => x.selected && x.closedTime == null);
React.useEffect(() => {
// cleanup closed tabs
if (
_.difference(
_.keys(mountedTabs),
_.map(
files.filter(x => x.closedTime == null),
'tabid'
)
).length > 0
) {
setMountedTabs(_.pickBy(mountedTabs, (v, k) => files.find(x => x.tabid == k && x.closedTime == null)));
}
if (selectedTab) {
const { tabid } = selectedTab;
if (tabid && !mountedTabs[tabid])
setMountedTabs({
...mountedTabs,
[tabid]: createTabComponent(selectedTab),
});
}
}, [mountedTabs, files]);
return _.keys(mountedTabs).map(tabid => {
const { TabComponent, props } = mountedTabs[tabid];
const tabVisible = tabid == (selectedTab && selectedTab.tabid);
return (
<TabContainerMemo
key={tabid}
{...props}
tabid={tabid}
tabVisible={tabVisible}
toolbarPortalRef={toolbarPortalRef}
TabComponent={TabComponent}
/>
);
});
}

View File

@@ -0,0 +1,35 @@
<script lang="ts">
import { setContext } from 'svelte';
import { writable } from 'svelte/store';
export let tabid;
export let tabVisible;
export let tabComponent;
const tabVisibleStore = writable(tabVisible);
setContext('tabid', tabid);
setContext('tabVisible', tabVisibleStore);
$: tabVisibleStore.set(tabVisible);
</script>
<div class:tabVisible>
<svelte:component this={tabComponent} {...$$restProps} {tabid} {tabVisible} />
</div>
<style>
div {
position: absolute;
left: 0;
top: 0;
right: 0;
bottom: 0;
display: flex;
}
.tabVisible {
visibility: visible;
}
:not(.tabVisible) {
visibility: hidden;
}
</style>

Some files were not shown because too many files have changed in this diff Show More