@designofadecade/server 4.0.0

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 (84) hide show
  1. package/CHANGELOG.md +62 -0
  2. package/LICENSE +21 -0
  3. package/README.md +297 -0
  4. package/dist/client/ApiClient.d.ts +121 -0
  5. package/dist/client/ApiClient.d.ts.map +1 -0
  6. package/dist/client/ApiClient.js +289 -0
  7. package/dist/client/ApiClient.js.map +1 -0
  8. package/dist/context/Context.d.ts +71 -0
  9. package/dist/context/Context.d.ts.map +1 -0
  10. package/dist/context/Context.js +81 -0
  11. package/dist/context/Context.js.map +1 -0
  12. package/dist/docs/OpenApiGenerator.d.ts +135 -0
  13. package/dist/docs/OpenApiGenerator.d.ts.map +1 -0
  14. package/dist/docs/OpenApiGenerator.js +165 -0
  15. package/dist/docs/OpenApiGenerator.js.map +1 -0
  16. package/dist/events/Events.d.ts +52 -0
  17. package/dist/events/Events.d.ts.map +1 -0
  18. package/dist/events/Events.js +70 -0
  19. package/dist/events/Events.js.map +1 -0
  20. package/dist/events/EventsManager.d.ts +46 -0
  21. package/dist/events/EventsManager.d.ts.map +1 -0
  22. package/dist/events/EventsManager.js +137 -0
  23. package/dist/events/EventsManager.js.map +1 -0
  24. package/dist/index.d.ts +32 -0
  25. package/dist/index.d.ts.map +1 -0
  26. package/dist/index.js +38 -0
  27. package/dist/index.js.map +1 -0
  28. package/dist/local/Local.d.ts +83 -0
  29. package/dist/local/Local.d.ts.map +1 -0
  30. package/dist/local/Local.js +114 -0
  31. package/dist/local/Local.js.map +1 -0
  32. package/dist/logger/Logger.d.ts +365 -0
  33. package/dist/logger/Logger.d.ts.map +1 -0
  34. package/dist/logger/Logger.js +582 -0
  35. package/dist/logger/Logger.js.map +1 -0
  36. package/dist/middleware/RequestLogger.d.ts +62 -0
  37. package/dist/middleware/RequestLogger.d.ts.map +1 -0
  38. package/dist/middleware/RequestLogger.js +71 -0
  39. package/dist/middleware/RequestLogger.js.map +1 -0
  40. package/dist/notifications/Slack.d.ts +19 -0
  41. package/dist/notifications/Slack.d.ts.map +1 -0
  42. package/dist/notifications/Slack.js +55 -0
  43. package/dist/notifications/Slack.js.map +1 -0
  44. package/dist/router/RouteError.d.ts +21 -0
  45. package/dist/router/RouteError.d.ts.map +1 -0
  46. package/dist/router/RouteError.js +31 -0
  47. package/dist/router/RouteError.js.map +1 -0
  48. package/dist/router/Router.d.ts +66 -0
  49. package/dist/router/Router.d.ts.map +1 -0
  50. package/dist/router/Router.js +327 -0
  51. package/dist/router/Router.js.map +1 -0
  52. package/dist/router/Routes.d.ts +30 -0
  53. package/dist/router/Routes.d.ts.map +1 -0
  54. package/dist/router/Routes.js +52 -0
  55. package/dist/router/Routes.js.map +1 -0
  56. package/dist/router/StaticFileHandler.d.ts +44 -0
  57. package/dist/router/StaticFileHandler.d.ts.map +1 -0
  58. package/dist/router/StaticFileHandler.js +148 -0
  59. package/dist/router/StaticFileHandler.js.map +1 -0
  60. package/dist/sanitizer/HtmlSanitizer.d.ts +306 -0
  61. package/dist/sanitizer/HtmlSanitizer.d.ts.map +1 -0
  62. package/dist/sanitizer/HtmlSanitizer.js +808 -0
  63. package/dist/sanitizer/HtmlSanitizer.js.map +1 -0
  64. package/dist/server/Server.d.ts +28 -0
  65. package/dist/server/Server.d.ts.map +1 -0
  66. package/dist/server/Server.js +95 -0
  67. package/dist/server/Server.js.map +1 -0
  68. package/dist/state/AppState.d.ts +64 -0
  69. package/dist/state/AppState.d.ts.map +1 -0
  70. package/dist/state/AppState.js +89 -0
  71. package/dist/state/AppState.js.map +1 -0
  72. package/dist/utils/HtmlRenderer.d.ts +6 -0
  73. package/dist/utils/HtmlRenderer.d.ts.map +1 -0
  74. package/dist/utils/HtmlRenderer.js +128 -0
  75. package/dist/utils/HtmlRenderer.js.map +1 -0
  76. package/dist/websocket/WebSocketMessageFormatter.d.ts +40 -0
  77. package/dist/websocket/WebSocketMessageFormatter.d.ts.map +1 -0
  78. package/dist/websocket/WebSocketMessageFormatter.js +99 -0
  79. package/dist/websocket/WebSocketMessageFormatter.js.map +1 -0
  80. package/dist/websocket/WebSocketServer.d.ts +14 -0
  81. package/dist/websocket/WebSocketServer.d.ts.map +1 -0
  82. package/dist/websocket/WebSocketServer.js +138 -0
  83. package/dist/websocket/WebSocketServer.js.map +1 -0
  84. package/package.json +97 -0
