
Security News
/Research
Wallet-Draining npm Package Impersonates Nodemailer to Hijack Crypto Transactions
Malicious npm package impersonates Nodemailer and drains wallets by hijacking crypto transactions across multiple blockchains.
@netless/appliance-plugin
Advanced tools
[中文文档](https://github.com/netless-io/fastboard/blob/main/docs/zh/appliance-plugin.md)
This plugin is based on the plugin mechanism of white-web-sdk, and realizes a set of whiteboard teaching AIDS drawing tools. At the same time, it is also based on @netless/window-manager, which can be used on multiple Windows.
appliance-plugin, Depend on white-web-SDK, @netless/window-manager, And based on web API support for offscreenCanvas.
npm install @netless/appliance-plugin
Plug-ins can support two scenarios, their access plug-in names are different:
import { ApplianceMultiPlugin } from "@netless/appliance-plugin";
import { ApplianceSinglePlugin } from "@netless/appliance-plugin";
workerjs file cdn deployment
We used two-worker concurrency to improve drawing efficiency, which improved it by more than 40% over single-thread efficiency. However, the common dependencies on the two worker files are repeated, so building directly into the package will greatly increase the package size. So we allow the workerjs file cdn deployment by simply deploying the file under @netless/appliance-plugin/cdn into the cdn and then configuring the c of the last two workerjs via the second parameter of getInstance in the plug-in, options.cdn The dn address is fine. This solves the problem of excessive package size
- The total package is about 300kB, and the two wokerjs are 600kB each If you need to consider the size of the package you are building, select Configure cdn.
// The method of importing worker.js is optional. If cdn is used, it does not need to be imported from dist. If dist is imported, it needs to be configured into options.cdn in the form of resource module and bolb inline. Such as '?raw', this requires packer support,vite default support '?raw',webpack needs to configure raw-loader or asset/source.
import fullWorkerString from '@netless/appliance-plugin/dist/fullWorker.js?raw';
import subWorkerString from '@netless/appliance-plugin/dist/subWorker.js?raw';
const fullWorkerBlob = new Blob([fullWorkerString], {type: 'text/javascript'});
const fullWorkerUrl = URL.createObjectURL(fullWorkerBlob);
const subWorkerBlob = new Blob([subWorkerString], {type: 'text/javascript'});
const subWorkerUrl = URL.createObjectURL(subWorkerBlob);
// interconnection with fastboard-react
// Full package mode reference
// import { useFastboard, Fastboard } from "@netless/fastboard-react/full";
// Subcontract reference
import { useFastboard, Fastboard } from "@netless/fastboard-react";
const app = useFastboard(() => ({
sdkConfig: {
...
},
joinRoom: {
...
},
managerConfig: {
cursor: true,
enableAppliancePlugin: true,
...
},
enableAppliancePlugin: {
cdn: {
fullWorkerUrl,
subWorkerUrl,
}
...
}
}));
// interconnection with fastboard
// Full package mode reference
// import { createFastboard, createUI } from "@netless/fastboard/full";
// Subcontract reference
import { createFastboard, createUI } from "@netless/fastboard";
const fastboard = await createFastboard({
sdkConfig: {
...
},
joinRoom: {
...
},
managerConfig: {
cursor: true,
supportAppliancePlugin: true,
...
},
enableAppliancePlugin: {
cdn: {
fullWorkerUrl,
subWorkerUrl,
}
...
}
});
import '@netless/window-manager/dist/style.css';
import '@netless/appliance-plugin/dist/style.css';
import { WhiteWebSdk } from "white-web-sdk";
import { WindowManager } from "@netless/window-manager";
import { ApplianceMultiPlugin } from '@netless/appliance-plugin';
// The method of importing worker.js is optional. If cdn is used, it does not need to be imported from dist. If dist is imported, it needs to be configured into options.cdn in the form of resource module and bolb inline. Such as '?raw', this requires packer support,vite default support '?raw',webpack needs to configure raw-loader or asset/source.
import fullWorkerString from '@netless/appliance-plugin/dist/fullWorker.js?raw';
import subWorkerString from '@netless/appliance-plugin/dist/subWorker.js?raw';
const fullWorkerBlob = new Blob([fullWorkerString], {type: 'text/javascript'});
const fullWorkerUrl = URL.createObjectURL(fullWorkerBlob);
const subWorkerBlob = new Blob([subWorkerString], {type: 'text/javascript'});
const subWorkerUrl = URL.createObjectURL(subWorkerBlob);
const whiteWebSdk = new WhiteWebSdk(...)
const room = await whiteWebSdk.joinRoom({
...
invisiblePlugins: [WindowManager, ApplianceMultiPlugin],
useMultiViews: true,
})
const manager = await WindowManager.mount({ room , container:elm, chessboard: true, cursor: true, supportAppliancePlugin: true});
if (manager) {
// await manager.switchMainViewToWriter();
await ApplianceMultiPlugin.getInstance(manager,
{
options: {
cdn: {
fullWorkerUrl,
subWorkerUrl,
}
}
}
);
}
Note the css file
import '@netless/appliance-plugin/dist/style.css'
needs to be imported into the project;
import { WhiteWebSdk } from "white-web-sdk";
import { ApplianceSinglePlugin, ApplianceSigleWrapper } from '@netless/appliance-plugin';
// The method of importing worker.js is optional. If cdn is used, it does not need to be imported from dist. If dist is imported, it needs to be configured into options.cdn in the form of resource module and bolb inline. Such as '?raw', this requires packer support,vite default support '?raw',webpack needs to configure raw-loader or asset/source.
import fullWorkerString from '@netless/appliance-plugin/dist/fullWorker.js?raw';
import subWorkerString from '@netless/appliance-plugin/dist/subWorker.js?raw';
const fullWorkerBlob = new Blob([fullWorkerString], {type: 'text/javascript'});
const fullWorkerUrl = URL.createObjectURL(fullWorkerBlob);
const subWorkerBlob = new Blob([subWorkerString], {type: 'text/javascript'});
const subWorkerUrl = URL.createObjectURL(subWorkerBlob);
const whiteWebSdk = new WhiteWebSdk(...)
const room = await whiteWebSdk.joinRoom({
...
invisiblePlugins: [ApplianceSinglePlugin],
wrappedComponents: [ApplianceSigleWrapper]
})
await ApplianceSinglePlugin.getInstance(room,
{
options: {
cdn: {
fullWorkerUrl,
subWorkerUrl,
}
}
}
);
Note the css file
import '@netless/appliance-plugin/dist/style.css'
needs to be imported into the project;
module: {
rules: [
// ...
{
test: /\.m?js$/,
resourceQuery: { not: [/raw/] },
use: [ ... ]
},
{
resourceQuery: /raw/,
type: 'asset/source',
}
]
},
The plugin re-implements some of the interfaces of the same name on room or Windows Manager, but internally we have re-injected them back into the original object via injectMethodToObject. No changes are required for external users. As follows:
// Internal hack
injectMethodToObject(windowmanager, "undo");
injectMethodToObject(windowmanager, "redo");
injectMethodToObject(windowmanager, "cleanCurrentScene");
injectMethodToObject(windowmanager, "insertImage");
injectMethodToObject(windowmanager, "completeImageUpload");
injectMethodToObject(windowmanager, "lockImage");
injectMethodToObject(room, "getImagesInformation");
injectMethodToObject(room, "callbacks");
injectMethodToObject(room, "screenshotToCanvasAsync");
injectMethodToObject(room, "getBoundingRectAsync");
injectMethodToObject(room, "scenePreviewAsync");
injectMethodToObject(windowmanager.mainView, "setMemberState");
// These we can see the call behavior through the front-end log, for example:
// [ApplianceMultiPlugin] setMemberState
// [ApplianceMultiPlugin] cleanCurrentScene
The following interfaces are involved:
setMemberState
undo
redo
callbacks
insertImage
lockImage
completeImageUpload
getImagesInformation
cleanCurrentScene
cleanCurrentScene
setMemberState
undo
redo
callbacks
insertImage
lockImage
completeImageUpload
getImagesInformation
cleanCurrentScene
getBoundingRectAsync
Replace the api room.getBoundingRect
screenshotToCanvasAsync
Replace the api room.screenshotToCanvas
scenePreviewAsync
Replace the api room.scenePreview
destroy
Destroy the instance of appliance-plugin
addListener
add appliance plugin Listener
removeListener
remove appliance plugin Listener
disableDeviceInputs
Replace the api room.disableDeviceInputs
disableEraseImage
Replace the api room.disableEraseImage
This method only suppert when currentApplianceName is eraser
disableCameraTransform
Replace the api room.disableCameraTransform
5.Incompatible
exportScene
When the appliance-plugin is enabled, notes cannot be exported in room mode
Server-side screenshot, after the appliance-plugin is turned on, notes cannot be obtained by calling server-side screenshot, but need to use screenshotToCanvasAsync
to obtain the screenshot
laserPen teaching aids (Version >=1.1.1)
import { EStrokeType, ApplianceNames } from "@netless/appliance-plugin";
room.setMemberState({
currentApplianceName: ApplianceNames.laserPen,
strokeType: EStrokeType.Normal,
});
Extended Teaching AIDS (Version >=1.1.1)
export enum EStrokeType {
/** Solid line */
Normal = 'Normal',
/** Line with pen edge */
Stroke = 'Stroke',
/** Dotted line */
Dotted = 'Dotted',
/** Long dotted line */
LongDotted = 'LongDotted'
};
export type ExtendMemberState = {
/** The teaching AIDS selected by the current user */
currentApplianceName: ApplianceNames;
/** Whether to open the pen tip */
strokeType? : EStrokeType;
/** Whether to delete the entire line segment */
isLine? : boolean;
/** Wireframe transparency */
strokeOpacity? : number;
/** Whether to turn on laser pointer */
useLaserPen? : boolean;
/** Laser pointer holding time, second */
duration? : number;
/** Fill style */
fillColor? : Color;
/** Fill transparency */
fillOpacity? : number;
/** The specific type of graph to draw when using shape */
shapeType? : ShapeType;
/** Number of polygon vertices */
vertices? :number;
/** Length of the inner vertex of the polygon */
innerVerticeStep? :number;
/** Ratio of the radius of the inner vertex of the polygon to the outer vertex */
innerRatio? : number;
/** Text transparency */
textOpacity? : number;
/** Text background color */
textBgColor? : Color;
/** Text background color transparency */
textBgOpacity? : number;
/** Location */
placement? : SpeechBalloonPlacement;
};
import { ExtendMemberState, ApplianceNames } from '@netless/appliance-plugin';
/** Set the state of teaching AIDS */
room.setMemberState({ ... } as ExtendMemberState);
manager.mainView.setMemberState({ ... } as ExtendMemberState);
appliance.setMemberState({ ... } as ExtendMemberState);
// Solid line
setMemberState({ strokeType: EStrokeType.Normal });
// Line with pen edge
setMemberState({ strokeType: EStrokeType.Stroke });
// Dotted line
setMemberState({ strokeType: EStrokeType.Dotted });
// Long dotted line
setMemberState({ strokeType: EStrokeType.LongDotted });
setMemberState({ strokeOpacity: 0.5 });
setMemberState({
textOpacity: 0.5,
textBgOpacity: 0.5,
textBgColor: [0, 0, 0],
});
setMemberState({ fillOpacity: 0.5, fillColor: [0, 0, 0] });
// regular pentagon
setMemberState({
currentApplianceName: ApplianceNames.shape,
shapeType: ShapeType.Polygon,
vertices: 5,
});
// fat hexagonal star
setMemberState({
currentApplianceName: ApplianceNames.shape,
shapeType: ShapeType.Star,
vertices: 12,
innerVerticeStep: 2,
innerRatio: 0.8,
});
// The dialog box in the lower left corner
setMemberState({
currentApplianceName: ApplianceNames.shape,
shapeType: ShapeType.SpeechBalloon,
placement: "bottomLeft",
});
Split screen display Elements (little whiteboard featrue), need to combine @netless/app-little-white-board
(Version >=1.1.3)
Minimap function (Version >=1.1.6)
/** Create a minimap
* @param viewId ID of the whiteboard under windowManager. The ID of the main whiteboard is mainView, and the ID of other whiteboards is the appID of addApp() return
* @param div Small map DOM container
*/
createMiniMap(viewId: string, div: HTMLElement): Promise<void>;
/** Destroy minimap */
destroyMiniMap(viewId: string): Promise<void>;
Filter Elements (Version >=1.1.6) `js /** Filter Elements
getInstance(wm: WindowManager, adaptor: ApplianceAdaptor)
options: AppliancePluginOptions
; The cdn addresses of both workers must be configured.
export type AppliancePluginOptions = {
/** cdn Configuration item */
cdn: CdnOpt;
/** Extended configuration items */
extras?: ExtrasOptions;
}
cursorAdapter? : CursorAdapter
; This parameter is optional. In single whiteboard mode, customize the mouse style.logger?: Logger
; This parameter is optional. Configure the log printer object. The default output is on the local console. If logs need to be uploaded to the specified server, you need to manually configure the configuration.
If you need to upload the log to the whiteboard log server, configure the
room.logger
to this item。
During the interconnection process, if you want to understand and track the internal status of the plug-in, you can view the internal data through the following console commands.
const applianPlugin = await ApplianceSinglePlugin.getInstance(...)
applianPlugin.CurrentManager // can see the package version number, internal state, etc
applianPlugin.CurrentManager.ConsoleWorkerInfo () // can check information to draw on the worker
FAQs
[中文文档](https://github.com/netless-io/fastboard/blob/main/docs/zh/appliance-plugin.md)
We found that @netless/appliance-plugin demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 3 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
/Research
Malicious npm package impersonates Nodemailer and drains wallets by hijacking crypto transactions across multiple blockchains.
Security News
This episode explores the hard problem of reachability analysis, from static analysis limits to handling dynamic languages and massive dependency trees.
Security News
/Research
Malicious Nx npm versions stole secrets and wallet info using AI CLI tools; Socket’s AI scanner detected the supply chain attack and flagged the malware.