@prestizni-software/server-dem 0.2.58 → 0.2.60

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 (33) hide show
  1. package/.npmignore +148 -0
  2. package/AutoUpdateManagerClass.ts +87 -0
  3. package/AutoUpdateServerManagerClass.ts +309 -0
  4. package/AutoUpdatedClientObjectClass.ts +673 -0
  5. package/AutoUpdatedServerObjectClass.ts +129 -0
  6. package/CHANGELOG.md +4 -0
  7. package/CommonTypes.ts +105 -0
  8. package/CommonTypes_server.ts +81 -0
  9. package/{AutoUpdatedClientObjectClass.d.ts → dist/AutoUpdatedClientObjectClass.d.ts} +1 -2
  10. package/{AutoUpdatedClientObjectClass.js.map → dist/AutoUpdatedClientObjectClass.js.map} +1 -1
  11. package/dist/tsconfig.tsbuildinfo +1 -0
  12. package/package.json +9 -6
  13. package/server.ts +13 -0
  14. package/tsconfig.json +113 -0
  15. /package/{AutoUpdateManagerClass.d.ts → dist/AutoUpdateManagerClass.d.ts} +0 -0
  16. /package/{AutoUpdateManagerClass.js → dist/AutoUpdateManagerClass.js} +0 -0
  17. /package/{AutoUpdateManagerClass.js.map → dist/AutoUpdateManagerClass.js.map} +0 -0
  18. /package/{AutoUpdateServerManagerClass.d.ts → dist/AutoUpdateServerManagerClass.d.ts} +0 -0
  19. /package/{AutoUpdateServerManagerClass.js → dist/AutoUpdateServerManagerClass.js} +0 -0
  20. /package/{AutoUpdateServerManagerClass.js.map → dist/AutoUpdateServerManagerClass.js.map} +0 -0
  21. /package/{AutoUpdatedClientObjectClass.js → dist/AutoUpdatedClientObjectClass.js} +0 -0
  22. /package/{AutoUpdatedServerObjectClass.d.ts → dist/AutoUpdatedServerObjectClass.d.ts} +0 -0
  23. /package/{AutoUpdatedServerObjectClass.js → dist/AutoUpdatedServerObjectClass.js} +0 -0
  24. /package/{AutoUpdatedServerObjectClass.js.map → dist/AutoUpdatedServerObjectClass.js.map} +0 -0
  25. /package/{CommonTypes.d.ts → dist/CommonTypes.d.ts} +0 -0
  26. /package/{CommonTypes.js → dist/CommonTypes.js} +0 -0
  27. /package/{CommonTypes.js.map → dist/CommonTypes.js.map} +0 -0
  28. /package/{CommonTypes_server.d.ts → dist/CommonTypes_server.d.ts} +0 -0
  29. /package/{CommonTypes_server.js → dist/CommonTypes_server.js} +0 -0
  30. /package/{CommonTypes_server.js.map → dist/CommonTypes_server.js.map} +0 -0
  31. /package/{server.d.ts → dist/server.d.ts} +0 -0
  32. /package/{server.js → dist/server.js} +0 -0
  33. /package/{server.js.map → dist/server.js.map} +0 -0
