refactor server

Signed-off-by: Andrey Platov <andrey@hardcoreeng.com>
This commit is contained in:
Andrey Platov 2021-08-04 18:17:01 +02:00
parent 57c17692eb
commit a78f96ba7d
No known key found for this signature in database
GPG Key ID: C8787EFEB4B64AF0
13 changed files with 222 additions and 18 deletions

View File

@ -3,6 +3,7 @@ lockfileVersion: 5.3
specifiers:
'@microsoft/api-extractor': ^7.18.4
'@rush-temp/core': file:./projects/core.tgz
'@rush-temp/dev-server': file:./projects/dev-server.tgz
'@rush-temp/dev-storage': file:./projects/dev-storage.tgz
'@rush-temp/platform': file:./projects/platform.tgz
'@rush-temp/platform-rig': file:./projects/platform-rig.tgz
@ -28,6 +29,7 @@ specifiers:
dependencies:
'@microsoft/api-extractor': 7.18.4
'@rush-temp/core': file:projects/core.tgz_6c259fadfeb3a4b20890aefe87070b8b
'@rush-temp/dev-server': file:projects/dev-server.tgz_6c259fadfeb3a4b20890aefe87070b8b
'@rush-temp/dev-storage': file:projects/dev-storage.tgz_6c259fadfeb3a4b20890aefe87070b8b
'@rush-temp/platform': file:projects/platform.tgz_6c259fadfeb3a4b20890aefe87070b8b
'@rush-temp/platform-rig': file:projects/platform-rig.tgz_6ab28797e7a22071465f7d680ae81ae5
@ -5201,8 +5203,26 @@ packages:
- typescript
dev: false
file:projects/dev-server.tgz_6c259fadfeb3a4b20890aefe87070b8b:
resolution: {integrity: sha512-wWD3XTjUjOM0zuwKN3Y8nuzCmWTPR0m554z+Vun/V20HOJ3ykNCe+oYJUIbSgvZfL647XDkomAVGZtCMl7qu/w==, tarball: file:projects/dev-server.tgz}
id: file:projects/dev-server.tgz
name: '@rush-temp/dev-server'
version: 0.0.0
dependencies:
'@types/heft-jest': 1.0.2
'@typescript-eslint/eslint-plugin': 4.28.5_a8e83fcad666e1ba86be4b2e27a20aea
eslint: 7.32.0
eslint-plugin-import: 2.23.4_eslint@7.32.0
eslint-plugin-node: 11.1.0_eslint@7.32.0
eslint-plugin-promise: 4.3.1
transitivePeerDependencies:
- '@typescript-eslint/parser'
- supports-color
- typescript
dev: false
file:projects/dev-storage.tgz_6c259fadfeb3a4b20890aefe87070b8b:
resolution: {integrity: sha512-Jf/W4dvCH5kUvK2ItxuijWJM4IGZf8Hi1noFrzk9PLL+2WNrD5t7sPB9c/lv4KzKEhX5gTV4ZkLBHNyD+lo2oQ==, tarball: file:projects/dev-storage.tgz}
resolution: {integrity: sha512-upE4u9M1Pna5G71wXTMBeMd+pGiZbjjyhpXfHZR+lgSthSPnC7avk/Ozb7L2gL87zaqK4ZhKaiI+WNg6FeGQDg==, tarball: file:projects/dev-storage.tgz}
id: file:projects/dev-storage.tgz
name: '@rush-temp/dev-storage'
version: 0.0.0

6
dev/server/.eslintrc.js Normal file
View File

@ -0,0 +1,6 @@
module.exports = {
extends: ['./node_modules/@anticrm/platform-rig/profiles/default/config/eslint.config.json'],
parserOptions: {
project: './tsconfig.json'
}
}

View File

@ -0,0 +1,18 @@
// The "rig.json" file directs tools to look for their config files in an external package.
// Documentation for this system: https://www.npmjs.com/package/@rushstack/rig-package
{
"$schema": "https://developer.microsoft.com/json-schemas/rig-package/rig.schema.json",
/**
* (Required) The name of the rig package to inherit from.
* It should be an NPM package name with the "-rig" suffix.
*/
"rigPackageName": "@anticrm/platform-rig"
/**
* (Optional) Selects a config profile from the rig package. The name must consist of
* lowercase alphanumeric words separated by hyphens, for example "sample-profile".
* If omitted, then the "default" profile will be used."
*/
// "rigProfile": "your-profile-name"
}

