feat: binary messages through ws + camera demo (#34)
* feat: binary messages through ws + camera demo * chore: prepare release
This commit is contained in:
parent
6f95c6561b
commit
e5cfd8ee38
205
apps/demo-nextjs-app-router/app/camera-turbo/page.tsx
Normal file
205
apps/demo-nextjs-app-router/app/camera-turbo/page.tsx
Normal file
@ -0,0 +1,205 @@
|
|||||||
|
/* eslint-disable @next/next/no-img-element */
|
||||||
|
'use client';
|
||||||
|
|
||||||
|
import * as fal from '@fal-ai/serverless-client';
|
||||||
|
import { MutableRefObject, useEffect, useRef, useState } from 'react';
|
||||||
|
|
||||||
|
fal.config({
|
||||||
|
proxyUrl: '/api/fal/proxy',
|
||||||
|
});
|
||||||
|
|
||||||
|
const EMPTY_IMG =
|
||||||
|
'';
|
||||||
|
|
||||||
|
type WebcamOptions = {
|
||||||
|
videoRef: MutableRefObject<HTMLVideoElement | null>;
|
||||||
|
previewRef: MutableRefObject<HTMLCanvasElement | null>;
|
||||||
|
onFrameUpdate?: (data: Uint8Array) => void;
|
||||||
|
width?: number;
|
||||||
|
height?: number;
|
||||||
|
};
|
||||||
|
const useWebcam = ({
|
||||||
|
videoRef,
|
||||||
|
previewRef,
|
||||||
|
onFrameUpdate,
|
||||||
|
width = 512,
|
||||||
|
height = 512,
|
||||||
|
}: WebcamOptions) => {
|
||||||
|
useEffect(() => {
|
||||||
|
if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {
|
||||||
|
navigator.mediaDevices.getUserMedia({ video: true }).then((stream) => {
|
||||||
|
if (videoRef.current !== null) {
|
||||||
|
videoRef.current.srcObject = stream;
|
||||||
|
videoRef.current.play();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [videoRef]);
|
||||||
|
|
||||||
|
const captureFrame = () => {
|
||||||
|
const canvas = previewRef.current;
|
||||||
|
const video = videoRef.current;
|
||||||
|
if (canvas === null || video === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate the aspect ratio and crop dimensions
|
||||||
|
const aspectRatio = video.videoWidth / video.videoHeight;
|
||||||
|
let sourceX, sourceY, sourceWidth, sourceHeight;
|
||||||
|
|
||||||
|
if (aspectRatio > 1) {
|
||||||
|
// If width is greater than height
|
||||||
|
sourceWidth = video.videoHeight;
|
||||||
|
sourceHeight = video.videoHeight;
|
||||||
|
sourceX = (video.videoWidth - video.videoHeight) / 2;
|
||||||
|
sourceY = 0;
|
||||||
|
} else {
|
||||||
|
// If height is greater than or equal to width
|
||||||
|
sourceWidth = video.videoWidth;
|
||||||
|
sourceHeight = video.videoWidth;
|
||||||
|
sourceX = 0;
|
||||||
|
sourceY = (video.videoHeight - video.videoWidth) / 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resize the canvas to the target dimensions
|
||||||
|
canvas.width = width;
|
||||||
|
canvas.height = height;
|
||||||
|
|
||||||
|
const context = canvas.getContext('2d');
|
||||||
|
if (context === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Draw the image on the canvas (cropped and resized)
|
||||||
|
context.drawImage(
|
||||||
|
video,
|
||||||
|
sourceX,
|
||||||
|
sourceY,
|
||||||
|
sourceWidth,
|
||||||
|
sourceHeight,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
width,
|
||||||
|
height
|
||||||
|
);
|
||||||
|
|
||||||
|
// Callback with frame data
|
||||||
|
if (onFrameUpdate) {
|
||||||
|
canvas.toBlob(
|
||||||
|
(blob) => {
|
||||||
|
blob?.arrayBuffer().then((buffer) => {
|
||||||
|
const frameData = new Uint8Array(buffer);
|
||||||
|
onFrameUpdate(frameData);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
'image/jpeg',
|
||||||
|
0.7
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const interval = setInterval(() => {
|
||||||
|
captureFrame();
|
||||||
|
}, 16); // Adjust interval as needed
|
||||||
|
|
||||||
|
return () => clearInterval(interval);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
type LCMInput = {
|
||||||
|
prompt: string;
|
||||||
|
image: Uint8Array;
|
||||||
|
strength?: number;
|
||||||
|
negative_prompt?: string;
|
||||||
|
seed?: number | null;
|
||||||
|
guidance_scale?: number;
|
||||||
|
num_inference_steps?: number;
|
||||||
|
enable_safety_checks?: boolean;
|
||||||
|
request_id?: string;
|
||||||
|
height?: number;
|
||||||
|
width?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
type LCMOutput = {
|
||||||
|
image: Uint8Array;
|
||||||
|
timings: Record<string, number>;
|
||||||
|
seed: number;
|
||||||
|
num_inference_steps: number;
|
||||||
|
request_id: string;
|
||||||
|
nsfw_content_detected: boolean[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function WebcamPage() {
|
||||||
|
const [enabled, setEnabled] = useState(false);
|
||||||
|
const processedImageRef = useRef<HTMLImageElement | null>(null);
|
||||||
|
const videoRef = useRef<HTMLVideoElement | null>(null);
|
||||||
|
const previewRef = useRef<HTMLCanvasElement | null>(null);
|
||||||
|
|
||||||
|
const { send } = fal.realtime.connect<LCMInput, LCMOutput>(
|
||||||
|
'110602490-sd-turbo-real-time-high-fps-msgpack',
|
||||||
|
{
|
||||||
|
connectionKey: 'camera-turbo-demo',
|
||||||
|
// not throttling the client, handling throttling of the camera itself
|
||||||
|
// and letting all requests through in real-time
|
||||||
|
throttleInterval: 0,
|
||||||
|
onResult(result) {
|
||||||
|
if (processedImageRef.current && result.image) {
|
||||||
|
const blob = new Blob([result.image], { type: 'image/jpeg' });
|
||||||
|
const url = URL.createObjectURL(blob);
|
||||||
|
processedImageRef.current.src = url;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const onFrameUpdate = (data: Uint8Array) => {
|
||||||
|
if (!enabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
send({
|
||||||
|
prompt: 'a picture of leonardo di caprio, elegant, in a suit, 8k, uhd',
|
||||||
|
image: data,
|
||||||
|
num_inference_steps: 3,
|
||||||
|
strength: 0.44,
|
||||||
|
guidance_scale: 1,
|
||||||
|
seed: 6252023,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
useWebcam({
|
||||||
|
videoRef,
|
||||||
|
previewRef,
|
||||||
|
onFrameUpdate,
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<main className="flex-col px-32 mx-auto my-20">
|
||||||
|
<h1 className="text-4xl font-mono mb-8 text-current text-center">
|
||||||
|
fal<code className="font-light text-pink-600">camera</code>
|
||||||
|
</h1>
|
||||||
|
<video ref={videoRef} style={{ display: 'none' }}></video>
|
||||||
|
<div className="py-12 flex items-center justify-center">
|
||||||
|
<button
|
||||||
|
className="py-3 px-4 bg-indigo-700 text-white text-lg rounded"
|
||||||
|
onClick={() => {
|
||||||
|
setEnabled(!enabled);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{enabled ? 'Stop' : 'Start'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col lg:flex-row space-y-4 lg:space-y-0 lg:space-x-4 justify-between">
|
||||||
|
<canvas ref={previewRef} width="512" height="512"></canvas>
|
||||||
|
<img
|
||||||
|
ref={processedImageRef}
|
||||||
|
src={EMPTY_IMG}
|
||||||
|
width={512}
|
||||||
|
height={512}
|
||||||
|
className="min-w-[512px] min-h-[512px]"
|
||||||
|
alt="generated"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
);
|
||||||
|
}
|
||||||
@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "@fal-ai/serverless-client",
|
"name": "@fal-ai/serverless-client",
|
||||||
"description": "The fal serverless JS/TS client",
|
"description": "The fal serverless JS/TS client",
|
||||||
"version": "0.7.0",
|
"version": "0.7.1",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
@ -16,6 +16,7 @@
|
|||||||
"ml"
|
"ml"
|
||||||
],
|
],
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"msgpackr": "^1.10.0",
|
||||||
"robot3": "^0.4.1"
|
"robot3": "^0.4.1"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
|
|||||||
@ -1,4 +1,5 @@
|
|||||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
|
import { pack, unpack } from 'msgpackr';
|
||||||
import {
|
import {
|
||||||
ContextFunction,
|
ContextFunction,
|
||||||
createMachine,
|
createMachine,
|
||||||
@ -75,7 +76,14 @@ function closeConnection(context: Context): Context {
|
|||||||
|
|
||||||
function sendMessage(context: Context, event: SendEvent): Context {
|
function sendMessage(context: Context, event: SendEvent): Context {
|
||||||
if (context.websocket && context.websocket.readyState === WebSocket.OPEN) {
|
if (context.websocket && context.websocket.readyState === WebSocket.OPEN) {
|
||||||
context.websocket.send(JSON.stringify(event.message));
|
if (event.message instanceof Uint8Array) {
|
||||||
|
context.websocket.send(event.message);
|
||||||
|
} else if (shouldSendBinary(event.message)) {
|
||||||
|
context.websocket.send(pack(event.message));
|
||||||
|
} else {
|
||||||
|
context.websocket.send(JSON.stringify(event.message));
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...context,
|
...context,
|
||||||
enqueuedMessage: undefined,
|
enqueuedMessage: undefined,
|
||||||
@ -260,6 +268,16 @@ function buildRealtimeUrl(
|
|||||||
const TOKEN_EXPIRATION_SECONDS = 120;
|
const TOKEN_EXPIRATION_SECONDS = 120;
|
||||||
const DEFAULT_THROTTLE_INTERVAL = 128;
|
const DEFAULT_THROTTLE_INTERVAL = 128;
|
||||||
|
|
||||||
|
function shouldSendBinary(message: any): boolean {
|
||||||
|
return Object.values(message).some(
|
||||||
|
(value) =>
|
||||||
|
value instanceof Buffer ||
|
||||||
|
value instanceof Blob ||
|
||||||
|
value instanceof ArrayBuffer ||
|
||||||
|
value instanceof Uint8Array
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a token to connect to the realtime endpoint.
|
* Get a token to connect to the realtime endpoint.
|
||||||
*/
|
*/
|
||||||
@ -452,7 +470,33 @@ export const realtimeImpl: RealtimeClient = {
|
|||||||
onError(new ApiError({ message: 'Unknown error', status: 500 }));
|
onError(new ApiError({ message: 'Unknown error', status: 500 }));
|
||||||
};
|
};
|
||||||
ws.onmessage = (event) => {
|
ws.onmessage = (event) => {
|
||||||
|
const { onResult } = getCallbacks();
|
||||||
|
|
||||||
|
// Handle binary messages as msgpack messages
|
||||||
|
if (event.data instanceof ArrayBuffer) {
|
||||||
|
const result = unpack(new Uint8Array(event.data));
|
||||||
|
onResult(result);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
event.data instanceof Buffer ||
|
||||||
|
event.data instanceof Uint8Array
|
||||||
|
) {
|
||||||
|
const result = unpack(event.data);
|
||||||
|
onResult(result);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (event.data instanceof Blob) {
|
||||||
|
event.data.arrayBuffer().then((buffer) => {
|
||||||
|
const result = unpack(new Uint8Array(buffer));
|
||||||
|
onResult(result);
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise handle strings as plain JSON messages
|
||||||
const data = JSON.parse(event.data);
|
const data = JSON.parse(event.data);
|
||||||
|
|
||||||
// Drop messages that are not related to the actual result.
|
// Drop messages that are not related to the actual result.
|
||||||
// In the future, we might want to handle other types of messages.
|
// In the future, we might want to handle other types of messages.
|
||||||
// TODO: specify the fal ws protocol format
|
// TODO: specify the fal ws protocol format
|
||||||
@ -461,7 +505,6 @@ export const realtimeImpl: RealtimeClient = {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (isSuccessfulResult(data)) {
|
if (isSuccessfulResult(data)) {
|
||||||
const { onResult } = getCallbacks();
|
|
||||||
onResult(data);
|
onResult(data);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -485,12 +528,18 @@ export const realtimeImpl: RealtimeClient = {
|
|||||||
|
|
||||||
const send = (input: Input & Partial<WithRequestId>) => {
|
const send = (input: Input & Partial<WithRequestId>) => {
|
||||||
// Use throttled send to avoid sending too many messages
|
// Use throttled send to avoid sending too many messages
|
||||||
|
|
||||||
|
const message =
|
||||||
|
input instanceof Uint8Array
|
||||||
|
? input
|
||||||
|
: {
|
||||||
|
...input,
|
||||||
|
request_id: input['request_id'] ?? crypto.randomUUID(),
|
||||||
|
};
|
||||||
|
|
||||||
stateMachine.throttledSend({
|
stateMachine.throttledSend({
|
||||||
type: 'send',
|
type: 'send',
|
||||||
message: {
|
message,
|
||||||
...input,
|
|
||||||
request_id: input['request_id'] ?? crypto.randomUUID(),
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
113
package-lock.json
generated
113
package-lock.json
generated
@ -24,6 +24,7 @@
|
|||||||
"http-proxy": "^1.18.1",
|
"http-proxy": "^1.18.1",
|
||||||
"http-proxy-middleware": "^2.0.6",
|
"http-proxy-middleware": "^2.0.6",
|
||||||
"js-base64": "^3.7.5",
|
"js-base64": "^3.7.5",
|
||||||
|
"msgpackr": "^1.10.0",
|
||||||
"next": "^14.0.3",
|
"next": "^14.0.3",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0",
|
"react-dom": "^18.2.0",
|
||||||
@ -3716,6 +3717,78 @@
|
|||||||
"integrity": "sha512-Hcv+nVC0kZnQ3tD9GVu5xSMR4VVYOteQIr/hwFPVEvPdlXqgGEuRjiheChHgdM+JyqdgNcmzZOX/tnl0JOiI7A==",
|
"integrity": "sha512-Hcv+nVC0kZnQ3tD9GVu5xSMR4VVYOteQIr/hwFPVEvPdlXqgGEuRjiheChHgdM+JyqdgNcmzZOX/tnl0JOiI7A==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"node_modules/@msgpackr-extract/msgpackr-extract-darwin-arm64": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-darwin-arm64/-/msgpackr-extract-darwin-arm64-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-9bfjwDxIDWmmOKusUcqdS4Rw+SETlp9Dy39Xui9BEGEk19dDwH0jhipwFzEff/pFg95NKymc6TOTbRKcWeRqyQ==",
|
||||||
|
"cpu": [
|
||||||
|
"arm64"
|
||||||
|
],
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"node_modules/@msgpackr-extract/msgpackr-extract-darwin-x64": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-darwin-x64/-/msgpackr-extract-darwin-x64-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-lwriRAHm1Yg4iDf23Oxm9n/t5Zpw1lVnxYU3HnJPTi2lJRkKTrps1KVgvL6m7WvmhYVt/FIsssWay+k45QHeuw==",
|
||||||
|
"cpu": [
|
||||||
|
"x64"
|
||||||
|
],
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"darwin"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"node_modules/@msgpackr-extract/msgpackr-extract-linux-arm": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-linux-arm/-/msgpackr-extract-linux-arm-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-MOI9Dlfrpi2Cuc7i5dXdxPbFIgbDBGgKR5F2yWEa6FVEtSWncfVNKW5AKjImAQ6CZlBK9tympdsZJ2xThBiWWA==",
|
||||||
|
"cpu": [
|
||||||
|
"arm"
|
||||||
|
],
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"linux"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"node_modules/@msgpackr-extract/msgpackr-extract-linux-arm64": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-linux-arm64/-/msgpackr-extract-linux-arm64-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-FU20Bo66/f7He9Fp9sP2zaJ1Q8L9uLPZQDub/WlUip78JlPeMbVL8546HbZfcW9LNciEXc8d+tThSJjSC+tmsg==",
|
||||||
|
"cpu": [
|
||||||
|
"arm64"
|
||||||
|
],
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"linux"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"node_modules/@msgpackr-extract/msgpackr-extract-linux-x64": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-linux-x64/-/msgpackr-extract-linux-x64-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-gsWNDCklNy7Ajk0vBBf9jEx04RUxuDQfBse918Ww+Qb9HCPoGzS+XJTLe96iN3BVK7grnLiYghP/M4L8VsaHeA==",
|
||||||
|
"cpu": [
|
||||||
|
"x64"
|
||||||
|
],
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"linux"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"node_modules/@msgpackr-extract/msgpackr-extract-win32-x64": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/@msgpackr-extract/msgpackr-extract-win32-x64/-/msgpackr-extract-win32-x64-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-O+6Gs8UeDbyFpbSh2CPEz/UOrrdWPTBYNblZK5CxxLisYt4kGX3Sc+czffFonyjiGSq3jWLwJS/CCJc7tBr4sQ==",
|
||||||
|
"cpu": [
|
||||||
|
"x64"
|
||||||
|
],
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"win32"
|
||||||
|
]
|
||||||
|
},
|
||||||
"node_modules/@next/env": {
|
"node_modules/@next/env": {
|
||||||
"version": "14.0.3",
|
"version": "14.0.3",
|
||||||
"resolved": "https://registry.npmjs.org/@next/env/-/env-14.0.3.tgz",
|
"resolved": "https://registry.npmjs.org/@next/env/-/env-14.0.3.tgz",
|
||||||
@ -18867,6 +18940,35 @@
|
|||||||
"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz",
|
||||||
"integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
|
"integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w=="
|
||||||
},
|
},
|
||||||
|
"node_modules/msgpackr": {
|
||||||
|
"version": "1.10.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/msgpackr/-/msgpackr-1.10.0.tgz",
|
||||||
|
"integrity": "sha512-rVQ5YAQDoZKZLX+h8tNq7FiHrPJoeGHViz3U4wIcykhAEpwF/nH2Vbk8dQxmpX5JavkI8C7pt4bnkJ02ZmRoUw==",
|
||||||
|
"optionalDependencies": {
|
||||||
|
"msgpackr-extract": "^3.0.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/msgpackr-extract": {
|
||||||
|
"version": "3.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/msgpackr-extract/-/msgpackr-extract-3.0.2.tgz",
|
||||||
|
"integrity": "sha512-SdzXp4kD/Qf8agZ9+iTu6eql0m3kWm1A2y1hkpTeVNENutaB0BwHlSvAIaMxwntmRUAUjon2V4L8Z/njd0Ct8A==",
|
||||||
|
"hasInstallScript": true,
|
||||||
|
"optional": true,
|
||||||
|
"dependencies": {
|
||||||
|
"node-gyp-build-optional-packages": "5.0.7"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"download-msgpackr-prebuilds": "bin/download-prebuilds.js"
|
||||||
|
},
|
||||||
|
"optionalDependencies": {
|
||||||
|
"@msgpackr-extract/msgpackr-extract-darwin-arm64": "3.0.2",
|
||||||
|
"@msgpackr-extract/msgpackr-extract-darwin-x64": "3.0.2",
|
||||||
|
"@msgpackr-extract/msgpackr-extract-linux-arm": "3.0.2",
|
||||||
|
"@msgpackr-extract/msgpackr-extract-linux-arm64": "3.0.2",
|
||||||
|
"@msgpackr-extract/msgpackr-extract-linux-x64": "3.0.2",
|
||||||
|
"@msgpackr-extract/msgpackr-extract-win32-x64": "3.0.2"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/multicast-dns": {
|
"node_modules/multicast-dns": {
|
||||||
"version": "7.2.5",
|
"version": "7.2.5",
|
||||||
"resolved": "https://registry.npmjs.org/multicast-dns/-/multicast-dns-7.2.5.tgz",
|
"resolved": "https://registry.npmjs.org/multicast-dns/-/multicast-dns-7.2.5.tgz",
|
||||||
@ -19131,6 +19233,17 @@
|
|||||||
"node-gyp-build-test": "build-test.js"
|
"node-gyp-build-test": "build-test.js"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/node-gyp-build-optional-packages": {
|
||||||
|
"version": "5.0.7",
|
||||||
|
"resolved": "https://registry.npmjs.org/node-gyp-build-optional-packages/-/node-gyp-build-optional-packages-5.0.7.tgz",
|
||||||
|
"integrity": "sha512-YlCCc6Wffkx0kHkmam79GKvDQ6x+QZkMjFGrIMxgFNILFvGSbCp2fCBC55pGTT9gVaz8Na5CLmxt/urtzRv36w==",
|
||||||
|
"optional": true,
|
||||||
|
"bin": {
|
||||||
|
"node-gyp-build-optional-packages": "bin.js",
|
||||||
|
"node-gyp-build-optional-packages-optional": "optional.js",
|
||||||
|
"node-gyp-build-optional-packages-test": "build-test.js"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/node-int64": {
|
"node_modules/node-int64": {
|
||||||
"version": "0.4.0",
|
"version": "0.4.0",
|
||||||
"resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz",
|
"resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz",
|
||||||
|
|||||||
@ -40,6 +40,7 @@
|
|||||||
"http-proxy": "^1.18.1",
|
"http-proxy": "^1.18.1",
|
||||||
"http-proxy-middleware": "^2.0.6",
|
"http-proxy-middleware": "^2.0.6",
|
||||||
"js-base64": "^3.7.5",
|
"js-base64": "^3.7.5",
|
||||||
|
"msgpackr": "^1.10.0",
|
||||||
"next": "^14.0.3",
|
"next": "^14.0.3",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0",
|
"react-dom": "^18.2.0",
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user