@saltcorn/server 0.9.6-beta.9 → 0.9.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (47) hide show
  1. package/app.js +9 -2
  2. package/auth/admin.js +51 -52
  3. package/auth/roleadmin.js +6 -2
  4. package/auth/routes.js +28 -10
  5. package/auth/testhelp.js +86 -0
  6. package/help/Field label.tmd +11 -0
  7. package/help/Field types.tmd +39 -0
  8. package/help/Inclusion Formula.tmd +38 -0
  9. package/help/Ownership field.tmd +76 -0
  10. package/help/Ownership formula.tmd +75 -0
  11. package/help/Table roles.tmd +20 -0
  12. package/help/User groups.tmd +35 -0
  13. package/help/User roles.tmd +30 -0
  14. package/load_plugins.js +28 -4
  15. package/locales/en.json +28 -1
  16. package/locales/it.json +3 -2
  17. package/markup/forms.js +5 -1
  18. package/package.json +9 -9
  19. package/public/log_viewer_utils.js +32 -0
  20. package/public/mermaid.min.js +705 -306
  21. package/public/saltcorn-builder.css +23 -0
  22. package/public/saltcorn-common.js +195 -71
  23. package/public/saltcorn.css +72 -0
  24. package/public/saltcorn.js +78 -0
  25. package/restart_watcher.js +1 -0
  26. package/routes/actions.js +27 -0
  27. package/routes/admin.js +180 -66
  28. package/routes/api.js +6 -0
  29. package/routes/common_lists.js +42 -32
  30. package/routes/fields.js +9 -1
  31. package/routes/homepage.js +2 -0
  32. package/routes/menu.js +69 -4
  33. package/routes/notifications.js +90 -10
  34. package/routes/pageedit.js +18 -13
  35. package/routes/plugins.js +5 -1
  36. package/routes/search.js +10 -4
  37. package/routes/tables.js +47 -27
  38. package/routes/tenant.js +4 -15
  39. package/routes/utils.js +20 -6
  40. package/routes/viewedit.js +11 -7
  41. package/serve.js +27 -5
  42. package/tests/edit.test.js +426 -0
  43. package/tests/fields.test.js +21 -0
  44. package/tests/filter.test.js +68 -0
  45. package/tests/page.test.js +2 -2
  46. package/tests/sync.test.js +59 -0
  47. package/wrapper.js +4 -1