30
dev/server/package.json Normal file
View File

@ -0,0 +1,30 @@
{
"name": "@anticrm/dev-server",
"version": "0.6.0",
"main": "lib/index.js",
"author": "Anticrm Platform Contributors",
"license": "EPL-2.0",
"scripts": {
"build": "heft build",
"build:docs": "api-extractor run --local",
"test": "jest",
"lint": "ts-standard src",
"lint:fix": "eslint --fix ./src",
"format": "prettier --write 'src/**/*.{ts*,js*,yml}' && ts-standard --fix src"
},
"devDependencies": {
"@anticrm/platform-rig":"~0.6.0",
"@types/heft-jest":"^1.0.2",
"@typescript-eslint/eslint-plugin":"4",
"eslint-plugin-import":"2",
"eslint-plugin-promise":"4",
"eslint-plugin-node":"11",
"eslint":"^7.32.0"
},
"dependencies": {
"@anticrm/dev-storage": "~0.6.0",
"@anticrm/server-ws": "~0.6.0",
"@anticrm/core": "~0.6.0",
"@anticrm/platform": "~0.6.0"
}
}

17
dev/server/src/index.ts Normal file
View File

@ -0,0 +1,17 @@
//
// Copyright © 2020, 2021 Anticrm Platform Contributors.
// Copyright © 2021 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
//
export { start } from './server'

27
dev/server/src/server.ts Normal file
View File

@ -0,0 +1,27 @@
//
// Copyright © 2020, 2021 Anticrm Platform Contributors.
// Copyright © 2021 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
//
import { start as startJsonRpc } from '@anticrm/server-ws'
import { createStorage } from '@anticrm/dev-storage'
import { DevSession } from './session'
/**
* @public
*/
export async function start (): Promise<void> {
const storage = await createStorage()
startJsonRpc(server => new DevSession(server, storage), 3333)
}

37
dev/server/src/session.ts Normal file
View File

@ -0,0 +1,37 @@
//
// Copyright © 2020, 2021 Anticrm Platform Contributors.
// Copyright © 2021 Hardcore Engineering Inc.
//
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License. You may
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
//
// See the License for the specific language governing permissions and
// limitations under the License.
//
import type { Class, Doc, DocumentQuery, FindOptions, FindResult, Ref, Storage, Tx } from '@anticrm/core'
import { JsonRpcServer } from '@anticrm/server-ws'
import type { ServerStorage } from '@anticrm/dev-storage'
export class DevSession implements Storage {
constructor (
private readonly server: JsonRpcServer,
private readonly storage: ServerStorage
) {}
async findAll <T extends Doc>(_class: Ref<Class<T>>, query: DocumentQuery<T>, options?: FindOptions<T>): Promise<FindResult<T>> {
return await this.storage.findAll(_class, query, options)
}
async tx (tx: Tx<Doc>): Promise<void> {
const derived = await this.storage.tx(tx)
for (const tx of derived) {
this.server.broadcast(this, { result: tx })
}
}
}

8
dev/server/tsconfig.json Normal file
View File

@ -0,0 +1,8 @@
{
"extends": "./node_modules/@anticrm/platform-rig/profiles/default/tsconfig.json",
"compilerOptions": {
"rootDir": "./src",
"outDir": "./lib"
}
}

View File

@ -14,4 +14,4 @@
// limitations under the License.
//
export { createStorage } from './storage'
export { createStorage, ServerStorage } from './storage'

View File