@@ -0,0 +1,138 @@
1
+ import { EventEmitter } from 'events';
2
+ import { WebSocketServer as WebSocketServerLibrary, WebSocket } from 'ws';
3
+ import { logger } from '../logger/Logger.js';
4
+ import WebSocketMessageFormatter from './WebSocketMessageFormatter.js';
5
+ export default class WebSocketServer extends EventEmitter {
6
+ #wss;
7
+ constructor({ port = 8080, host = '0.0.0.0' } = {}) {
8
+ super();
9
+ // Validate port
10
+ if (isNaN(port) || port < 1 || port > 65535) {
11
+ throw new Error(`Port ${port} is invalid. Must be between 1 and 65535.`);
12
+ }
13
+ this.#init(port, host);
14
+ }
15
+ get clientCount() {
16
+ return this.#wss.clients.size;
17
+ }
18
+ #init(port, host) {
19
+ this.#wss = new WebSocketServerLibrary({ port, host });
20
+ // Error handler for the WebSocket server itself
21
+ this.#wss.on('error', (error) => {
22
+ if (error.code === 'EADDRINUSE') {
23
+ logger.error('WebSocket port is already in use', {
24
+ code: 'WEBSOCKET_PORT_IN_USE',
25
+ source: 'WebSocketServer.init',
26
+ port,
27
+ error,
28
+ });
29
+ process.exit(1);
30
+ }
31
+ logger.error('WebSocket Server error', {
32
+ code: 'WEBSOCKET_SERVER_ERROR',
33
+ source: 'WebSocketServer.init',
34
+ error,
35
+ errorCode: error.code,
36
+ });
37
+ process.exit(1);
38
+ });
39
+ this.#wss.on('listening', () => {
40
+ logger.info('WebSocket Server listening', {
41
+ source: 'WebSocketServer.init',
42
+ host,
43
+ port,
44
+ });
45
+ });
46
+ this.#wss.on('connection', (ws) => {
47
+ logger.info('WebSocket client connected', {
48
+ source: 'WebSocketServer.connection',
49
+ clientCount: this.clientCount,
50
+ });
51
+ ws.send(WebSocketMessageFormatter.format('ws:connected', {
52
+ message: 'WebSocket connection established',
53
+ }));
54
+ ws.on('message', async (message) => {
55
+ const messageString = message.toString();
56
+ if (messageString.includes('ws:ping')) {
57
+ ws.send(WebSocketMessageFormatter.format('ws:pong', {}));
58
+ return;
59
+ }
60
+ try {
61
+ const parsed = WebSocketMessageFormatter.parse(messageString);
62
+ if (!parsed) {
63
+ ws.send(WebSocketMessageFormatter.format('ws:error', {
64
+ error: 'Invalid message format',
65
+ }));
66
+ return;
67
+ }
68
+ this.emit('message', parsed);
69
+ }
70
+ catch (error) {
71
+ logger.error('Message handling error', {
72
+ code: 'WEBSOCKET_MESSAGE_ERROR',
73
+ source: 'WebSocketServer.onMessage',
74
+ error,
75
+ });
76
+ ws.send(WebSocketMessageFormatter.format('ws:error', {
77
+ error: error.message,
78
+ }));
79
+ }
80
+ });
81
+ ws.on('close', () => {
82
+ logger.info('WebSocket client disconnected', {
83
+ source: 'WebSocketServer.onClose',
84
+ clientCount: this.clientCount,
85
+ });
86
+ });
87
+ ws.on('error', (error) => {
88
+ logger.error('WebSocket error', {
89
+ code: 'WEBSOCKET_CLIENT_ERROR',
90
+ source: 'WebSocketServer.onError',
91
+ error,
92
+ });
93
+ });
94
+ });
95
+ }
96
+ broadcast(message) {
97
+ this.#wss.clients.forEach((client) => {
98
+ if (client.readyState === WebSocket.OPEN) {
99
+ try {
100
+ client.send(message);
101
+ }
102
+ catch (error) {
103
+ logger.error('Broadcast send error', {
104
+ code: 'WEBSOCKET_BROADCAST_ERROR',
105
+ source: 'WebSocketServer.broadcast',
106
+ error,
107
+ });
108
+ }
109
+ }
110
+ });
111
+ }
112
+ close() {
113
+ return new Promise((resolve, reject) => {
114
+ if (this.#wss) {
115
+ this.#wss.close((error) => {
116
+ if (error) {
117
+ logger.error('Error closing WebSocket Server', {
118
+ code: 'WEBSOCKET_CLOSE_ERROR',
119
+ source: 'WebSocketServer.close',
120
+ error,
121
+ });
122
+ reject(error);
123
+ }
124
+ else {
125
+ logger.info('WebSocket Server closed', {
126
+ source: 'WebSocketServer.close',
127
+ });
128
+ resolve();
129
+ }
130
+ });
131
+ }
132
+ else {
133
+ resolve();
134
+ }
135
+ });
136
+ }
137
+ }
138
+ //# sourceMappingURL=WebSocketServer.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"WebSocketServer.js","sourceRoot":"","sources":["../../src/websocket/WebSocketServer.ts"],"names":[],"mappings":"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"}
package/package.json ADDED
@@ -0,0 +1,97 @@
1
+ {
2
+ "name": "@designofadecade/server",
3
+ "version": "4.0.0",
4
+ "description": "Design of a Decade Server - A modern TypeScript server with WebSocket support",
5
+ "type": "module",
6
+ "main": "./dist/index.js",
7
+ "types": "./dist/index.d.ts",
8
+ "exports": {
9
+ ".": {
10
+ "types": "./dist/index.d.ts",
11
+ "import": "./dist/index.js"
12
+ }
13
+ },
14
+ "scripts": {
15
+ "test": "vitest run",
16
+ "test:watch": "vitest",
17
+ "test:coverage": "vitest run --coverage",
18
+ "test:ui": "vitest --ui",
19
+ "bench": "vitest bench",
20
+ "bench:watch": "vitest bench --watch",
21
+ "build": "tsc",
22
+ "build:watch": "tsc --watch",
23
+ "clean": "rm -rf dist",
24
+ "prepare": "husky",
25
+ "prepublishOnly": "npm run build",
26
+ "lint": "eslint src/**/*.ts",
27
+ "format": "prettier --write \"src/**/*.ts\"",
28
+ "typecheck": "tsc --noEmit"
29
+ },
30
+ "engines": {
31
+ "node": ">=24.0.0"
32
+ },
33
+ "keywords": [
34
+ "server",
35
+ "websocket",
36
+ "typescript",
37
+ "node",
38
+ "api",
39
+ "http-server",
40
+ "router",
41
+ "middleware",
42
+ "framework",
43
+ "esm",
44
+ "websocket-server",
45
+ "routing",
46
+ "static-files",
47
+ "html-sanitizer",
48
+ "event-emitter",
49
+ "state-management",
50
+ "openapi",
51
+ "swagger",
52
+ "rest-api"
53
+ ],
54
+ "author": "Design of a Decade <info@designofadecade.com>",
55
+ "license": "MIT",
56
+ "homepage": "https://github.com/designofadecade/server#readme",
57
+ "bugs": {
58
+ "url": "https://github.com/designofadecade/server/issues"
59
+ },
60
+ "dependencies": {
61
+ "ws": "^8.16.0"
62
+ },
63
+ "devDependencies": {
64
+ "@types/node": "^20.11.0",
65
+ "@types/ws": "^8.5.10",
66
+ "@typescript-eslint/eslint-plugin": "^7.0.0",
67
+ "@typescript-eslint/parser": "^7.0.0",
68
+ "@vitest/coverage-v8": "^2.0.0",
69
+ "@vitest/ui": "^2.0.0",
70
+ "eslint": "^8.57.0",
71
+ "husky": "^9.1.7",
72
+ "lint-staged": "^16.2.7",
73
+ "prettier": "^3.2.5",
74
+ "typescript": "^5.3.0",
75
+ "vitest": "^2.0.0"
76
+ },
77
+ "repository": {
78
+ "type": "git",
79
+ "url": "git+https://github.com/designofadecade/server.git"
80
+ },
81
+ "publishConfig": {
82
+ "access": "public"
83
+ },
84
+ "files": [
85
+ "dist",
86
+ "README.md",
87
+ "LICENSE",
88
+ "CHANGELOG.md"
89
+ ],
90
+ "lint-staged": {
91
+ "*.ts": [
92
+ "prettier --write",
93
+ "eslint --fix",
94
+ "vitest related --run"
95
+ ]
96
+ }
97
+ }