refactor: reuse vite's websocket

This commit is contained in:
Arash Sheyda
2023-06-18 12:20:57 +03:00
parent 2ba8bdadfc
commit aaef56cb29
9 changed files with 122 additions and 100 deletions

1
.nuxtrc Normal file
View File

@ -0,0 +1 @@
typescript.includeWorkspace=true

View File

@ -81,8 +81,9 @@ function editDocument(document: any) {
async function saveDocument(document: any, create = true) { async function saveDocument(document: any, create = true) {
const method = create ? rpc.createDocument : rpc.updateDocument const method = create ? rpc.createDocument : rpc.updateDocument
const newDocument = await method(props.collection, document) const newDocument = await method(props.collection, document)
// TODO: show toast
if (newDocument?.error) if (newDocument?.error)
return alert(newDocument.error.message) return
if (create) { if (create) {
if (!documents.value.length) { if (!documents.value.length) {
@ -105,8 +106,9 @@ function discardEditing() {
async function deleteDocument(document: any) { async function deleteDocument(document: any) {
const newDocument = await rpc.deleteDocument(props.collection, document._id) const newDocument = await rpc.deleteDocument(props.collection, document._id)
// TODO: show toast
if (newDocument.deletedCount === 0) if (newDocument.deletedCount === 0)
return alert('Failed to delete document') return
documents.value = documents.value.filter((doc: any) => doc._id !== document._id) documents.value = documents.value.filter((doc: any) => doc._id !== document._id)
} }

View File

@ -1,14 +1,14 @@
import { createBirpc } from 'birpc' import { createBirpc } from 'birpc'
import { parse, stringify } from 'flatted' import { parse, stringify } from 'flatted'
import { createHotContext } from 'vite-hot-client'
import type { ClientFunctions, ServerFunctions } from '../../src/types' import type { ClientFunctions, ServerFunctions } from '../../src/types'
import { PATH_ENTRY } from '../../src/constants' import { WS_EVENT_NAME } from '../../src/constants'
const RECONNECT_INTERVAL = 2000
export const wsConnecting = ref(true) export const wsConnecting = ref(true)
export const wsError = ref<any>() export const wsError = ref<any>()
export const wsConnectingDebounced = useDebounce(wsConnecting, 2000)
let connectPromise = connectWS() const connectPromise = connectVite()
let onMessage: Function = () => {} let onMessage: Function = () => {}
export const clientFunctions = { export const clientFunctions = {
@ -19,9 +19,11 @@ export const extendedRpcMap = new Map<string, any>()
export const rpc = createBirpc<ServerFunctions>(clientFunctions, { export const rpc = createBirpc<ServerFunctions>(clientFunctions, {
post: async (d) => { post: async (d) => {
(await connectPromise).send(d) (await connectPromise).send(WS_EVENT_NAME, d)
},
on: (fn) => {
onMessage = fn
}, },
on: (fn) => { onMessage = fn },
serialize: stringify, serialize: stringify,
deserialize: parse, deserialize: parse,
resolver(name, fn) { resolver(name, fn) {
@ -35,35 +37,22 @@ export const rpc = createBirpc<ServerFunctions>(clientFunctions, {
onError(error, name) { onError(error, name) {
console.error(`[nuxt-devtools] RPC error on executing "${name}":`, error) console.error(`[nuxt-devtools] RPC error on executing "${name}":`, error)
}, },
timeout: 120_000,
}) })
async function connectWS() { async function connectVite() {
const wsUrl = new URL(`ws://host${PATH_ENTRY}`) const hot = await createHotContext()
wsUrl.protocol = location.protocol === 'https:' ? 'wss:' : 'ws:'
wsUrl.host = 'localhost:3000'
const ws = new WebSocket(wsUrl.toString()) if (!hot)
ws.addEventListener('message', e => onMessage(String(e.data))) throw new Error('Unable to connect to devtools')
ws.addEventListener('error', (e) => {
console.error(e) hot.on(WS_EVENT_NAME, (data) => {
wsError.value = e onMessage(data)
}) })
ws.addEventListener('close', () => {
// eslint-disable-next-line no-console
console.log('[nuxt-devtools] WebSocket closed, reconnecting...')
wsConnecting.value = true
setTimeout(async () => {
connectPromise = connectWS()
}, RECONNECT_INTERVAL)
})
wsConnecting.value = true
if (ws.readyState !== WebSocket.OPEN)
await new Promise(resolve => ws.addEventListener('open', resolve))
// eslint-disable-next-line no-console // TODO:
console.log('[nuxt-devtools] WebSocket connected.') // hot.on('vite:connect', (data) => {})
wsConnecting.value = false // hot.on('vite:disconnect', (data) => {})
wsError.value = null
return ws return hot
} }

View File

@ -14,10 +14,6 @@
"require": "./module.cjs", "require": "./module.cjs",
"import": "./dist/module.mjs" "import": "./dist/module.mjs"
}, },
"./types": {
"types": "./dist/types.d.ts",
"import": "./dist/types.mjs"
},
"./*": "./*" "./*": "./*"
}, },
"main": "./module.cjs", "main": "./module.cjs",
@ -46,10 +42,11 @@
"flatted": "^3.2.7", "flatted": "^3.2.7",
"fs-extra": "^11.1.1", "fs-extra": "^11.1.1",
"mongoose": "^7.2.2", "mongoose": "^7.2.2",
"ofetch": "^1.1.0",
"pathe": "^1.1.0", "pathe": "^1.1.0",
"pluralize": "^8.0.0", "pluralize": "^8.0.0",
"sirv": "^2.0.3", "sirv": "^2.0.3",
"tinyws": "^0.1.0", "vite-hot-client": "^0.2.1",
"ws": "^8.13.0" "ws": "^8.13.0"
}, },
"devDependencies": { "devDependencies": {

66
pnpm-lock.yaml generated
View File

@ -1,4 +1,8 @@
lockfileVersion: '6.0' lockfileVersion: '6.1'
settings:
autoInstallPeers: true
excludeLinksFromLockfile: false
dependencies: dependencies:
'@nuxt/devtools-kit': '@nuxt/devtools-kit':
@ -25,6 +29,9 @@ dependencies:
mongoose: mongoose:
specifier: ^7.2.2 specifier: ^7.2.2
version: 7.2.2 version: 7.2.2
ofetch:
specifier: ^1.1.0
version: 1.1.0
pathe: pathe:
specifier: ^1.1.0 specifier: ^1.1.0
version: 1.1.0 version: 1.1.0
@ -34,9 +41,9 @@ dependencies:
sirv: sirv:
specifier: ^2.0.3 specifier: ^2.0.3
version: 2.0.3 version: 2.0.3
tinyws: vite-hot-client:
specifier: ^0.1.0 specifier: ^0.2.1
version: 0.1.0(ws@8.13.0) version: 0.2.1(vite@4.3.9)
ws: ws:
specifier: ^8.13.0 specifier: ^8.13.0
version: 8.13.0 version: 8.13.0
@ -1150,7 +1157,7 @@ packages:
mri: 1.2.0 mri: 1.2.0
nanoid: 4.0.2 nanoid: 4.0.2
node-fetch: 3.3.1 node-fetch: 3.3.1
ofetch: 1.0.1 ofetch: 1.1.0
parse-git-config: 3.0.0 parse-git-config: 3.0.0
rc9: 2.1.0 rc9: 2.1.0
std-env: 3.3.3 std-env: 3.3.3
@ -1180,7 +1187,7 @@ packages:
defu: 6.1.2 defu: 6.1.2
execa: 7.1.1 execa: 7.1.1
get-port-please: 3.0.1 get-port-please: 3.0.1
ofetch: 1.0.1 ofetch: 1.1.0
pathe: 1.1.0 pathe: 1.1.0
ufo: 1.1.2 ufo: 1.1.2
vitest: 0.31.2(sass@1.62.1) vitest: 0.31.2(sass@1.62.1)
@ -1820,7 +1827,7 @@ packages:
dependencies: dependencies:
'@iconify/utils': 2.1.5 '@iconify/utils': 2.1.5
'@unocss/core': 0.52.5 '@unocss/core': 0.52.5
ofetch: 1.0.1 ofetch: 1.1.0
transitivePeerDependencies: transitivePeerDependencies:
- supports-color - supports-color
dev: true dev: true
@ -1857,7 +1864,7 @@ packages:
resolution: {integrity: sha512-oynjtFC05NQM1RXith7LM0MCQGqy4yF0tFRnpM8zmq3p6L7XPFitL5Lmx/walXOB7bw9QSL+gMjamdpEo+KjQg==} resolution: {integrity: sha512-oynjtFC05NQM1RXith7LM0MCQGqy4yF0tFRnpM8zmq3p6L7XPFitL5Lmx/walXOB7bw9QSL+gMjamdpEo+KjQg==}
dependencies: dependencies:
'@unocss/core': 0.52.5 '@unocss/core': 0.52.5
ofetch: 1.0.1 ofetch: 1.1.0
dev: true dev: true
/@unocss/preset-wind@0.52.5: /@unocss/preset-wind@0.52.5:
@ -2900,7 +2907,7 @@ packages:
execa: 7.1.1 execa: 7.1.1
mri: 1.2.0 mri: 1.2.0
node-fetch-native: 1.1.1 node-fetch-native: 1.1.1
ofetch: 1.0.1 ofetch: 1.1.0
open: 9.1.0 open: 9.1.0
pathe: 1.1.0 pathe: 1.1.0
pkg-types: 1.0.3 pkg-types: 1.0.3
@ -4385,7 +4392,7 @@ packages:
defu: 6.1.2 defu: 6.1.2
https-proxy-agent: 5.0.1 https-proxy-agent: 5.0.1
mri: 1.2.0 mri: 1.2.0
node-fetch-native: 1.1.1 node-fetch-native: 1.2.0
pathe: 1.1.0 pathe: 1.1.0
tar: 6.1.15 tar: 6.1.15
transitivePeerDependencies: transitivePeerDependencies:
@ -5827,8 +5834,8 @@ packages:
mime: 3.0.0 mime: 3.0.0
mlly: 1.3.0 mlly: 1.3.0
mri: 1.2.0 mri: 1.2.0
node-fetch-native: 1.1.1 node-fetch-native: 1.2.0
ofetch: 1.0.1 ofetch: 1.1.0
ohash: 1.1.2 ohash: 1.1.2
openapi-typescript: 6.2.6 openapi-typescript: 6.2.6
pathe: 1.1.0 pathe: 1.1.0
@ -5868,6 +5875,10 @@ packages:
/node-fetch-native@1.1.1: /node-fetch-native@1.1.1:
resolution: {integrity: sha512-9VvspTSUp2Sxbl+9vbZTlFGq9lHwE8GDVVekxx6YsNd1YH59sb3Ba8v3Y3cD8PkLNcileGGcA21PFjVl0jzDaw==} resolution: {integrity: sha512-9VvspTSUp2Sxbl+9vbZTlFGq9lHwE8GDVVekxx6YsNd1YH59sb3Ba8v3Y3cD8PkLNcileGGcA21PFjVl0jzDaw==}
dev: true
/node-fetch-native@1.2.0:
resolution: {integrity: sha512-5IAMBTl9p6PaAjYCnMv5FmqIF6GcZnawAVnzaCG0rX2aYZJ4CxEkZNtVPuTRug7fL7wyM5BQYTlAzcyMPi6oTQ==}
/node-fetch@2.6.11: /node-fetch@2.6.11:
resolution: {integrity: sha512-4I6pdBY1EthSqDmJkiNk3JIT8cswwR9nfeW/cPdUagJYEQG7R95WRH74wpz7ma8Gh/9dI9FP+OU+0E4FvtA55w==} resolution: {integrity: sha512-4I6pdBY1EthSqDmJkiNk3JIT8cswwR9nfeW/cPdUagJYEQG7R95WRH74wpz7ma8Gh/9dI9FP+OU+0E4FvtA55w==}
@ -6106,7 +6117,7 @@ packages:
nitropack: 2.4.1 nitropack: 2.4.1
nuxi: 3.5.2 nuxi: 3.5.2
nypm: 0.2.0 nypm: 0.2.0
ofetch: 1.0.1 ofetch: 1.1.0
ohash: 1.1.2 ohash: 1.1.2
pathe: 1.1.0 pathe: 1.1.0
perfect-debounce: 1.0.0 perfect-debounce: 1.0.0
@ -6200,11 +6211,11 @@ packages:
es-abstract: 1.21.2 es-abstract: 1.21.2
dev: true dev: true
/ofetch@1.0.1: /ofetch@1.1.0:
resolution: {integrity: sha512-icBz2JYfEpt+wZz1FRoGcrMigjNKjzvufE26m9+yUiacRQRHwnNlGRPiDnW4op7WX/MR6aniwS8xw8jyVelF2g==} resolution: {integrity: sha512-yjq2ZUUMto1ITpge2J5vNlUfteLzxfHn9aJC55WtVGD3okKwSfPoLaKpcHXmmKd2kZZUGo+jdkFuuj09Blyeig==}
dependencies: dependencies:
destr: 1.2.2 destr: 1.2.2
node-fetch-native: 1.1.1 node-fetch-native: 1.2.0
ufo: 1.1.2 ufo: 1.1.2
/ohash@1.1.2: /ohash@1.1.2:
@ -7661,15 +7672,6 @@ packages:
engines: {node: '>=14.0.0'} engines: {node: '>=14.0.0'}
dev: true dev: true
/tinyws@0.1.0(ws@8.13.0):
resolution: {integrity: sha512-6WQ2FlFM7qm6lAXxeKnzsAEfmnBHz5W5EwonNs52V0++YfK1IoCCAWM429afcChFE9BFrDgOFnq7ligaWMsa/A==}
engines: {node: '>=12.4'}
peerDependencies:
ws: '>=8'
dependencies:
ws: 8.13.0
dev: false
/titleize@3.0.0: /titleize@3.0.0:
resolution: {integrity: sha512-KxVu8EYHDPBdUYdKZdKtU2aj2XfEx9AfjXxE/Aj0vT06w2icA09Vus1rh6eSu1y01akYg6BjIK/hxyLJINoMLQ==} resolution: {integrity: sha512-KxVu8EYHDPBdUYdKZdKtU2aj2XfEx9AfjXxE/Aj0vT06w2icA09Vus1rh6eSu1y01akYg6BjIK/hxyLJINoMLQ==}
engines: {node: '>=12'} engines: {node: '>=12'}
@ -7887,7 +7889,7 @@ packages:
consola: 3.1.0 consola: 3.1.0
defu: 6.1.2 defu: 6.1.2
mime: 3.0.0 mime: 3.0.0
node-fetch-native: 1.1.1 node-fetch-native: 1.2.0
pathe: 1.1.0 pathe: 1.1.0
/unhead@1.1.27: /unhead@1.1.27:
@ -8064,8 +8066,8 @@ packages:
listhen: 1.0.4 listhen: 1.0.4
lru-cache: 9.1.1 lru-cache: 9.1.1
mri: 1.2.0 mri: 1.2.0
node-fetch-native: 1.1.1 node-fetch-native: 1.2.0
ofetch: 1.0.1 ofetch: 1.1.0
ufo: 1.1.2 ufo: 1.1.2
transitivePeerDependencies: transitivePeerDependencies:
- supports-color - supports-color
@ -8152,6 +8154,14 @@ packages:
builtins: 5.0.1 builtins: 5.0.1
dev: true dev: true
/vite-hot-client@0.2.1(vite@4.3.9):
resolution: {integrity: sha512-UqsQdw5PODnSrTDT85nr09RlhV0gkm2Xat74U2l8JZ5R8M/wTCggWSyPjxbLk5fbbVnWfr0JwW+vVoosjQnYrA==}
peerDependencies:
vite: ^2.6.0 || ^3.0.0 || ^4.0.0
dependencies:
vite: 4.3.9(@types/node@20.2.5)(sass@1.62.1)
dev: false
/vite-node@0.31.2(@types/node@20.2.5)(sass@1.62.1): /vite-node@0.31.2(@types/node@20.2.5)(sass@1.62.1):
resolution: {integrity: sha512-NvoO7+zSvxROC4JY8cyp/cO7DHAX3dwMOHQVDdNtCZ4Zq8wInnR/bJ/lfsXqE6wrUgtYCE5/84qHS+A7vllI3A==} resolution: {integrity: sha512-NvoO7+zSvxROC4JY8cyp/cO7DHAX3dwMOHQVDdNtCZ4Zq8wInnR/bJ/lfsXqE6wrUgtYCE5/84qHS+A7vllI3A==}
engines: {node: '>=v14.18.0'} engines: {node: '>=v14.18.0'}

View File

@ -1,3 +1,3 @@
export const PATH = '/__nuxt_mongoose__' export const PATH = '/__nuxt_mongoose__'
export const PATH_ENTRY = `${PATH}/entry`
export const PATH_CLIENT = `${PATH}/client` export const PATH_CLIENT = `${PATH}/client`
export const WS_EVENT_NAME = 'nuxt:devtools:mongoose:rpc'

View File

@ -2,17 +2,19 @@ import {
addImportsDir, addImportsDir,
addServerPlugin, addServerPlugin,
addTemplate, addTemplate,
addVitePlugin,
createResolver, createResolver,
defineNuxtModule, defineNuxtModule,
logger, logger,
} from '@nuxt/kit' } from '@nuxt/kit'
import { pathExists } from 'fs-extra' import { pathExists } from 'fs-extra'
import { tinyws } from 'tinyws'
import { join } from 'pathe' import { join } from 'pathe'
import { defu } from 'defu' import { defu } from 'defu'
import sirv from 'sirv' import sirv from 'sirv'
import { $fetch } from 'ofetch'
import { version } from '../package.json'
import { PATH_CLIENT, PATH_ENTRY } from './constants' import { PATH_CLIENT } from './constants'
import type { ModuleOptions } from './types' import type { ModuleOptions } from './types'
import { setupRPC } from './server-rpc' import { setupRPC } from './server-rpc'
@ -34,6 +36,13 @@ export default defineNuxtModule<ModuleOptions>({
const { resolve } = createResolver(import.meta.url) const { resolve } = createResolver(import.meta.url)
const runtimeConfig = nuxt.options.runtimeConfig const runtimeConfig = nuxt.options.runtimeConfig
if (nuxt.options.dev) {
$fetch('https://registry.npmjs.org/nuxt-mongoose/latest').then((release) => {
if (release.version > version)
logger.info(`A new version of Nuxt Mongoose (v${release.version}) is available: https://github.com/arashsheyda/nuxt-mongoose/releases/latest`)
}).catch(() => {})
}
addImportsDir(resolve('./runtime/composables')) addImportsDir(resolve('./runtime/composables'))
if (!options.uri) { if (!options.uri) {
@ -58,11 +67,11 @@ export default defineNuxtModule<ModuleOptions>({
} }
const clientPath = distResolve('./client') const clientPath = distResolve('./client')
const { middleware: rpcMiddleware } = setupRPC(nuxt, options) const { vitePlugin } = setupRPC(nuxt, options)
addVitePlugin(vitePlugin)
nuxt.hook('vite:serverCreated', async (server) => { nuxt.hook('vite:serverCreated', async (server) => {
server.middlewares.use(PATH_ENTRY, tinyws() as any)
server.middlewares.use(PATH_ENTRY, rpcMiddleware as any)
if (await pathExists(clientPath)) if (await pathExists(clientPath))
server.middlewares.use(PATH_CLIENT, sirv(clientPath, { dev: true, single: true })) server.middlewares.use(PATH_CLIENT, sirv(clientPath, { dev: true, single: true }))
}) })

View File

@ -11,7 +11,7 @@ export async function defineMongooseConnection({ uri, options }: { uri?: string;
try { try {
await mongoose.connect(mongooseUri, { ...mongooseOptions }) await mongoose.connect(mongooseUri, { ...mongooseOptions })
logger.info('Connected to mongoose database') logger.info('Connected to MONGODB')
} }
catch (err) { catch (err) {
logger.error('Error connecting to database', err) logger.error('Error connecting to database', err)

View File

@ -1,12 +1,12 @@
import type { TinyWSRequest } from 'tinyws'
import type { NodeIncomingMessage, NodeServerResponse } from 'h3'
import type { WebSocket } from 'ws' import type { WebSocket } from 'ws'
import { createBirpcGroup } from 'birpc' import { createBirpcGroup } from 'birpc'
import type { ChannelOptions } from 'birpc' import type { ChannelOptions } from 'birpc'
import { parse, stringify } from 'flatted' import { parse, stringify } from 'flatted'
import type { Plugin } from 'vite'
import type { Nuxt } from 'nuxt/schema' import type { Nuxt } from 'nuxt/schema'
import type { ClientFunctions, ModuleOptions, NuxtDevtoolsServerContext, ServerFunctions } from '../types' import type { ClientFunctions, ModuleOptions, NuxtDevtoolsServerContext, ServerFunctions } from '../types'
import { WS_EVENT_NAME } from '../constants'
import { setupDatabaseRPC } from './database' import { setupDatabaseRPC } from './database'
import { setupResourceRPC } from './resource' import { setupResourceRPC } from './resource'
@ -68,14 +68,30 @@ export function setupRPC(nuxt: Nuxt, options: ModuleOptions): any {
} satisfies Partial<ServerFunctions>) } satisfies Partial<ServerFunctions>)
const wsClients = new Set<WebSocket>() const wsClients = new Set<WebSocket>()
const middleware = async (req: NodeIncomingMessage & TinyWSRequest, _res: NodeServerResponse, next: Function) => {
// Handle WebSocket const vitePlugin: Plugin = {
if (req.ws) { name: 'nuxt:devtools:rpc',
const ws = await req.ws() configureServer(server) {
server.ws.on('connection', (ws) => {
wsClients.add(ws) wsClients.add(ws)
const channel: ChannelOptions = { const channel: ChannelOptions = {
post: d => ws.send(d), post: d => ws.send(JSON.stringify({
on: fn => ws.on('message', fn), type: 'custom',
event: WS_EVENT_NAME,
data: d,
})),
on: (fn) => {
ws.on('message', (e) => {
try {
const data = JSON.parse(String(e)) || {}
if (data.type === 'custom' && data.event === WS_EVENT_NAME) {
// console.log(data.data)
fn(data.data)
}
}
catch {}
})
},
serialize: stringify, serialize: stringify,
deserialize: parse, deserialize: parse,
} }
@ -90,14 +106,12 @@ export function setupRPC(nuxt: Nuxt, options: ModuleOptions): any {
c.splice(index, 1) c.splice(index, 1)
}) })
}) })
} })
else { },
next()
}
} }
return { return {
middleware, vitePlugin,
...ctx, ...ctx,
} }
} }