@carbonorm/carbonnode 6.0.14 → 6.0.18

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 (77) hide show
  1. package/dist/executors/SqlExecutor.d.ts +17 -0
  2. package/dist/index.cjs.js +450 -248
  3. package/dist/index.cjs.js.map +1 -1
  4. package/dist/index.esm.js +450 -249
  5. package/dist/index.esm.js.map +1 -1
  6. package/dist/types/ormInterfaces.d.ts +1 -0
  7. package/dist/utils/cacheManager.d.ts +3 -1
  8. package/dist/utils/logLevel.d.ts +3 -3
  9. package/dist/utils/logSql.d.ts +10 -1
  10. package/package.json +2 -2
  11. package/scripts/assets/handlebars/C6.ts.handlebars +1 -1
  12. package/src/__tests__/fixtures/sqlResponses/sqlAllowList.json +1 -1
  13. package/src/__tests__/httpExecutor.cacheEviction.test.ts +70 -0
  14. package/src/__tests__/logSql.test.ts +54 -2
  15. package/src/__tests__/sakila-db/C6.js +1 -1
  16. package/src/__tests__/sakila-db/C6.mysqldump.json +1 -1
  17. package/src/__tests__/sakila-db/C6.mysqldump.sql +1 -1
  18. package/src/__tests__/sakila-db/C6.sqlAllowList.json +59 -70
  19. package/src/__tests__/sakila-db/C6.ts +2 -2
  20. package/src/__tests__/sakila-db/sqlResponses/C6.actor.post.json +3 -3
  21. package/src/__tests__/sakila-db/sqlResponses/C6.actor.post.latest.json +3 -3
  22. package/src/__tests__/sakila-db/sqlResponses/C6.actor.put.json +1 -1
  23. package/src/__tests__/sakila-db/sqlResponses/C6.actor.put.lookup.json +3 -3
  24. package/src/__tests__/sakila-db/sqlResponses/C6.address.post.json +5 -5
  25. package/src/__tests__/sakila-db/sqlResponses/C6.address.post.latest.json +5 -5
  26. package/src/__tests__/sakila-db/sqlResponses/C6.address.put.json +1 -1
  27. package/src/__tests__/sakila-db/sqlResponses/C6.address.put.lookup.json +5 -5
  28. package/src/__tests__/sakila-db/sqlResponses/C6.category.post.json +2 -2
  29. package/src/__tests__/sakila-db/sqlResponses/C6.category.post.latest.json +2 -2
  30. package/src/__tests__/sakila-db/sqlResponses/C6.category.put.json +1 -1
  31. package/src/__tests__/sakila-db/sqlResponses/C6.category.put.lookup.json +2 -2
  32. package/src/__tests__/sakila-db/sqlResponses/C6.city.post.json +2 -2
  33. package/src/__tests__/sakila-db/sqlResponses/C6.city.post.latest.json +2 -2
  34. package/src/__tests__/sakila-db/sqlResponses/C6.city.put.json +1 -1
  35. package/src/__tests__/sakila-db/sqlResponses/C6.city.put.lookup.json +2 -2
  36. package/src/__tests__/sakila-db/sqlResponses/C6.country.post.json +2 -2
  37. package/src/__tests__/sakila-db/sqlResponses/C6.country.post.latest.json +2 -2
  38. package/src/__tests__/sakila-db/sqlResponses/C6.country.put.json +1 -1
  39. package/src/__tests__/sakila-db/sqlResponses/C6.country.put.lookup.json +2 -2
  40. package/src/__tests__/sakila-db/sqlResponses/C6.customer.post.json +5 -5
  41. package/src/__tests__/sakila-db/sqlResponses/C6.customer.post.latest.json +5 -5
  42. package/src/__tests__/sakila-db/sqlResponses/C6.customer.put.json +1 -1
  43. package/src/__tests__/sakila-db/sqlResponses/C6.customer.put.lookup.json +5 -5
  44. package/src/__tests__/sakila-db/sqlResponses/C6.film.post.json +2 -2
  45. package/src/__tests__/sakila-db/sqlResponses/C6.film.post.latest.json +2 -2
  46. package/src/__tests__/sakila-db/sqlResponses/C6.film.put.json +1 -1
  47. package/src/__tests__/sakila-db/sqlResponses/C6.film.put.lookup.json +2 -2
  48. package/src/__tests__/sakila-db/sqlResponses/C6.inventory.post.json +1 -1
  49. package/src/__tests__/sakila-db/sqlResponses/C6.inventory.post.latest.json +1 -1
  50. package/src/__tests__/sakila-db/sqlResponses/C6.inventory.put.json +1 -1
  51. package/src/__tests__/sakila-db/sqlResponses/C6.inventory.put.lookup.json +1 -1
  52. package/src/__tests__/sakila-db/sqlResponses/C6.language.post.json +2 -2
  53. package/src/__tests__/sakila-db/sqlResponses/C6.language.post.latest.json +2 -2
  54. package/src/__tests__/sakila-db/sqlResponses/C6.language.put.json +1 -1
  55. package/src/__tests__/sakila-db/sqlResponses/C6.language.put.lookup.json +2 -2
  56. package/src/__tests__/sakila-db/sqlResponses/C6.payment.post.json +2 -2
  57. package/src/__tests__/sakila-db/sqlResponses/C6.payment.post.latest.json +2 -2
  58. package/src/__tests__/sakila-db/sqlResponses/C6.payment.put.lookup.json +2 -2
  59. package/src/__tests__/sakila-db/sqlResponses/C6.rental.join.json +10 -10
  60. package/src/__tests__/sakila-db/sqlResponses/C6.rental.post.json +3 -3
  61. package/src/__tests__/sakila-db/sqlResponses/C6.rental.post.latest.json +3 -3
  62. package/src/__tests__/sakila-db/sqlResponses/C6.rental.put.json +1 -1
  63. package/src/__tests__/sakila-db/sqlResponses/C6.rental.put.lookup.json +3 -3
  64. package/src/__tests__/sqlAllowList.test.ts +100 -0
  65. package/src/__tests__/sqlBuilders.test.ts +3 -4
  66. package/src/__tests__/sqlExecutor.cacheEviction.test.ts +79 -0
  67. package/src/executors/HttpExecutor.ts +20 -4
  68. package/src/executors/SqlExecutor.ts +131 -12
  69. package/src/orm/queries/DeleteQueryBuilder.ts +0 -4
  70. package/src/orm/queries/PostQueryBuilder.ts +0 -4
  71. package/src/orm/queries/SelectQueryBuilder.ts +0 -4
  72. package/src/orm/queries/UpdateQueryBuilder.ts +0 -4
  73. package/src/types/ormInterfaces.ts +4 -1
  74. package/src/utils/cacheManager.ts +26 -9
  75. package/src/utils/logLevel.ts +3 -4
  76. package/src/utils/logSql.ts +51 -6
  77. package/src/utils/sqlAllowList.ts +111 -9
