just-another-http-api 1.2.5 → 1.2.7

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.
@@ -0,0 +1,51 @@
1
+ module.exports = [ {
2
+ languageOptions: {
3
+ ecmaVersion: 2021,
4
+ sourceType: 'module',
5
+ parserOptions: {
6
+ ecmaFeatures: {
7
+ jsx: true
8
+ }
9
+ }
10
+ },
11
+ rules: {
12
+ 'max-len': 'off',
13
+ 'semi': [ 'error', 'always' ],
14
+ 'no-underscore-dangle': 'off',
15
+ 'class-methods-use-this': 'off',
16
+ 'no-param-reassign': 'off',
17
+ 'no-restricted-syntax': 'warn',
18
+ 'no-continue': 'warn',
19
+ 'consistent-return': 'warn',
20
+ 'guard-for-in': 'warn',
21
+ 'import/no-commonjs': 'off',
22
+ 'import/no-dynamic-require': 'off',
23
+ 'new-cap': 'warn',
24
+ 'no-nested-ternary': 'off',
25
+ 'prefer-const': 'error',
26
+ 'newline-before-return': [ 'error', 'always' ],
27
+ 'quotes': [ 'error', 'single' ],
28
+ 'no-multiple-empty-lines': [ 'error', { max: 1 } ],
29
+ 'space-before-function-paren': [ 'error', 'always' ],
30
+ 'func-call-spacing': [ 'error', 'always' ],
31
+ 'block-spacing': [ 'error', 'always' ],
32
+ 'space-in-parens': [ 'error', 'always' ],
33
+ 'object-curly-spacing': [ 'error', 'always' ],
34
+ 'no-spaced-func': 'off',
35
+ 'space-unary-ops': [ 'error', { 'words': true, 'nonwords': false, 'overrides': { 'new': false, '++': false } } ],
36
+ 'arrow-spacing': [ 'error', { 'before': true, 'after': true } ],
37
+ 'indent': [ 'error', 4, { 'SwitchCase': 1 } ],
38
+ 'key-spacing': [ 'error', { 'beforeColon': false, 'afterColon': true } ],
39
+ 'computed-property-spacing': [ 'error', 'always' ],
40
+ 'array-bracket-spacing': [ 'error', 'never' ],
41
+ 'comma-spacing': [ 'error', { 'before': false, 'after': true } ],
42
+ 'brace-style': [ 'error', 'stroustrup' ],
43
+ 'space-infix-ops': 'error',
44
+ 'keyword-spacing': [ 'error', { overrides: {
45
+ if: { after: true },
46
+ for: { after: true },
47
+ while: { after: true }
48
+ } } ],
49
+ 'array-bracket-spacing': [ 'error', 'always' ],
50
+ },
51
+ } ];
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "just-another-http-api",
3
- "version": "1.2.5",
3
+ "version": "1.2.7",
4
4
  "description": "A framework built on top of fastify aimed at removing the need for any network or server configuration. ",
5
5
  "homepage": "https://github.com/OllieEdge/just-another-http-api#readme",
6
6
  "repository": {
@@ -25,22 +25,23 @@
25
25
  "author": "Oliver Edgington <oliver@edgington.com> (https://github.com/OllieEdge)",
26
26
  "license": "MIT",
27
27
  "dependencies": {
28
- "@aws-sdk/lib-storage": "^3.454.0",
28
+ "@aws-sdk/lib-storage": "^3.569.0",
29
29
  "@fastify/caching": "^8.3.0",
30
- "@fastify/cors": "^8.4.1",
30
+ "@fastify/cors": "^9.0.1",
31
31
  "@fastify/jwt": "^7.2.3",
32
- "@fastify/redis": "^6.1.1",
33
- "@fastify/websocket": "^8.2.0",
32
+ "@fastify/redis": "^6.2.0",
33
+ "@fastify/websocket": "^10.0.1",
34
34
  "abstract-cache": "^1.0.1",
35
35
  "abstract-cache-redis": "^2.0.0",
36
- "fastify": "^4.24.3",
36
+ "eip-cloud-services": "^1.1.9",
37
+ "fastify": "^4.27.0",
37
38
  "fastify-multer": "^2.0.3",
38
39
  "recursive-readdir": "^2.2.3"
39
40
  },
40
41
  "devDependencies": {
41
42
  "chai": "^4.3.10",
42
43
  "chai-as-promised": "^7.1.1",
43
- "eip-cloud-services": "^1.1.0",
44
- "mocha": "^10.2.0"
44
+ "eslint": "^9.2.0",
45
+ "mocha": "^10.4.0"
45
46
  }
46
47
  }