@@ -0,0 +1,426 @@
1
+ const request = require("supertest");
2
+ const getApp = require("../app");
3
+ const { resetToFixtures, load_url_dom } = require("../auth/testhelp");
4
+ const db = require("@saltcorn/data/db");
5
+ const { getState } = require("@saltcorn/data/db/state");
6
+ const View = require("@saltcorn/data/models/view");
7
+ const Field = require("@saltcorn/data/models/field");
8
+ const Table = require("@saltcorn/data/models/table");
9
+ const { plugin_with_routes, sleep } = require("@saltcorn/data/tests/mocks");
10
+
11
+ afterAll(db.close);
12
+ beforeAll(async () => {
13
+ await resetToFixtures();
14
+ const table = Table.findOne("books");
15
+ await table.update({ min_role_read: 100 });
16
+ await Field.create({
17
+ table,
18
+ name: "sequel_to",
19
+ type: "Key to books",
20
+ attributes: { summary_field: "author" },
21
+ });
22
+ await Field.create({
23
+ table,
24
+ label: "pagesp1",
25
+ type: "Integer",
26
+ calculated: true,
27
+ expression: "pages+1",
28
+ });
29
+ await table.insertRow({
30
+ author: "Peter Kropotkin",
31
+ pages: 189,
32
+ publisher: 1,
33
+ });
34
+
35
+ await table.insertRow({
36
+ author: "Mary Boas",
37
+ pages: 864,
38
+ publisher: 2,
39
+ });
40
+ const ptable = Table.findOne("publisher");
41
+ await ptable.update({ min_role_read: 100 });
42
+
43
+ //await getState().setConfig("log_level", 6);
44
+ });
45
+
46
+ jest.setTimeout(30000);
47
+
48
+ const makeJoinSelectView = async ({ name, showIfFormula }) => {
49
+ await View.create({
50
+ viewtemplate: "Edit",
51
+ description: "",
52
+ min_role: 100,
53
+ name,
54
+ table_id: Table.findOne("books")?.id,
55
+ default_render_page: "",
56
+ slug: {},
57
+ attributes: {},
58
+ configuration: {
59
+ layout: {
60
+ above: [
61
+ {
62
+ gx: null,
63
+ gy: null,
64
+ style: {
65
+ "margin-bottom": "1.5rem",
66
+ },
67
+ aligns: ["end", "start"],
68
+ widths: [2, 10],
69
+ besides: [
70
+ {
71
+ font: "",
72
+ type: "blank",
73
+ block: false,
74
+ style: {},
75
+ inline: false,
76
+ contents: "Publisher",
77
+ labelFor: "publisher",
78
+ isFormula: {},
79
+ textStyle: "",
80
+ },
81
+ {
82
+ above: [
83
+ {
84
+ type: "field",
85
+ block: false,
86
+ fieldview: "select",
87
+ textStyle: "",
88
+ field_name: "publisher",
89
+ configuration: {},
90
+ },
91
+ {
92
+ type: "container",
93
+ style: {},
94
+ bgType: "None",
95
+ hAlign: "left",
96
+ margin: [0, 0, 0, 0],
97
+ rotate: 0,
98
+ vAlign: "top",
99
+ bgColor: "#ffffff",
100
+ display: "block",
101
+ padding: [0, 0, 0, 0],
102
+ bgFileId: 0,
103
+ contents: {
104
+ above: [
105
+ {
106
+ font: "",
107
+ icon: "",
108
+ type: "blank",
109
+ block: false,
110
+ style: {},
111
+ inline: false,
112
+ contents: "Warning",
113
+ labelFor: "",
114
+ isFormula: {},
115
+ textStyle: "",
116
+ },
117
+ {
118
+ type: "join_field",
119
+ block: false,
120
+ fieldview: "as_text",
121
+ textStyle: "",
122
+ join_field: "publisher.name",
123
+ configuration: {},
124
+ },
125
+ ],
126
+ },
127
+ imageSize: "contain",
128
+ isFormula: {},
129
+ minHeight: 0,
130
+ textColor: "#ffffff",
131
+ widthUnit: "px",
132
+ heightUnit: "px",
133
+ customClass: "pubwarn",
134
+ htmlElement: "div",
135
+ showForRole: [],
136
+ gradEndColor: "#88ff88",
137
+ setTextColor: false,
138
+ fullPageWidth: false,
139
+ gradDirection: "0",
140
+ minHeightUnit: "px",
141
+ showIfFormula,
142
+ gradStartColor: "#ff8888",
143
+ maxScreenWidth: "",
144
+ minScreenWidth: "",
145
+ show_for_owner: false,
146
+ },
147
+ ],
148
+ },
149
+ ],
150
+ breakpoints: ["", ""],
151
+ },
152
+ {
153
+ gx: null,
154
+ gy: null,
155
+ style: {
156
+ "margin-bottom": "1.5rem",
157
+ },
158
+ aligns: ["end", "start"],
159
+ widths: [2, 10],
160
+ besides: [
161
+ {
162
+ font: "",
163
+ type: "blank",
164
+ block: false,
165
+ style: {},
166
+ inline: false,
167
+ contents: "sequel_to",
168
+ labelFor: "sequel_to",
169
+ isFormula: {},
170
+ textStyle: "",
171
+ },
172
+ {
173
+ type: "field",
174
+ block: false,
175
+ fieldview: "select",
176
+ textStyle: "",
177
+ field_name: "sequel_to",
178
+ configuration: {
179
+ where: "publisher == $publisher",
180
+ },
181
+ },
182
+ ],
183
+ breakpoints: ["", ""],
184
+ },
185
+ {
186
+ gx: null,
187
+ gy: null,
188
+ style: {
189
+ "margin-bottom": "1.5rem",
190
+ },
191
+ aligns: ["end", "start"],
192
+ widths: [2, 10],
193
+ besides: [
194
+ {
195
+ font: "",
196
+ icon: "",
197
+ type: "blank",
198
+ block: false,
199
+ style: {},
200
+ inline: false,
201
+ contents: "Pages",
202
+ labelFor: "sequel_to",
203
+ isFormula: {},
204
+ textStyle: "",
205
+ },
206
+ {
207
+ above: [
208
+ {
209
+ type: "field",
210
+ block: false,
211
+ fieldview: "edit",
212
+ textStyle: "",
213
+ field_name: "pages",
214
+ configuration: {
215
+ where: "publisher == $publisher",
216
+ },
217
+ },
218
+ {
219
+ type: "field",
220
+ block: false,
221
+ fieldview: "show",
222
+ textStyle: "",
223
+ field_name: "pagesp1",
224
+ configuration: {
225
+ input_type: "text",
226
+ },
227
+ },
228
+ ],
229
+ },
230
+ ],
231
+ breakpoints: ["", ""],
232
+ },
233
+ {
234
+ type: "action",
235
+ block: false,
236
+ rndid: "cb94bd",
237
+ nsteps: "",
238
+ minRole: 100,
239
+ isFormula: {},
240
+ action_icon: "",
241
+ action_name: "Save",
242
+ action_size: "",
243
+ action_bgcol: "",
244
+ action_label: "",
245
+ action_style: "btn-primary",
246
+ action_title: "",
247
+ configuration: {},
248
+ step_only_ifs: "",
249
+ action_textcol: "",
250
+ action_bordercol: "",
251
+ step_action_names: "",
252
+ },
253
+ ],
254
+ },
255
+ columns: [
256
+ {
257
+ type: "Field",
258
+ block: false,
259
+ fieldview: "select",
260
+ textStyle: "",
261
+ field_name: "publisher",
262
+ configuration: {},
263
+ },
264
+ {
265
+ type: "JoinField",
266
+ block: false,
267
+ fieldview: "as_text",
268
+ textStyle: "",
269
+ join_field: "publisher.name",
270
+ configuration: {},
271
+ },
272
+ {
273
+ type: "Field",
274
+ block: false,
275
+ fieldview: "select",
276
+ textStyle: "",
277
+ field_name: "sequel_to",
278
+ configuration: {
279
+ where: "publisher == $publisher",
280
+ },
281
+ },
282
+ {
283
+ type: "Field",
284
+ block: false,
285
+ fieldview: "edit",
286
+ textStyle: "",
287
+ field_name: "pages",
288
+ configuration: {
289
+ where: "publisher == $publisher",
290
+ },
291
+ },
292
+ {
293
+ type: "Field",
294
+ block: false,
295
+ fieldview: "show",
296
+ textStyle: "",
297
+ field_name: "pagesp1",
298
+ configuration: {
299
+ input_type: "text",
300
+ },
301
+ },
302
+ {
303
+ type: "Action",
304
+ rndid: "cb94bd",
305
+ nsteps: "",
306
+ minRole: 100,
307
+ isFormula: {},
308
+ action_icon: "",
309
+ action_name: "Save",
310
+ action_size: "",
311
+ action_bgcol: "",
312
+ action_label: "",
313
+ action_style: "btn-primary",
314
+ action_title: "",
315
+ configuration: {},
316
+ step_only_ifs: "",
317
+ action_textcol: "",
318
+ action_bordercol: "",
319
+ step_action_names: "",
320
+ },
321
+ ],
322
+ viewname: "AuthorEditForTest",
323
+ auto_save: false,
324
+ split_paste: false,
325
+ exttable_name: null,
326
+ page_when_done: null,
327
+ view_when_done: "authorlist",
328
+ dest_url_formula: null,
329
+ destination_type: "View",
330
+ formula_destinations: [],
331
+ page_group_when_done: null,
332
+ },
333
+ });
334
+ };
335
+
336
+ const newEvent = (dom, type) =>
337
+ new dom.window.CustomEvent(type, {
338
+ bubbles: true,
339
+ cancelable: true,
340
+ });
341
+
342
+ describe("JSDOM-E2E edit test", () => {
343
+ it("join select should set dynamic where and show if with joinfield", async () => {
344
+ await makeJoinSelectView({
345
+ name: "AuthorEditForTest",
346
+ showIfFormula: 'publisher?.name == "AK Press"',
347
+ });
348
+ const dom = await load_url_dom("/view/AuthorEditForTest");
349
+ await sleep(1000);
350
+ const pubwarn = dom.window.document.querySelector("div.pubwarn");
351
+ //console.log(dom.serialize());
352
+ expect(pubwarn.style.display).toBe("none");
353
+
354
+ const select_seq = dom.window.document.querySelector(
355
+ "select[name=sequel_to]"
356
+ );
357
+ expect([...select_seq.options].map((o) => o.text)).toStrictEqual([
358
+ "",
359
+ "Herman Melville",
360
+ ]);
361
+ const select = dom.window.document.querySelector("select[name=publisher]");
362
+ select.value = "1";
363
+ select.dispatchEvent(newEvent(dom, "change"));
364
+
365
+ await sleep(1000);
366
+ expect([...select_seq.options].map((o) => o.text)).toStrictEqual([
367
+ "",
368
+ "Leo Tolstoy",
369
+ "Peter Kropotkin",
370
+ ]);
371
+
372
+ expect(pubwarn.style.display).toBe("");
373
+
374
+ const jf = dom.window.document.querySelector(
375
+ "div.pubwarn div[data-source-url]"
376
+ );
377
+ expect(jf.innerHTML).toBe("AK Press");
378
+ });
379
+ it("calculated field", async () => {
380
+ const dom = await load_url_dom("/view/AuthorEditForTest");
381
+ const input = dom.window.document.querySelector("input[name=pages]");
382
+ input.value = "13";
383
+ input.dispatchEvent(newEvent(dom, "change"));
384
+ await sleep(1000);
385
+ const cf = dom.window.document.querySelector(
386
+ `div[data-source-url="/field/show-calculated/books/pagesp1/show?input_type=text"]`
387
+ );
388
+ expect(cf.innerHTML).toBe("14");
389
+ });
390
+
391
+ it("join select should set dynamic where and show if with no joinfield", async () => {
392
+ await makeJoinSelectView({
393
+ name: "AuthorEditForTest1",
394
+ showIfFormula: "publisher == 1",
395
+ });
396
+ const dom = await load_url_dom("/view/AuthorEditForTest1");
397
+ await sleep(1000);
398
+ const pubwarn = dom.window.document.querySelector("div.pubwarn");
399
+
400
+ expect(pubwarn.style.display).toBe("none");
401
+
402
+ const select_seq = dom.window.document.querySelector(
403
+ "select[name=sequel_to]"
404
+ );
405
+ expect([...select_seq.options].map((o) => o.text)).toStrictEqual([
406
+ "",
407
+ "Herman Melville",
408
+ ]);
409
+ const select = dom.window.document.querySelector("select[name=publisher]");
410
+ select.value = "1";
411
+ select.dispatchEvent(newEvent(dom, "change"));
412
+
413
+ await sleep(1000);
414
+ expect([...select_seq.options].map((o) => o.text)).toStrictEqual([
415
+ "",
416
+ "Leo Tolstoy",
417
+ "Peter Kropotkin",
418
+ ]);
419
+
420
+ expect(pubwarn.style.display).toBe("");
421
+ const jf = dom.window.document.querySelector(
422
+ "div.pubwarn div[data-source-url]"
423
+ );
424
+ expect(jf.innerHTML).toBe("AK Press");
425
+ });
426
+ });
@@ -391,6 +391,27 @@ describe("Field Endpoints", () => {
391
391
  })