@ -15,7 +15,6 @@
import type {
Tx,
Storage,
Ref,
Doc,
Class,
@ -26,18 +25,30 @@ import type {
import { getResource } from '@anticrm/platform'
import core, { ModelDb, TxDb, Hierarchy, DOMAIN_TX, DefaultTxFactory } from '@anticrm/core'
/**
* @public
*/
export interface ServerStorage {
findAll: <T extends Doc>(
_class: Ref<Class<T>>,
query: DocumentQuery<T>,
options?: FindOptions<T>
) => Promise<FindResult<T>>
tx: (tx: Tx) => Promise<Tx[]>
}
async function getModel (): Promise<Tx[]> {
return import('./model.tx.json') as unknown as Tx[]
}
class DevStorage implements Storage {
class DevStorage implements ServerStorage {
private readonly txFactory: DefaultTxFactory
constructor (
private readonly hierarchy: Hierarchy,
private readonly txdb: TxDb,
private readonly modeldb: ModelDb,
private readonly handler: (tx: Tx) => void) {
private readonly modeldb: ModelDb
) {
this.txFactory = new DefaultTxFactory(core.account.System)
}
@ -51,7 +62,7 @@ class DevStorage implements Storage {
return await this.modeldb.findAll(_class, query, options)
}
async tx (tx: Tx): Promise<void> {
async tx (tx: Tx): Promise<Tx[]> {
if (tx.objectSpace === core.space.Model) {
this.hierarchy.tx(tx)
}
@ -59,15 +70,18 @@ class DevStorage implements Storage {
// invoke triggers
const triggers = this.hierarchy.getClass(tx.objectClass).triggers
if (triggers !== undefined) {
const derived: Tx[] = []
for (const trigger of triggers) {
const impl = await getResource(trigger)
const txes = await impl(tx, this.txFactory)
derived.push(...txes)
for (const tx of txes) {
await Promise.all([this.modeldb.tx(tx), this.txdb.tx(tx)])
this.handler(tx)
}
}
return derived
}
return []
}
}
@ -76,7 +90,7 @@ class DevStorage implements Storage {
* @param handler -
* @returns
*/
export async function createStorage (handler: (tx: Tx) => void): Promise<Storage> {
export async function createStorage (): Promise<ServerStorage> {
const txes = await getModel()
const hierarchy = new Hierarchy()
@ -88,5 +102,5 @@ export async function createStorage (handler: (tx: Tx) => void): Promise<Storage
await Promise.all([transactions.tx(tx), model.tx(tx)])
}
return new DevStorage(hierarchy, transactions, model, handler)
return new DevStorage(hierarchy, transactions, model)
}

View File

@ -470,6 +470,11 @@
"packageName": "@anticrm/dev-storage",
"projectFolder": "dev/storage",
"shouldPublish": true
},
{
"packageName": "@anticrm/dev-server",
"projectFolder": "dev/server",
"shouldPublish": true
}
]
}

View File

@ -14,4 +14,4 @@
// limitations under the License.
//
export { start } from './server'
export * from './server'

View File

@ -14,7 +14,7 @@
// limitations under the License.
//
import { readRequest, serialize } from '@anticrm/platform'
import { readRequest, serialize, Response } from '@anticrm/platform'
import { createServer, IncomingMessage } from 'http'
import WebSocket, { Server } from 'ws'
import { decode } from 'jwt-simple'
@ -38,21 +38,43 @@ async function handleRequest<S> (service: S, ws: WebSocket, msg: string): Promis
/**
* @public
* @param serviceFactory -
*/
// eslint-disable-next-line @typescript-eslint/no-empty-interface
export interface Session {}
/**
* @public
*/
export interface JsonRpcServer {
broadcast: (from: Session, resp: Response<any>) => void
}
/**
* @public
* @param sessionFactory -
* @param port -
* @param host -
*/
export function start<S> (serviceFactory: () => Promise<S>, port: number, host?: string): void {
export function start (sessionFactory: (server: JsonRpcServer) => Session, port: number, host?: string): void {
// console.log('starting server on port ' + port + '...')
// console.log('host: ' + host)
const connections: S[] = []
const sessions: [Session, WebSocket][] = []
const jsonServer: JsonRpcServer = {
broadcast (from: Session, resp: Response<[]>) {
const msg = serialize(resp)
for (const session of sessions) {
if (session[0] !== from) { session[1].send(msg) }
}
}
}
const wss = new Server({ noServer: true })
// eslint-disable-next-line @typescript-eslint/no-misused-promises
wss.on('connection', async (ws: WebSocket, request: any, token: _Token) => {
const service = await serviceFactory()
connections.push(service)
wss.on('connection', (ws: WebSocket, request: any, token: _Token) => {
const service = sessionFactory(jsonServer)
sessions.push([service, ws])
// eslint-disable-next-line @typescript-eslint/no-misused-promises
ws.on('message', async (msg: string) => await handleRequest(service, ws, msg))
})