package/.npmignore ADDED
@@ -0,0 +1,148 @@
1
+ # Logs
2
+ logs
3
+ *.log
4
+ npm-debug.log*
5
+ yarn-debug.log*
6
+ yarn-error.log*
7
+ lerna-debug.log*
8
+
9
+ # Diagnostic reports (https://nodejs.org/api/report.html)
10
+ report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
11
+
12
+ # Runtime data
13
+ pids
14
+ *.pid
15
+ *.seed
16
+ *.pid.lock
17
+
18
+ # Directory for instrumented libs generated by jscoverage/JSCover
19
+ lib-cov
20
+
21
+ # Coverage directory used by tools like istanbul
22
+ coverage
23
+ *.lcov
24
+
25
+ # nyc test coverage
26
+ .nyc_output
27
+
28
+ # Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
29
+ .grunt
30
+
31
+ # Bower dependency directory (https://bower.io/)
32
+ bower_components
33
+
34
+ # node-waf configuration
35
+ .lock-wscript
36
+
37
+ # Compiled binary addons (https://nodejs.org/api/addons.html)
38
+ build/Release
39
+
40
+ # Dependency directories
41
+ node_modules/
42
+ jspm_packages/
43
+
44
+ # Snowpack dependency directory (https://snowpack.dev/)
45
+ web_modules/
46
+
47
+ # TypeScript cache
48
+ *.tsbuildinfo
49
+
50
+ # Optional npm cache directory
51
+ .npm
52
+
53
+ # Optional eslint cache
54
+ .eslintcache
55
+
56
+ # Optional stylelint cache
57
+ .stylelintcache
58
+
59
+ # Optional REPL history
60
+ .node_repl_history
61
+
62
+ # Output of 'npm pack'
63
+ *.tgz
64
+
65
+ # Yarn Integrity file
66
+ .yarn-integrity
67
+
68
+ # dotenv environment variable files
69
+ .env
70
+ .env.*
71
+ !.env.example
72
+
73
+ # parcel-bundler cache (https://parceljs.org/)
74
+ .cache
75
+ .parcel-cache
76
+
77
+ # Next.js build output
78
+ .next
79
+ out
80
+
81
+ # Nuxt.js build / generate output
82
+ .nuxt
83
+ .output
84
+
85
+ # Gatsby files
86
+ .cache/
87
+ # Comment in the public line in if your project uses Gatsby and not Next.js
88
+ # https://nextjs.org/blog/next-9-1#public-directory-support
89
+ # public
90
+
91
+ # vuepress build output
92
+ .vuepress/dist
93
+
94
+ # vuepress v2.x temp and cache directory
95
+ .temp
96
+ .cache
97
+
98
+ # Sveltekit cache directory
99
+ .svelte-kit/
100
+
101
+ # vitepress build output
102
+ **/.vitepress/dist
103
+
104
+ # vitepress cache directory
105
+ **/.vitepress/cache
106
+
107
+ # Docusaurus cache and generated files
108
+ .docusaurus
109
+
110
+ # Serverless directories
111
+ .serverless/
112
+
113
+ # FuseBox cache
114
+ .fusebox/
115
+
116
+ # DynamoDB Local files
117
+ .dynamodb/
118
+
119
+ # Firebase cache directory
120
+ .firebase/
121
+
122
+ # TernJS port file
123
+ .tern-port
124
+
125
+ server/
126
+ client/
127
+ # Stores VSCode versions used for testing VSCode extensions
128
+ .vscode-test
129
+
130
+ # yarn v3
131
+ .pnp.*
132
+ .yarn/*
133
+ !.yarn/patches
134
+ !.yarn/plugins
135
+ !.yarn/releases
136
+ !.yarn/sdks
137
+ !.yarn/versions
138
+
139
+ # Vite files
140
+ vite.config.js.timestamp-*
141
+ vite.config.ts.timestamp-*
142
+ .vite/
143
+
144
+ # ignore the .ts files
145
+ *.ts
146
+
147
+ # include the .d.ts files
148
+ !*.d.ts
@@ -0,0 +1,87 @@
1
+ import { AutoUpdated } from "./AutoUpdatedClientObjectClass.js";
2
+ import { Constructor, EventEmitter3, IsData, LoggersType, LoggersTypeInternal } from "./CommonTypes.js";
3
+ import "reflect-metadata";
4
+ export abstract class AutoUpdateManager<T extends Constructor<any>> {
5
+ protected classes: { [_id: string]: AutoUpdated<T> } = {};
6
+ public socket: any;
7
+ protected classParam: T;
8
+ protected properties: (keyof T)[];
9
+ public readonly classers: Record<string, AutoUpdateManager<any>>;
10
+ protected loggers: LoggersTypeInternal = {
11
+ info: () => {},
12
+ debug: () => {},
13
+ error: () => {},
14
+ warn: () => {},
15
+ };
16
+ protected classesAsArray: AutoUpdated<T>[] = [];
17
+ protected emitter: EventEmitter3;
18
+ private isLoaded = false;
19
+ constructor(
20
+ classParam: T,
21
+ socket: any,
22
+ loggers: LoggersType,
23
+ classers: Record<string, AutoUpdateManager<any>>,
24
+ emitter: EventEmitter3
25
+ ) {
26
+ this.emitter = emitter;
27
+ this.emitter.on("*",(e) =>
28
+ {
29
+ console.log("a");
30
+
31
+ })
32
+ this.classers = classers;
33
+ this.socket = socket;
34
+ this.classParam = classParam;
35
+ this.properties = Reflect.getMetadata(
36
+ "props",
37
+ Object.getPrototypeOf(classParam)
38
+ );
39
+ loggers.warn = loggers.warn ?? loggers.info;
40
+ this.loggers = loggers as LoggersTypeInternal;
41
+ }
42
+
43
+
44
+ public getObject(
45
+ _id: string
46
+ ): AutoUpdated<T> | null {
47
+ return this.classes[_id];
48
+ }
49
+
50
+ public async isLoadedAsync(): Promise<boolean> {
51
+ if (this.isLoaded) return this.isLoaded;
52
+ await new Promise((resolve) => this.emitter.on("ManagerLoaded"+this.classParam.name+this.className, resolve));
53
+ this.isLoaded = true;
54
+ return this.isLoaded;
55
+ }
56
+
57
+ public async deleteObject(_id: string): Promise<void> {
58
+ if (typeof this.classes[_id] === "string")
59
+ this.classes[_id] = await this.handleGetMissingObject(this.classes[_id]);
60
+ (this.classes[_id]).destroy();
61
+ this.classesAsArray.splice(this.classesAsArray.indexOf(this.classes[_id]), 1);
62
+ delete this.classes[_id];
63
+ }
64
+
65
+ public get objectIDs(): string[] {
66
+ return Object.keys(this.classes);
67
+ }
68
+
69
+ public get objects(): { [_id: string]: AutoUpdated<T> | string } {
70
+ return this.classes;
71
+ }
72
+
73
+ public get objectsAsArray(): AutoUpdated<T>[] {
74
+ return this.classesAsArray;
75
+ }
76
+
77
+ public get className(): string {
78
+ return this.classParam.name;
79
+ }
80
+
81
+ protected abstract handleGetMissingObject(
82
+ _id: string
83
+ ): Promise<AutoUpdated<T>>;
84
+ public abstract createObject(
85
+ data: IsData<InstanceType<T>>
86
+ ): Promise<AutoUpdated<T>>;
87
+ }
@@ -0,0 +1,309 @@
1
+ import { Server, Socket } from "socket.io";
2
+ import { AutoUpdateManager } from "./AutoUpdateManagerClass.js";
3
+ import { createAutoUpdatedClass } from "./AutoUpdatedServerObjectClass.js";
4
+ import {
5
+ Constructor,
6
+ EventEmitter3,
7
+ IsData,
8
+ LoggersType,
9
+ ServerResponse,
10
+ ServerUpdateRequest,
11
+ } from "./CommonTypes.js";
12
+ import { BeAnObject, ReturnModelType } from "@typegoose/typegoose/lib/types.js";
13
+ import { getModelForClass } from "@typegoose/typegoose";
14
+ import EventEmitter from "eventemitter3";
15
+ import { Paths } from "./CommonTypes_server.js";
16
+
17
+ export type WrappedInstances<T extends Record<string, Constructor<any>>> = {
18
+ [K in keyof T]: AutoUpdateServerManager<T[K]>;
19
+ };
20
+
21
+ export type AutoStatusDefinitions<
22
+ C extends Constructor<any>,
23
+ E extends Record<string, string | number>,
24
+ K extends keyof E
25
+ > = {
26
+ statusProperty: Paths<C>;
27
+ statusEnum: E;
28
+ definition: (data: InstanceType<C>) => Promise<E[K] | void>;
29
+ };
30
+
31
+ export function createAutoStatusDefinitions<
32
+ C extends Constructor<any>,
33
+ E extends { [k: string]: string | number },
34
+ K extends keyof E
35
+ >(
36
+ _class: C,
37
+ statusProperty: Paths<C>,
38
+ statusEnum: E,
39
+ definition: (data: InstanceType<C>) => Promise<E[K] | void>
40
+ ): AutoStatusDefinitions<C, E, keyof E> {
41
+ return {
42
+ statusProperty,
43
+ statusEnum,
44
+ definition,
45
+ };
46
+ }
47
+
48
+ export type AUSDefinitions<T extends Record<string, Constructor<any>>> = {
49
+ [K in keyof T]: ServerManagerDefinition<T[K], T>;
50
+ };
51
+
52
+ export type AUSOption<
53
+ C extends Constructor<any>,
54
+ T extends Record<string, Constructor<any>>
55
+ > = {
56
+ accessDefinitions?: (
57
+ data: InstanceType<C>,
58
+ managers: { [K in keyof T]: AutoUpdateServerManager<T[K]> },
59
+ userId: string
60
+ ) => Promise<IsData<InstanceType<C>>>;
61
+ autoStatusDefinitions?: AutoStatusDefinitions<
62
+ C,
63
+ { [k: string]: string | number },
64
+ keyof { [k: string]: string | number }
65
+ >;
66
+ };
67
+
68
+ export type ServerManagerDefinition<
69
+ C extends Constructor<any>,
70
+ T extends Record<string, Constructor<any>>
71
+ > = {
72
+ class: C;
73
+ options?: AUSOption<C, T>;
74
+ };
75
+
76
+ export async function AUSManagerFactory<
77
+ T extends Record<string, Constructor<any>>
78
+ >(
79
+ defs: AUSDefinitions<T>,
80
+ loggers: LoggersType,
81
+ socket: Server,
82
+ emitter: EventEmitter3 = new EventEmitter()
83
+ ): Promise<{ [K in keyof T]: AutoUpdateServerManager<T[K]> }> {
84
+ const classers: any = {};
85
+ let i = 0;
86
+ for (const key in defs) {
87
+ loggers.debug(`Creating manager for ${key}`);
88
+ const def = defs[key];
89
+ try {
90
+ const c = new AutoUpdateServerManager(
91
+ def.class,
92
+ loggers,
93
+ socket,
94
+ getModelForClass(def.class),
95
+ classers,
96
+ emitter,
97
+ def.options
98
+ ) as any;
99
+ i++;
100
+ classers[key] = c;
101
+ } catch (error) {
102
+ loggers.error("Error creating manager: " + key);
103
+ loggers.error(error);
104
+ continue;
105
+ }
106
+ loggers.debug("Loading DB for manager: " + key);
107
+ try {
108
+ classers[key].loadDB();
109
+ } catch (error) {
110
+ loggers.error("Error loading DB for manager: " + key);
111
+ loggers.error(error);
112
+ }
113
+ }
114
+ socket.on("connection", async (socket) => {
115
+ loggers.debug(`Client connected: ${socket.id}`);
116
+ for (const manager of Object.values(classers)) {
117
+ (manager as any).registerSocket(socket);
118
+ }
119
+ // Client disconnect
120
+ socket.on("disconnect", () => {
121
+ loggers.debug(`Client disconnected: ${socket.id}`);
122
+ });
123
+ });
124
+ return classers as WrappedInstances<T>;
125
+ }
126
+
127
+ export class AutoUpdateServerManager<
128
+ T extends Constructor<any>
129
+ > extends AutoUpdateManager<T> {
130
+ public readonly model: ReturnModelType<T, BeAnObject>;
131
+ private readonly clientSockets: Set<Socket> = new Set<Socket>();
132
+ public readonly options?: AUSOption<T, any>;
133
+ private readonly doDebug = false;
134
+ constructor(
135
+ classParam: T,
136
+ loggers: LoggersType,
137
+ socket: Server,
138
+ model: ReturnModelType<T, BeAnObject>,
139
+ classers: Record<string, AutoUpdateManager<any>>,
140
+ emitter: EventEmitter3,
141
+ options?: AUSOption<T, any>
142
+ ) {
143
+ super(classParam, socket, loggers, classers, emitter);
144
+ this.model = model;
145
+ this.options = options;
146
+ }
147
+
148
+ public async loadDB() {
149
+ this.loggers.debug("Loading manager DB " + this.className);
150
+ const docs = await this.model.find({});
151
+ for (const doc of docs) {
152
+ this.classes[(doc as any)._id] =
153
+ this.classes[(doc as any)._id] ??
154
+ (await this.handleGetMissingObject((doc as any)._id.toString()));
155
+ }
156
+ this.loggers.debug(
157
+ "Loaded manager DB " + this.className + " - [" + docs.length + "] entries"
158
+ );
159
+ }
160
+
161
+ public registerSocket(socket: Socket) {
162
+ this.clientSockets.add(socket);
163
+
164
+ if (this.doDebug)
165
+ socket.onAny((event: string, data: any) => {
166
+ this.loggers.debug("Client Event", event, data);
167
+ });
168
+
169
+ socket.on(
170
+ "startup" + this.className,
171
+ async (ack: (ids: string[]) => void) => {
172
+ ack(this.objectIDs);
173
+ }
174
+ );
175
+ socket.on("delete" + this.className, async (id: string) => {
176
+ this.loggers.debug("Deleting object from manager " + this.className, id);
177
+ try {
178
+ this.classes[id].destroy();
179
+ this.classesAsArray.splice(
180
+ this.classesAsArray.indexOf(this.classes[id]),
181
+ 1
182
+ );
183
+ delete this.classes[id];
184
+ } catch (error) {
185
+ this.loggers.error(
186
+ "Error deleting object from manager " + this.className + " - " + id
187
+ );
188
+ this.loggers.error(error);
189
+ }
190
+ });
191
+ socket.on(
192
+ "new" + this.className,
193
+ async (
194
+ data: IsData<InstanceType<T>>,
195
+ ack: (res: ServerResponse<T>) => void
196
+ ) => {
197
+ this.loggers.debug("Creating new object in manager " + this.className);
198
+ try {
199
+ const newDoc = await this.createObject(data);
200
+ ack({
201
+ data: newDoc.extractedData,
202
+ success: true,
203
+ message: "Created successfully",
204
+ });
205
+ } catch (error) {
206
+ this.loggers.error(
207
+ "Error creating new object in manager " + this.className
208
+ );
209
+ this.loggers.error(error);
210
+ ack({ success: false, message: (error as any).message });
211
+ }
212
+ }
213
+ );
214
+
215
+ socket.onAny(
216
+ async (
217
+ event: string,
218
+ data: ServerUpdateRequest<T>,
219
+ ack: (res: ServerResponse<T>) => void
220
+ ) => {
221
+ if (
222
+ event.startsWith("update" + this.className) &&
223
+ event.replace("update" + this.className, "").length === 24
224
+ ) {
225
+ this.loggers.debug(
226
+ "Updating object in manager " +
227
+ this.className +
228
+ ": " +
229
+ event +
230
+ " - " +
231
+ JSON.stringify(data)
232
+ );
233
+ try {
234
+ const id = event.replace("update" + this.className, "");
235
+ let obj = this.classes[id];
236
+ if (typeof obj === "string")
237
+ this.classes[id] = obj = await this.handleGetMissingObject(obj);
238
+ if (typeof obj === "string")
239
+ throw new Error(`Never... failed to get object somehow: ${obj}`);
240
+ const res = await obj.setValue(data.key as any, data.value);
241
+
242
+ res.success
243
+ ? ack({
244
+ data: obj.extractedData,
245
+ success: res.success,
246
+ message: res.msg,
247
+ })
248
+ : ack({ success: res.success, message: res.msg });
249
+ } catch (error) {
250
+ this.loggers.warn(
251
+ "Failed to update object in manager " + this.className
252
+ );
253
+ ack({ success: false, message: (error as any).message });
254
+ }
255
+ } else if (
256
+ event.startsWith("get" + this.className) &&
257
+ event.replace("get" + this.className, "").length === 24
258
+ ) {
259
+ try {
260
+ const id = event.replace("get" + this.className, "");
261
+ let obj = this.classes[id];
262
+ ack({
263
+ data: obj.extractedData,
264
+ success: true,
265
+ message: "Updated successfully",
266
+ });
267
+ } catch (error) {
268
+ ack({ success: false, message: (error as any).message });
269
+ }
270
+ }
271
+ }
272
+ );
273
+ socket.on("disconnect", () => {
274
+ this.clientSockets.delete(socket);
275
+ });
276
+ }
277
+
278
+ protected async handleGetMissingObject(_id: string) {
279
+ const document = await this.model.findById(_id);
280
+ if (!document) throw new Error(`No document with id ${_id} in DB.`);
281
+ if (!this.classers) throw new Error(`No classers.`);
282
+ return await createAutoUpdatedClass(
283
+ this.classParam,
284
+ this.socket,
285
+ document,
286
+ this.loggers,
287
+ this,
288
+ this.emitter
289
+ );
290
+ }
291
+
292
+ public async createObject(data: Omit<IsData<InstanceType<T>>, "_id">) {
293
+ if (!this.classers) throw new Error(`No classers.`);
294
+ (data as any)._id = undefined;
295
+ const entry = await this.model.create(data);
296
+ const object = await createAutoUpdatedClass(
297
+ this.classParam,
298
+ this.socket,
299
+ entry,
300
+ this.loggers,
301
+ this,
302
+ this.emitter
303
+ );
304
+ object.checkAutoStatusChange();
305
+ this.classes[object._id] = object;
306
+ this.classesAsArray.push(object);
307
+ return object;
308
+ }
309
+ }