@@ -4,11 +4,14 @@ const { redis } = require ( 'eip-cloud-services' );
4
4
  class WebsocketGroup {
5
5
 
6
6
  constructor ( groupName, destroy, messageReceivedHandler, connectionClosedHandler ) {
7
+ this.timeOfCreation = Date.now ();
7
8
  this.groupName = groupName;
8
9
  this.destroy = destroy;
10
+ this.groupUuid = crypto.randomUUID ().substring ( 0, 8 );
9
11
  this.connections = new Map ();
10
12
  this.messageReceivedHandler = messageReceivedHandler || ( async () => {} );
11
13
  this.connectionClosedHandler = connectionClosedHandler;
14
+ this.pingConnections = setInterval ( this.#pingConnections.bind ( this ), 30000 );
12
15
  }
13
16
 
14
17
  async initialize () {
@@ -40,6 +43,19 @@ class WebsocketGroup {
40
43
  }
41
44
  }
42
45
 
46
+ #pingConnections () {
47
+ this.connections.forEach ( ( connection, connectionId ) => {
48
+ if(!connection.isAlive) {
49
+ this.connections.delete ( connectionId );
50
+ this.#clean(connectionId);
51
+ }
52
+ else{
53
+ connection.isAlive = false;
54
+ connection.socket.ping ();
55
+ }
56
+ } );
57
+ }
58
+
43
59
  getConnections () {
44
60
  return this.connections;
45
61
  }
@@ -53,6 +69,7 @@ class WebsocketGroup {
53
69
  }
54
70
 
55
71
  addNewConnection ( connection, connectionId = crypto.randomUUID () ) {
72
+ connection.isAlive = true;
56
73
  this.connections.set ( connectionId, connection );
57
74
 
58
75
  connection.socket.on ( 'message', async message => {
@@ -68,6 +85,10 @@ class WebsocketGroup {
68
85
  this.connections.delete ( connectionId );
69
86
  this.#clean(connectionId);
70
87
  } );
88
+
89
+ connection.socket.on ( 'pong', () => {
90
+ connection.isAlive = true;
91
+ } );
71
92
 
72
93
  connection.socket.on ( 'error', error => {
73
94
  console.error ( 'WebSocket error:', error );
@@ -79,22 +100,24 @@ class WebsocketGroup {
79
100
  }
80
101
 
81
102
  async #clean (connectionId) {
82
- if ( this.connections.size === 0 ) {
83
- this.connections.clear ();
84
- await redis.unsubscribe ( `${this.groupName}_individualMessage`, this.#handleIndividualMessage.bind ( this ) );
85
- await redis.unsubscribe ( `${this.groupName}_broadcast`, this.#broadcastMessageToClients.bind ( this ) );
86
- await redis.unsubscribe ( `${this.groupName}_messageReceived`, this.#handleUserMessage.bind ( this ) );
87
- this.destroy(this.groupName);
88
- }
89
103
 
90
104
  if ( this.connectionClosedHandler !== null && typeof this.connectionClosedHandler === 'function') {
91
105
  try{
92
- await this.connectionClosedHandler ( {groupName:this.groupName, connectionId} );
106
+ await this.connectionClosedHandler ( {groupName:this.groupName, connectionId} );
93
107
  }
94
108
  catch (error) {
95
109
  console.error('Error in connectionClosedHandler:', error);
96
110
  }
97
111
  }
112
+
113
+ if ( this.connections.size === 0 ) {
114
+ clearInterval ( this.pingConnections );
115
+ this.connections.clear ();
116
+ await redis.unsubscribe ( `${this.groupName}_individualMessage` );
117
+ await redis.unsubscribe ( `${this.groupName}_broadcast` );
118
+ await redis.unsubscribe ( `${this.groupName}_messageReceived` );
119
+ this.destroy(this.groupName);
120
+ }
98
121
  }
99
122
  }
100
123
 
@@ -3,12 +3,12 @@ const WebsocketGroup = require ( './WebsocketGroup' );
3
3
  const websocketGroups = {};
4
4
 
5
5
  const getGroupInstance = async ( groupName, messageReceivedHandler, connectionClosedHandler ) => {
6
- if ( !websocketGroups[ groupName ] ) {
6
+ if ( !websocketGroups[ groupName ] && messageReceivedHandler && connectionClosedHandler ) {
7
7
  websocketGroups[ groupName ] = new WebsocketGroup ( groupName, removeGroupInstance, messageReceivedHandler, connectionClosedHandler );
8
8
  await websocketGroups[ groupName ].initialize ();
9
9
  }
10
10
 
11
- return websocketGroups[ groupName ];
11
+ return websocketGroups[ groupName ] || null;
12
12
  };
13
13
 
14
14
  const removeGroupInstance = async groupName => {
package/.eslintrc DELETED
@@ -1,148 +0,0 @@
1
- {
2
- "parserOptions": {
3
- "ecmaVersion": 12,
4
- "sourceType": "module",
5
- "ecmaFeatures": {
6
- "jsx": true
7
- }
8
- },
9
- "env": {
10
- "commonjs": true,
11
- "node": true
12
- },
13
- "globals": {
14
- "__lib": "readonly",
15
- "__routes": "readonly",
16
- "__base": "readonly",
17
- "__utils": "readonly"
18
- },
19
- "rules": {
20
- "max-len": [
21
- "off",
22
- 120
23
- ],
24
- "semi": [
25
- "error",
26
- "always"
27
- ],
28
- "no-underscore-dangle": 0,
29
- "class-methods-use-this": 0,
30
- "no-param-reassign": 0,
31
- "no-restricted-syntax": 1,
32
- "no-continue": 1,
33
- "consistent-return": 1,
34
- "guard-for-in": 1,
35
- "import/no-commonjs": 0,
36
- "import/no-dynamic-require": 0,
37
- "new-cap": 1,
38
- "no-nested-ternary": 0,
39
- "prefer-const": "error",
40
- "newline-before-return": "error",
41
- "quotes": [
42
- "error",
43
- "single"
44
- ],
45
- "no-multiple-empty-lines": [
46
- "error",
47
- {
48
- "max": 1
49
- }
50
- ],
51
- "space-before-function-paren": [
52
- "error",
53
- "always"
54
- ],
55
- "func-call-spacing": [
56
- "error",
57
- "always"
58
- ],
59
- "block-spacing": [
60
- "error",
61
- "always"
62
- ],
63
- "space-in-parens": [
64
- "error",
65
- "always"
66
- ],
67
- "object-curly-spacing": [
68
- "error",
69
- "always"
70
- ],
71
- "no-spaced-func": 0,
72
- "space-unary-ops": [
73
- "error",
74
- {
75
- "words": true,
76
- "nonwords": false,
77
- "overrides": {
78
- "new": false,
79
- "++": false
80
- }
81
- }
82
- ],
83
- "arrow-spacing": [
84
- "error",
85
- {
86
- "before": true,
87
- "after": true
88
- }
89
- ],
90
- "indent": [
91
- "error",
92
- 4,
93
- {
94
- "SwitchCase": 1
95
- }
96
- ],
97
- "key-spacing": [
98
- "error",
99
- {
100
- "beforeColon": false,
101
- "afterColon": true
102
- }
103
- ],
104
- "computed-property-spacing": [
105
- "error",
106
- "always"
107
- ],
108
- "array-bracket-spacing": [
109
- "error",
110
- "always"
111
- ],
112
- "comma-spacing": [
113
- "error",
114
- {
115
- "before": false,
116
- "after": true
117
- }
118
- ],
119
- "brace-style": [
120
- "error",
121
- "stroustrup"
122
- ],
123
- "space-infix-ops": [
124
- "error",
125
- {
126
- "int32Hint": true
127
- }
128
- ],
129
- "keyword-spacing": [
130
- "error",
131
- {
132
- "overrides": {
133
- "if": {
134
- "after": true
135
- },
136
- "for": {
137
- "after": true
138
- },
139
- "while": {
140
- "after": true
141
- }
142
- }
143
- }
144
- ],
145
- "no-octal-escape": 0,
146
- "no-octal": 0
147
- }
148
- }