underpost 3.1.3 → 3.2.2
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.
- package/.env.example +0 -2
- package/.github/workflows/ghpkg.ci.yml +4 -4
- package/.github/workflows/npmpkg.ci.yml +28 -11
- package/.github/workflows/publish.ci.yml +6 -0
- package/.github/workflows/pwa-microservices-template-page.cd.yml +4 -5
- package/.github/workflows/pwa-microservices-template-test.ci.yml +3 -3
- package/.github/workflows/release.cd.yml +13 -8
- package/CHANGELOG.md +396 -1
- package/CLI-HELP.md +53 -6
- package/Dockerfile +4 -2
- package/README.md +3 -2
- package/bin/build.js +18 -12
- package/bin/deploy.js +177 -124
- package/bin/file.js +3 -0
- package/conf.js +3 -2
- package/manifests/cronjobs/dd-cron/dd-cron-backup.yaml +5 -2
- package/manifests/cronjobs/dd-cron/dd-cron-dns.yaml +5 -2
- package/manifests/deployment/dd-default-development/deployment.yaml +2 -2
- package/manifests/deployment/dd-test-development/deployment.yaml +88 -74
- package/manifests/deployment/dd-test-development/proxy.yaml +13 -4
- package/manifests/deployment/playwright/deployment.yaml +1 -1
- package/nodemon.json +1 -1
- package/package.json +22 -15
- package/scripts/rhel-grpc-setup.sh +56 -0
- package/src/api/file/file.ref.json +18 -0
- package/src/api/user/user.service.js +8 -7
- package/src/cli/cluster.js +7 -7
- package/src/cli/db.js +726 -825
- package/src/cli/deploy.js +151 -93
- package/src/cli/env.js +19 -0
- package/src/cli/fs.js +5 -2
- package/src/cli/index.js +45 -2
- package/src/cli/kubectl.js +211 -0
- package/src/cli/release.js +284 -0
- package/src/cli/repository.js +434 -75
- package/src/cli/run.js +189 -34
- package/src/cli/secrets.js +73 -0
- package/src/cli/test.js +3 -3
- package/src/client/Default.index.js +3 -4
- package/src/client/components/core/AppStore.js +69 -0
- package/src/client/components/core/CalendarCore.js +2 -2
- package/src/client/components/core/DropDown.js +137 -17
- package/src/client/components/core/Keyboard.js +2 -2
- package/src/client/components/core/LogIn.js +2 -2
- package/src/client/components/core/LogOut.js +2 -2
- package/src/client/components/core/Modal.js +0 -1
- package/src/client/components/core/Panel.js +0 -1
- package/src/client/components/core/PanelForm.js +19 -19
- package/src/client/components/core/SocketIo.js +82 -29
- package/src/client/components/core/SocketIoHandler.js +75 -0
- package/src/client/components/core/Stream.js +143 -95
- package/src/client/components/core/Webhook.js +40 -7
- package/src/client/components/default/AppStoreDefault.js +5 -0
- package/src/client/components/default/LogInDefault.js +3 -3
- package/src/client/components/default/LogOutDefault.js +2 -2
- package/src/client/components/default/MenuDefault.js +5 -5
- package/src/client/components/default/SocketIoDefault.js +3 -51
- package/src/client/services/core/core.service.js +20 -8
- package/src/client/services/user/user.management.js +2 -2
- package/src/index.js +24 -1
- package/src/runtime/express/Dockerfile +4 -0
- package/src/runtime/express/Express.js +18 -1
- package/src/runtime/lampp/Dockerfile +13 -2
- package/src/runtime/lampp/Lampp.js +27 -4
- package/src/runtime/wp/Dockerfile +68 -0
- package/src/runtime/wp/Wp.js +639 -0
- package/src/server/auth.js +24 -1
- package/src/server/backup.js +57 -23
- package/src/server/client-build-docs.js +9 -2
- package/src/server/client-build.js +31 -31
- package/src/server/client-formatted.js +109 -57
- package/src/server/cron.js +23 -18
- package/src/server/ipfs-client.js +24 -1
- package/src/server/peer.js +8 -0
- package/src/server/runtime.js +25 -1
- package/src/server/start.js +3 -2
- package/src/ws/IoInterface.js +1 -10
- package/src/ws/IoServer.js +14 -33
- package/src/ws/core/channels/core.ws.chat.js +65 -20
- package/src/ws/core/channels/core.ws.mailer.js +113 -32
- package/src/ws/core/channels/core.ws.stream.js +90 -31
- package/src/ws/core/core.ws.connection.js +12 -33
- package/src/ws/core/core.ws.emit.js +10 -26
- package/src/ws/core/core.ws.server.js +25 -58
- package/src/ws/default/channels/default.ws.main.js +53 -12
- package/src/ws/default/default.ws.connection.js +26 -13
- package/src/ws/default/default.ws.server.js +30 -12
- package/src/client/components/default/ElementsDefault.js +0 -38
- package/src/ws/core/management/core.ws.chat.js +0 -8
- package/src/ws/core/management/core.ws.mailer.js +0 -16
- package/src/ws/core/management/core.ws.stream.js +0 -8
- package/src/ws/default/management/default.ws.main.js +0 -8
|
@@ -1,29 +1,24 @@
|
|
|
1
1
|
/**
|
|
2
|
-
*
|
|
3
|
-
* @module ws/core.ws.emit
|
|
4
|
-
* @namespace CoreWsEmitter
|
|
2
|
+
* Standardized WebSocket message emission.
|
|
3
|
+
* @module ws/core/core.ws.emit
|
|
5
4
|
*/
|
|
6
5
|
|
|
7
6
|
import { loggerFactory } from '../../server/logger.js';
|
|
8
|
-
import { Socket } from 'socket.io';
|
|
9
7
|
|
|
10
8
|
const logger = loggerFactory(import.meta);
|
|
11
9
|
|
|
12
10
|
/**
|
|
13
11
|
* @class CoreWsEmitter
|
|
14
|
-
* @
|
|
15
|
-
* @memberof CoreWsEmitter
|
|
16
|
-
* @classdesc Provides a static utility method for safely emitting messages over a WebSocket connection.
|
|
12
|
+
* @classdesc Provides a static utility for safely emitting JSON-serialized messages over Socket.IO.
|
|
17
13
|
*/
|
|
18
14
|
class CoreWsEmitter {
|
|
19
15
|
/**
|
|
20
|
-
* Emits a payload to a
|
|
21
|
-
* The payload is automatically JSON stringified.
|
|
16
|
+
* Emits a JSON-stringified payload to a client on a given channel.
|
|
22
17
|
*
|
|
23
18
|
* @static
|
|
24
|
-
* @param {string}
|
|
25
|
-
* @param {Socket
|
|
26
|
-
* @param {Object}
|
|
19
|
+
* @param {string} channel - The channel/event name to emit on.
|
|
20
|
+
* @param {import('socket.io').Socket} client - The Socket.IO socket to emit to.
|
|
21
|
+
* @param {Object} payload - The data object to send.
|
|
27
22
|
* @returns {void}
|
|
28
23
|
*/
|
|
29
24
|
static emit(channel = '', client = {}, payload = {}) {
|
|
@@ -31,23 +26,12 @@ class CoreWsEmitter {
|
|
|
31
26
|
if (client && typeof client.emit === 'function') {
|
|
32
27
|
client.emit(channel, JSON.stringify(payload));
|
|
33
28
|
} else {
|
|
34
|
-
logger.error('Invalid client: Cannot emit message.', { channel,
|
|
29
|
+
logger.error('Invalid client: Cannot emit message.', { channel, payload });
|
|
35
30
|
}
|
|
36
31
|
} catch (error) {
|
|
37
|
-
logger.error(error, { channel,
|
|
32
|
+
logger.error(error, { channel, payload, stack: error.stack });
|
|
38
33
|
}
|
|
39
34
|
}
|
|
40
35
|
}
|
|
41
36
|
|
|
42
|
-
|
|
43
|
-
* Backward compatibility export for the `emit` function.
|
|
44
|
-
* @memberof CoreWsEmitter
|
|
45
|
-
* @function CoreWsEmit
|
|
46
|
-
* @param {string} [channel=''] - The name of the channel/event to emit on.
|
|
47
|
-
* @param {Socket | Object} [client={}] - The Socket.IO client/socket object.
|
|
48
|
-
* @param {Object} [payload={}] - The data object to send.
|
|
49
|
-
* @returns {void}
|
|
50
|
-
*/
|
|
51
|
-
const CoreWsEmit = CoreWsEmitter.emit;
|
|
52
|
-
|
|
53
|
-
export { CoreWsEmitter, CoreWsEmit };
|
|
37
|
+
export { CoreWsEmitter };
|
|
@@ -1,76 +1,43 @@
|
|
|
1
1
|
/**
|
|
2
|
-
*
|
|
3
|
-
* @module ws/core.ws.server
|
|
4
|
-
* @namespace CoreWsServer
|
|
2
|
+
* Core WebSocket server factory — initializes channels and creates the Socket.IO server.
|
|
3
|
+
* @module ws/core/core.ws.server
|
|
5
4
|
*/
|
|
6
5
|
|
|
7
6
|
'use strict';
|
|
8
7
|
|
|
9
|
-
import {
|
|
10
|
-
import {
|
|
11
|
-
import {
|
|
12
|
-
import {
|
|
13
|
-
import {
|
|
14
|
-
import http from 'http'; // Added for JSDoc type hinting
|
|
15
|
-
|
|
16
|
-
// https://socket.io/docs/v3/
|
|
8
|
+
import { IoServer } from '../IoServer.js';
|
|
9
|
+
import { CoreWsConnectionHandler } from './core.ws.connection.js';
|
|
10
|
+
import { CoreWsChatChannel } from './channels/core.ws.chat.js';
|
|
11
|
+
import { CoreWsMailerChannel } from './channels/core.ws.mailer.js';
|
|
12
|
+
import { CoreWsStreamChannel } from './channels/core.ws.stream.js';
|
|
17
13
|
|
|
18
14
|
/**
|
|
19
|
-
* @class
|
|
20
|
-
* @
|
|
21
|
-
*
|
|
22
|
-
* @classdesc Manages the creation and initialization of the main WebSocket server,
|
|
23
|
-
* including setting up the management instances for all channels.
|
|
15
|
+
* @class CoreWsServer
|
|
16
|
+
* @classdesc Creates the core WebSocket server, initializing all channel state
|
|
17
|
+
* before attaching the connection handler.
|
|
24
18
|
*/
|
|
25
|
-
class
|
|
19
|
+
class CoreWsServer {
|
|
26
20
|
/**
|
|
27
|
-
* Initializes channel
|
|
28
|
-
*
|
|
29
|
-
* @
|
|
30
|
-
* @
|
|
31
|
-
* @param {
|
|
32
|
-
* @
|
|
33
|
-
* @param {string} options.host - The host address.
|
|
34
|
-
* @param {string} options.path - The base path for the API.
|
|
35
|
-
* @returns {Promise<Object>} The result object from IoServer creation.
|
|
21
|
+
* Initializes channel state and creates the Socket.IO server.
|
|
22
|
+
* @param {import('http').Server} httpServer
|
|
23
|
+
* @param {Object} options
|
|
24
|
+
* @param {string} options.host
|
|
25
|
+
* @param {string} options.path
|
|
26
|
+
* @returns {{ options: import('socket.io').ServerOptions, ioServer: import('socket.io').Server, meta: ImportMeta }}
|
|
36
27
|
*/
|
|
37
|
-
static
|
|
28
|
+
static create(httpServer, options) {
|
|
38
29
|
const { host, path } = options;
|
|
39
|
-
if (!host || !path) {
|
|
40
|
-
throw new Error('Host and path must be provided in server options.');
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
// Create a unique identifier for this server instance's management context
|
|
44
30
|
const wsManagementId = `${host}${path}`;
|
|
45
31
|
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
CoreWsStreamManagement.instance(wsManagementId);
|
|
32
|
+
CoreWsChatChannel.init(wsManagementId);
|
|
33
|
+
CoreWsMailerChannel.init(wsManagementId);
|
|
34
|
+
CoreWsStreamChannel.init(wsManagementId);
|
|
50
35
|
|
|
51
|
-
|
|
52
|
-
return IoServerClass.create(httpServer, options, (socket) => CoreWsConnection(socket, wsManagementId));
|
|
36
|
+
return IoServer.create(httpServer, options, (socket) => CoreWsConnectionHandler.handle(socket, wsManagementId));
|
|
53
37
|
}
|
|
54
38
|
}
|
|
55
39
|
|
|
56
|
-
/**
|
|
57
|
-
|
|
58
|
-
* @memberof CoreWsServer
|
|
59
|
-
* @function createIoServer
|
|
60
|
-
* @param {http.Server} httpServer - The HTTP server instance.
|
|
61
|
-
* @param {Object} options - Configuration options.
|
|
62
|
-
* @returns {Promise<Object>} The server creation result.
|
|
63
|
-
*/
|
|
64
|
-
const createIoServer = CoreWsServerClass.create;
|
|
65
|
-
|
|
66
|
-
/**
|
|
67
|
-
* Backward compatibility alias.
|
|
68
|
-
* @memberof CoreWsServer
|
|
69
|
-
* @function CoreWsServer
|
|
70
|
-
* @param {import('http').Server} httpServer - The HTTP server instance.
|
|
71
|
-
* @param {Object} options - Configuration options.
|
|
72
|
-
* @returns {Promise<Object>} The server creation result.
|
|
73
|
-
*/
|
|
74
|
-
const CoreWsServer = createIoServer;
|
|
40
|
+
/** Required by Express.js dynamic import: `const { createIoServer } = await import(...)` */
|
|
41
|
+
const createIoServer = CoreWsServer.create.bind(CoreWsServer);
|
|
75
42
|
|
|
76
|
-
export {
|
|
43
|
+
export { CoreWsServer, createIoServer };
|
|
@@ -1,16 +1,57 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
1
|
+
/**
|
|
2
|
+
* Default "main" WebSocket channel — minimal no-op channel.
|
|
3
|
+
* @module ws/default/channels/default.ws.main
|
|
4
|
+
*/
|
|
3
5
|
|
|
4
|
-
|
|
5
|
-
const logger = loggerFactory(import.meta);
|
|
6
|
+
import { IoChannel } from '../../IoInterface.js';
|
|
6
7
|
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
}
|
|
8
|
+
/**
|
|
9
|
+
* @class DefaultWsMainChannel
|
|
10
|
+
* @classdesc Provides a no-op main channel for the default WebSocket server.
|
|
11
|
+
*/
|
|
12
|
+
class DefaultWsMainChannel {
|
|
13
|
+
/** @type {Object.<string, Object>} Per-instance state keyed by wsManagementId. */
|
|
14
|
+
static #state = {};
|
|
13
15
|
|
|
14
|
-
|
|
16
|
+
/** @type {IoChannel} */
|
|
17
|
+
static #io = new IoChannel({ channel: 'main' });
|
|
15
18
|
|
|
16
|
-
|
|
19
|
+
/** @returns {Object.<string, import('socket.io').Socket>} Connected sockets map. */
|
|
20
|
+
static get client() {
|
|
21
|
+
return this.#io.client;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
/** @returns {string} Channel name. */
|
|
25
|
+
static get channel() {
|
|
26
|
+
return this.#io.channel;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Initializes state for a server instance.
|
|
31
|
+
* @param {string} wsManagementId
|
|
32
|
+
*/
|
|
33
|
+
static init(wsManagementId) {
|
|
34
|
+
this.#state[wsManagementId] = {};
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Registers a socket connection.
|
|
39
|
+
* @param {import('socket.io').Socket} socket
|
|
40
|
+
* @param {string} wsManagementId
|
|
41
|
+
*/
|
|
42
|
+
static connection(socket, wsManagementId) {
|
|
43
|
+
return this.#io.connection(socket, wsManagementId);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Handles socket disconnection.
|
|
48
|
+
* @param {import('socket.io').Socket} socket
|
|
49
|
+
* @param {string} reason
|
|
50
|
+
* @param {string} wsManagementId
|
|
51
|
+
*/
|
|
52
|
+
static disconnect(socket, reason, wsManagementId) {
|
|
53
|
+
return this.#io.disconnect(socket, reason, wsManagementId);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
export { DefaultWsMainChannel };
|
|
@@ -1,22 +1,35 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Default WebSocket connection handler — wires sockets to the main channel.
|
|
3
|
+
* @module ws/default/default.ws.connection
|
|
4
|
+
*/
|
|
5
|
+
|
|
1
6
|
import { loggerFactory } from '../../server/logger.js';
|
|
2
7
|
import { DefaultWsMainChannel } from './channels/default.ws.main.js';
|
|
3
8
|
|
|
4
9
|
const logger = loggerFactory(import.meta);
|
|
5
10
|
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
11
|
+
/**
|
|
12
|
+
* @class DefaultWsConnectionHandler
|
|
13
|
+
* @classdesc Subscribes a new socket to the default main channel
|
|
14
|
+
* and delegates disconnect events.
|
|
15
|
+
*/
|
|
16
|
+
class DefaultWsConnectionHandler {
|
|
17
|
+
/**
|
|
18
|
+
* Handles a new WebSocket connection.
|
|
19
|
+
* @param {import('socket.io').Socket} socket
|
|
20
|
+
* @param {string} wsManagementId
|
|
21
|
+
*/
|
|
22
|
+
static handle(socket, wsManagementId) {
|
|
23
|
+
logger.info(`DefaultWsConnection ${socket.id}`);
|
|
12
24
|
|
|
13
|
-
|
|
25
|
+
DefaultWsMainChannel.connection(socket, wsManagementId);
|
|
14
26
|
|
|
15
|
-
|
|
16
|
-
|
|
27
|
+
socket.on('disconnect', (reason) => {
|
|
28
|
+
logger.info(`DefaultWsConnection ${socket.id} due to reason: ${reason}`);
|
|
17
29
|
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
}
|
|
30
|
+
DefaultWsMainChannel.disconnect(socket, reason, wsManagementId);
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
}
|
|
21
34
|
|
|
22
|
-
export {
|
|
35
|
+
export { DefaultWsConnectionHandler };
|
|
@@ -1,20 +1,38 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Default WebSocket server factory — initializes the main channel and creates the Socket.IO server.
|
|
3
|
+
* @module ws/default/default.ws.server
|
|
4
|
+
*/
|
|
5
|
+
|
|
1
6
|
'use strict';
|
|
2
7
|
|
|
3
8
|
import { IoServer } from '../IoServer.js';
|
|
4
|
-
import {
|
|
5
|
-
import {
|
|
6
|
-
|
|
7
|
-
// https://socket.io/docs/v3/
|
|
9
|
+
import { DefaultWsConnectionHandler } from './default.ws.connection.js';
|
|
10
|
+
import { DefaultWsMainChannel } from './channels/default.ws.main.js';
|
|
8
11
|
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
+
/**
|
|
13
|
+
* @class DefaultWsServer
|
|
14
|
+
* @classdesc Creates the default WebSocket server with a single main channel.
|
|
15
|
+
*/
|
|
16
|
+
class DefaultWsServer {
|
|
17
|
+
/**
|
|
18
|
+
* Initializes the main channel and creates the Socket.IO server.
|
|
19
|
+
* @param {import('http').Server} httpServer
|
|
20
|
+
* @param {Object} options
|
|
21
|
+
* @param {string} options.host
|
|
22
|
+
* @param {string} options.path
|
|
23
|
+
* @returns {{ options: import('socket.io').ServerOptions, ioServer: import('socket.io').Server, meta: ImportMeta }}
|
|
24
|
+
*/
|
|
25
|
+
static create(httpServer, options) {
|
|
26
|
+
const { host, path } = options;
|
|
27
|
+
const wsManagementId = `${host}${path}`;
|
|
12
28
|
|
|
13
|
-
|
|
29
|
+
DefaultWsMainChannel.init(wsManagementId);
|
|
14
30
|
|
|
15
|
-
|
|
16
|
-
}
|
|
31
|
+
return IoServer.create(httpServer, options, (socket) => DefaultWsConnectionHandler.handle(socket, wsManagementId));
|
|
32
|
+
}
|
|
33
|
+
}
|
|
17
34
|
|
|
18
|
-
const
|
|
35
|
+
/** Required by Express.js dynamic import: `const { createIoServer } = await import(...)` */
|
|
36
|
+
const createIoServer = DefaultWsServer.create.bind(DefaultWsServer);
|
|
19
37
|
|
|
20
|
-
export {
|
|
38
|
+
export { DefaultWsServer, createIoServer };
|
|
@@ -1,38 +0,0 @@
|
|
|
1
|
-
import { loggerFactory } from '../core/Logger.js';
|
|
2
|
-
import { BaseElement } from './CommonDefault.js';
|
|
3
|
-
|
|
4
|
-
const logger = loggerFactory(import.meta);
|
|
5
|
-
|
|
6
|
-
const ElementsDefault = {
|
|
7
|
-
Data: BaseElement(),
|
|
8
|
-
Interval: {},
|
|
9
|
-
LocalDataScope: {},
|
|
10
|
-
Init: function (options = { type: 'user', id: 'main', element: {} }) {
|
|
11
|
-
const { type, id, element } = options;
|
|
12
|
-
this.Data[type][id] = {
|
|
13
|
-
...BaseElement()[type].main,
|
|
14
|
-
...this.Data[type][id],
|
|
15
|
-
...element,
|
|
16
|
-
};
|
|
17
|
-
if (!this.Interval[type]) this.Interval[type] = {};
|
|
18
|
-
if (!this.Interval[type][id]) this.Interval[type][id] = {};
|
|
19
|
-
if (!this.LocalDataScope[type]) this.LocalDataScope[type] = {};
|
|
20
|
-
if (!this.LocalDataScope[type][id])
|
|
21
|
-
this.LocalDataScope[type][id] = {
|
|
22
|
-
path: [],
|
|
23
|
-
};
|
|
24
|
-
},
|
|
25
|
-
removeAll: function () {
|
|
26
|
-
for (const type of Object.keys(this.Data)) {
|
|
27
|
-
for (const id of Object.keys(this.Data[type])) {
|
|
28
|
-
if (this.Interval[type] && this.Interval[type][id]) {
|
|
29
|
-
for (const interval of Object.keys(this.Interval[type][id])) clearInterval(this.Interval[type][id][interval]);
|
|
30
|
-
}
|
|
31
|
-
}
|
|
32
|
-
}
|
|
33
|
-
this.Interval = {};
|
|
34
|
-
this.Data = BaseElement();
|
|
35
|
-
},
|
|
36
|
-
};
|
|
37
|
-
|
|
38
|
-
export { ElementsDefault };
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
const CoreWsMailerManagement = {
|
|
2
|
-
element: {},
|
|
3
|
-
instance: function (wsManagementId = '') {
|
|
4
|
-
this.element[wsManagementId] = {};
|
|
5
|
-
},
|
|
6
|
-
getUserWsId: function (wsManagementId = '', id = '') {
|
|
7
|
-
for (const userWsId of Object.keys(this.element[wsManagementId])) {
|
|
8
|
-
if (this.element[wsManagementId][userWsId].model.user._id === id) {
|
|
9
|
-
return userWsId;
|
|
10
|
-
}
|
|
11
|
-
}
|
|
12
|
-
return undefined;
|
|
13
|
-
},
|
|
14
|
-
};
|
|
15
|
-
|
|
16
|
-
export { CoreWsMailerManagement };
|