@@ -125,6 +125,7 @@ export type C6RestResponse<Method extends iRestMethods, RestData extends {
125
125
  sql?: any;
126
126
  } & (Method extends 'GET' ? {
127
127
  next?: () => Promise<DetermineResponseDataType<'GET', RestData, Overrides>>;
128
+ evictFromCache?: () => boolean;
128
129
  } : {
129
130
  affected: number;
130
131
  insertId?: number | string;
@@ -1,8 +1,10 @@
1
1
  import type { iCacheAPI, iCacheResponse } from "../types/ormInterfaces";
2
+ import { LogContext } from "./logLevel";
2
3
  export declare const apiRequestCache: Map<string, iCacheAPI<any>>;
3
4
  export declare const userCustomClearCache: (() => void)[];
4
5
  export declare function clearCache(props?: {
5
6
  ignoreWarning?: boolean;
6
7
  }): void;
7
- export declare function checkCache<ResponseDataType = any>(method: string, tableName: string | string[], requestData: any): Promise<iCacheResponse<ResponseDataType>> | false;
8
+ export declare function checkCache<ResponseDataType = any>(method: string, tableName: string | string[], requestData: any, logContext: LogContext): Promise<iCacheResponse<ResponseDataType>> | false;
8
9
  export declare function setCache<ResponseDataType = any>(method: string, tableName: string | string[], requestData: any, cacheEntry: iCacheAPI<ResponseDataType>): void;
10
+ export declare function evictCacheEntry(method: string, tableName: string | string[], requestData: any): boolean;
@@ -23,10 +23,10 @@ export declare const applyLogLevelDefaults: (config: {
23
23
  }, request?: {
24
24
  debug?: boolean;
25
25
  } | null) => LogLevel;
26
- export declare const getLogContext: (config?: {
26
+ export declare const getLogContext: (config: {
27
27
  logLevel?: number | null;
28
28
  verbose?: boolean | null;
29
- }, request?: {
29
+ }, request: {
30
30
  debug?: boolean;
31
- } | null) => LogContext | undefined;
31
+ } | null) => LogContext;
32
32
  export declare const logWithLevel: (requiredLevel: LogLevel, context: LogContext | undefined, logger: (...args: any[]) => void, ...args: any[]) => void;
@@ -1,2 +1,11 @@
1
1
  import type { LogContext } from "./logLevel";
2
- export default function logSql(method: string, sql: string, context?: LogContext): void;
2
+ export type SqlAllowListStatus = "allowed" | "denied" | "not verified";
3
+ export type SqlCacheStatus = "hit" | "miss" | "ignored";
4
+ export type LogSqlContextOptions = {
5
+ cacheStatus: SqlCacheStatus;
6
+ allowListStatus: SqlAllowListStatus;
7
+ method: string;
8
+ sql: string;
9
+ context?: LogContext;
10
+ };
11
+ export default function logSql(options: LogSqlContextOptions): void;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@carbonorm/carbonnode",
3
- "version": "6.0.14",
3
+ "version": "6.0.18",
4
4
  "browser": "dist/index.umd.js",
5
5
  "repository": {
6
6
  "type": "git",
@@ -25,7 +25,7 @@
25
25
  "geojson": "^0.5.0",
26
26
  "handlebars": "^4.7.8",
27
27
  "named-placeholders": "^1.1.3",
28
- "qs": "^6.11.1",
28
+ "qs": "6.14.1",
29
29
  "tslib": "^2.8.1"
30
30
  },
31
31
  "peerDependencies": {
@@ -147,7 +147,7 @@ export const C6 : iC6Object<RestTableInterfaces> = {
147
147
  ...TABLES
148
148
  };
149
149
 
150
- export type tStatefulApiData<T> = T[] | undefined | null;
150
+ export type tStatefulApiData<T> = T[] | undefined;
151
151
 
152
152
  // this refers to the value types of the keys above, aka values in the state
153
153
  export interface iRestfulObjectArrayTypes {
@@ -1,3 +1,3 @@
1
1
  [
2
- "SELECT * FROM `actor` LIMIT 1"
2
+ "SELECT * FROM `actor` LIMIT ?"
3
3
  ]
@@ -0,0 +1,70 @@
1
+ import { beforeEach, describe, expect, it, vi } from "vitest";
2
+ import { C6C, restOrm, apiRequestCache, clearCache } from "@carbonorm/carbonnode";
3
+ import { buildTestConfig } from "./fixtures/c6.fixture";
4
+
5
+ describe("HttpExecutor cache eviction", () => {
6
+ const makeResponsePayload = () => ({
7
+ rest: [
8
+ {
9
+ actor_id: 1,
10
+ first_name: "ALICE",
11
+ last_name: "ONE",
12
+ },
13
+ ],
14
+ });
15
+
16
+ const buildOrm = (get: ReturnType<typeof vi.fn>) => {
17
+ const baseConfig = buildTestConfig() as any;
18
+
19
+ return restOrm<any>(() => ({
20
+ ...baseConfig,
21
+ requestMethod: C6C.GET,
22
+ restURL: "http://127.0.0.1:9999/rest/",
23
+ axios: { get },
24
+ verbose: false,
25
+ }));
26
+ };
27
+
28
+ beforeEach(() => {
29
+ clearCache({ ignoreWarning: true });
30
+ });
31
+
32
+ it("adds evictFromCache for cached GET responses", async () => {
33
+ const get = vi.fn().mockResolvedValue({ data: makeResponsePayload() });
34
+ const actorHttp = buildOrm(get);
35
+
36
+ const response = await actorHttp.Get({ actor_id: 1, cacheResults: true } as any);
37
+
38
+ expect(get).toHaveBeenCalledTimes(1);
39
+ expect(typeof response.evictFromCache).toBe("function");
40
+ expect(apiRequestCache.size).toBe(1);
41
+
42
+ expect(response.evictFromCache?.()).toBe(true);
43
+ expect(apiRequestCache.size).toBe(0);
44
+ expect(response.evictFromCache?.()).toBe(false);
45
+ });
46
+
47
+ it("does not add evictFromCache when cacheResults is false", async () => {
48
+ const get = vi.fn().mockResolvedValue({ data: makeResponsePayload() });
49
+ const actorHttp = buildOrm(get);
50
+
51
+ const response = await actorHttp.Get({ actor_id: 1, cacheResults: false } as any);
52
+
53
+ expect(get).toHaveBeenCalledTimes(1);
54
+ expect(response.evictFromCache).toBeUndefined();
55
+ expect(apiRequestCache.size).toBe(0);
56
+ });
57
+
58
+ it("keeps evictFromCache on cache hits", async () => {
59
+ const get = vi.fn().mockResolvedValue({ data: makeResponsePayload() });
60
+ const actorHttp = buildOrm(get);
61
+
62
+ await actorHttp.Get({ actor_id: 1, cacheResults: true } as any);
63
+ const cached = await actorHttp.Get({ actor_id: 1, cacheResults: true } as any);
64
+
65
+ expect(get).toHaveBeenCalledTimes(1);
66
+ expect(typeof cached.evictFromCache).toBe("function");
67
+ expect(cached.evictFromCache?.()).toBe(true);
68
+ expect(apiRequestCache.size).toBe(0);
69
+ });
70
+ });
@@ -1,3 +1,5 @@
1
+ // noinspection SqlResolve
2
+
1
3
  import { describe, it, expect, vi, afterEach } from "vitest";
2
4
  import logSql from "../utils/logSql";
3
5
  import colorSql from "../utils/colorSql";
@@ -35,13 +37,21 @@ describe("logSql", () => {
35
37
  process.env[LOG_LEVEL_KEY] = "DEBUG";
36
38
  const spy = vi.spyOn(console, "log").mockImplementation(() => {});
37
39
 
38
- logSql("SELECT", "SELECT * FROM `users`");
40
+ logSql({
41
+ method: "SELECT",
42
+ sql: "SELECT * FROM `users`",
43
+ cacheStatus: "miss",
44
+ allowListStatus: "not verified",
45
+ });
39
46
 
40
47
  expect(spy).toHaveBeenCalledTimes(1);
41
48
  const message = stripAnsi(String(spy.mock.calls[0][0]));
42
49
  expect(message).toContain(`[${version}]`);
50
+ expect(message).toContain("[CACHE MISS]");
51
+ expect(message).toContain("[NOT VERIFIED]");
43
52
  expect(message).toContain("[API]");
44
53
  expect(message).toContain("[SELECT]");
54
+ // noinspection SqlResolve - resolve not intended for this test
45
55
  expect(message).toContain("SELECT * FROM `users`");
46
56
  });
47
57
 
@@ -50,12 +60,53 @@ describe("logSql", () => {
50
60
  process.env[LOG_LEVEL_KEY] = "DEBUG";
51
61
  const spy = vi.spyOn(console, "log").mockImplementation(() => {});
52
62
 
53
- logSql("DELETE", "DELETE `users` FROM `users`");
63
+ logSql({
64
+ method: "DELETE",
65
+ sql: "DELETE `users` FROM `users`",
66
+ cacheStatus: "miss",
67
+ allowListStatus: "not verified",
68
+ });
54
69
 
55
70
  const message = stripAnsi(String(spy.mock.calls[0][0]));
71
+ expect(message).toContain("[CACHE MISS]");
72
+ expect(message).toContain("[NOT VERIFIED]");
56
73
  expect(message).toContain("[SSR]");
57
74
  expect(message).toContain("[DELETE]");
58
75
  });
76
+
77
+ it("supports explicit cache and allowlist indicators", () => {
78
+ process.env[SSR_ENV_KEY] = "false";
79
+ process.env[LOG_LEVEL_KEY] = "DEBUG";
80
+ const spy = vi.spyOn(console, "log").mockImplementation(() => {});
81
+
82
+ // noinspection SqlResolve - resolve not intended for this test
83
+ logSql({
84
+ method: "SELECT",
85
+ sql: "SELECT * FROM `users`",
86
+ cacheStatus: "ignored",
87
+ allowListStatus: "allowed",
88
+ });
89
+
90
+ const message = stripAnsi(String(spy.mock.calls[0][0]));
91
+ expect(message).toContain("[CACHE IGNORED]");
92
+ expect(message).toContain("[VERIFIED]");
93
+ });
94
+
95
+ it("supports cache-hit indicators", () => {
96
+ process.env[SSR_ENV_KEY] = "false";
97
+ process.env[LOG_LEVEL_KEY] = "DEBUG";
98
+ const spy = vi.spyOn(console, "log").mockImplementation(() => {});
99
+
100
+ logSql({
101
+ method: "SELECT",
102
+ sql: "SELECT * FROM `users`",
103
+ cacheStatus: "hit",
104
+ allowListStatus: "allowed",
105
+ });
106
+
107
+ const message = stripAnsi(String(spy.mock.calls[0][0]));
108
+ expect(message).toContain("[CACHE HIT]");
109
+ });
59
110
  });
60
111
 
61
112
  describe("colorSql", () => {
@@ -69,6 +120,7 @@ describe("colorSql", () => {
69
120
  });
70
121
 
71
122
  it("collapses repeated multi-row value groups", () => {
123
+ // noinspection SqlResolve - resolve not intended for this test
72
124
  const sql = `INSERT INTO \`valuation_report_comparables\` (a,b,c) VALUES
73
125
  (?, ?, ?),
74
126
  (?, ?, ?),
@@ -1342,7 +1342,7 @@ export const TABLES = {
1342
1342
  };
1343
1343
  export const C6 = {
1344
1344
  ...C6Constants,
1345
- C6VERSION: '6.0.14',
1345
+ C6VERSION: '6.0.18',
1346
1346
  IMPORT: async (tableName) => {
1347
1347
  tableName = tableName.toLowerCase();
1348
1348
  // if tableName is not a key in the TABLES object then throw an error