392
392
  .expect(toBeTrue((r) => +r.text > 2));
393
393
  });
394
+ it("should show calculated field with two single joinfields", async () => {
395
+ const loginCookie = await getAdminLoginCookie();
396
+ const table = Table.findOne({ name: "patients" });
397
+ await Field.create({
398
+ table,
399
+ label: "pagesp12",
400
+ type: "Integer",
401
+ calculated: true,
402
+ stored: true,
403
+ expression: "favbook.pages+1+favbook.id",
404
+ });
405
+ const app = await getApp({ disableCsrf: true });
406
+
407
+ await request(app)
408
+ .post("/field/show-calculated/patients/pagesp12/show")
409
+ .set("Cookie", loginCookie)
410
+ .send({
411
+ id: 1,
412
+ })
413
+ .expect(toBeTrue((r) => +r.text > 2));
414
+ });
394
415
  it("should show calculated field with double joinfield", async () => {
395
416
  const loginCookie = await getAdminLoginCookie();
396
417
  const table = Table.findOne({ name: "readings" });
@@ -0,0 +1,68 @@
1
+ const request = require("supertest");
2
+ const getApp = require("../app");
3
+ const { resetToFixtures, load_url_dom } = require("../auth/testhelp");
4
+ const db = require("@saltcorn/data/db");
5
+ const { getState } = require("@saltcorn/data/db/state");
6
+ const View = require("@saltcorn/data/models/view");
7
+ const Table = require("@saltcorn/data/models/table");
8
+ const { plugin_with_routes, sleep } = require("@saltcorn/data/tests/mocks");
9
+
10
+ afterAll(db.close);
11
+ beforeAll(async () => {
12
+ await resetToFixtures();
13
+ });
14
+
15
+ jest.setTimeout(30000);
16
+
17
+ describe("JSDOM-E2E filter test", () => {
18
+ it("should load authorlist", async () => {
19
+ const dom = await load_url_dom("/view/authorlist");
20
+ //console.log("dom", dom);
21
+ });
22
+ it("should user filter to change url", async () => {
23
+ await View.create({
24
+ viewtemplate: "Filter",
25
+ description: "",
26
+ min_role: 100,
27
+ name: `authorfilter1`,
28
+ table_id: Table.findOne("books")?.id,
29
+ default_render_page: "",
30
+ slug: {},
31
+ attributes: {},
32
+ configuration: {
33
+ layout: {
34
+ type: "field",
35
+ block: false,
36
+ fieldview: "edit",
37
+ textStyle: "",
38
+ field_name: "author",
39
+ configuration: {},
40
+ },
41
+ columns: [
42
+ {
43
+ type: "Field",
44
+ block: false,
45
+ fieldview: "edit",
46
+ textStyle: "",
47
+ field_name: "author",
48
+ configuration: {},
49
+ },
50
+ ],
51
+ },
52
+ });
53
+ const dom = await load_url_dom("/view/authorfilter1");
54
+ expect(dom.window.location.href).toBe(
55
+ "http://localhost/view/authorfilter1"
56
+ );
57
+ //console.log(dom.serialize());
58
+ const input = dom.window.document.querySelector("input[name=author]");
59
+ input.value = "Leo";
60
+ input.dispatchEvent(new dom.window.Event("change"));
61
+ await sleep(1000);
62
+ expect(dom.window.location.href).toBe(
63
+ "http://localhost/view/authorfilter1?author=Leo"
64
+ );
65
+
66
+ //console.log("dom", dom);
67
+ });
68
+ });
@@ -76,7 +76,7 @@ describe("page create", () => {
76
76
  await request(app)
77
77
  .get("/pageedit/new")
78
78
  .set("Cookie", loginCookie)
79
- .expect(toInclude("A short name that will be in your URL"));
79
+ .expect(toInclude("A short name that will be in the page URL"));
80
80
  });
81
81
  it("shows new with html file selector", async () => {
82
82
  const app = await getApp({ disableCsrf: true });
@@ -237,7 +237,7 @@ describe("pageedit", () => {
237
237
  await request(app)
238
238
  .get("/pageedit/edit-properties/a_page")
239
239
  .set("Cookie", loginCookie)
240
- .expect(toInclude("A short name that will be in your URL"));
240
+ .expect(toInclude("A short name that will be in the page URL"));
241
241
 
242
242
  //TODO full context
243
243
  const ctx = encodeURIComponent(JSON.stringify({}));
@@ -9,6 +9,7 @@ const {
9
9
  toInclude,
10
10
  toSucceed,
11
11
  } = require("../auth/testhelp");
12
+ const load_plugins = require("../load_plugins");
12
13
  const db = require("@saltcorn/data/db");
13
14
  const { sleep } = require("@saltcorn/data/tests/mocks");
14
15
 
@@ -16,6 +17,7 @@ const Table = require("@saltcorn/data/models/table");
16
17
  const TableConstraint = require("@saltcorn/data/models/table_constraints");
17
18
  const Field = require("@saltcorn/data/models/field");
18
19
  const User = require("@saltcorn/data/models/user");
20
+ const Plugin = require("@saltcorn/data/models/plugin");
19
21
 
20
22
  beforeAll(async () => {
21
23
  await resetToFixtures();
@@ -35,6 +37,27 @@ const initSyncInfo = async (tbls) => {
35
37
  }
36
38
  };
37
39
 
40
+ const createAnswersTbl = async () => {
41
+ await load_plugins.loadAndSaveNewPlugin(
42
+ new Plugin({
43
+ name: "json",
44
+ source: "npm",
45
+ location: "@saltcorn/json",
46
+ version: "latest",
47
+ })
48
+ );
49
+ const table = await Table.create("Answers", {
50
+ min_role_read: 100,
51
+ min_role_write: 100,
52
+ });
53
+ await Field.create({
54
+ table,
55
+ name: "answer",
56
+ label: "Answer",
57
+ type: "JSON",
58
+ });
59
+ };
60
+
38
61
  describe("load remote insert/updates", () => {
39
62
  if (!db.isSQLite) {
40
63
  beforeAll(async () => {
@@ -416,6 +439,42 @@ describe("Upload changes", () => {
416
439
  });
417
440
  });
418
441
 
442
+ it("upload json", async () => {
443
+ await createAnswersTbl();
444
+ const table = Table.findOne({ name: "Answers" });
445
+ expect(table).toBeDefined();
446
+ const app = await getApp({ disableCsrf: true });
447
+ const loginCookie = await getAdminLoginCookie();
448
+ const rows = [
449
+ {
450
+ id: 1,
451
+ answer: true,
452
+ },
453
+ {
454
+ id: 2,
455
+ answer: false,
456
+ },
457
+ {
458
+ id: 3,
459
+ answer: 1,
460
+ },
461
+ {
462
+ id: 4,
463
+ answer: ["latte", "americano", "filter"],
464
+ },
465
+ ];
466
+ const resp = await doUpload(app, loginCookie, new Date().valueOf(), {
467
+ Answers: {
468
+ inserts: rows,
469
+ },
470
+ });
471
+ expect(resp.status).toBe(200);
472
+ const { syncDir } = resp._body;
473
+ const result = await getResult(app, loginCookie, syncDir);
474
+ expect(result).toBeDefined();
475
+ expect(await table.getRows()).toEqual(rows);
476
+ });
477
+
419
478
  it("handles inserts with TableConstraint conflicts", async () => {
420
479
  const books = Table.findOne({ name: "books" });
421
480
  const oldCount = await books.countRows();
package/wrapper.js CHANGED
@@ -170,6 +170,7 @@ const get_headers = (req, version_tag, description, extras = []) => {
170
170
  const favicon = state.getConfig("favicon_id", null);
171
171
  const notification_in_menu = state.getConfig("notification_in_menu");
172
172
  const pwa_enabled = state.getConfig("pwa_enabled");
173
+ const is_root = req.user?.role_id === 1;
173
174
 
174
175
  const iconHeader = favicon
175
176
  ? [
@@ -219,7 +220,9 @@ const get_headers = (req, version_tag, description, extras = []) => {
219
220
  from_cfg.push({ scriptBody: domReady(`check_saltcorn_notifications()`) });
220
221
  if (pwa_enabled) {
221
222
  from_cfg.push({
222
- headerTag: `<link rel="manifest" href="/notifications/manifest.json">`,
223
+ headerTag: `<link rel="manifest" href="/notifications/manifest.json${
224
+ is_root ? new Date().valueOf() : ""
225
+ }">`,
223
226
  });
224
227
  from_cfg.push({
225
228
  scriptBody: `if('serviceWorker' in navigator) {