Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

libkernel

Package Overview
Dependencies
Maintainers
1
Versions
127
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

libkernel - npm Package Compare versions

Comparing version 0.0.49 to 0.0.50

2

dist/index.d.ts

@@ -1,2 +0,2 @@

export { init, postKernelQuery } from './init';
export { init, newKernelQuery } from './init';
export { testMessage, callModule, upload, padAndEncrypt } from './messages';

@@ -1,2 +0,2 @@

export { init, postKernelQuery } from './init';
export { init, newKernelQuery } from './init';
export { testMessage, callModule, upload, padAndEncrypt } from './messages';
export declare function log(...inputs: any): void;
export declare function logErr(...inputs: any): void;
export declare function postKernelQuery(kernelQuery: any): Promise<any>;
export declare function composeErr(...inputs: any): string;
export declare function newKernelQuery(data: any, update: Function): [Function, Promise<any>];
export declare function init(): Promise<string>;

@@ -10,34 +10,42 @@ // log provides a wrapper for console.log that prefixes 'libkernel'.

}
var initialized;
var bridgeExists;
var bridgeAvailable;
var blockForBridge = new Promise((resolve, reject) => {
bridgeAvailable = { resolve, reject };
});
// Establish a system for matching messages to the kernel with responses from
// the kernel. The nonce is incremented every time a new message is sent, and
// the queries object is used as a hashmap that maps a given message nonce to
// the {resolve, reject} of a promise that will be resolved/rejected when a
// response to the corresponding message is provided.
// composeErr takes a series of inputs and composes them into a single string.
// Each element will be separated by a newline. If the input is not a string,
// it will be transformed into a string with JSON.stringify.
//
// Any object that cannot be stringified will be skipped, though an error will
// be logged.
export function composeErr(...inputs) {
let result = "";
for (let i = 0; i < inputs.length; i++) {
// Prepend a newline if this isn't the first element.
if (i !== 0) {
result += "\n";
}
// Strings can be added without modification.
if (typeof inputs[i] === "string") {
result += inputs[i];
continue;
}
// Everything else needs to be stringified, log an error if it
// fails.
try {
let str = JSON.stringify(inputs[i]);
result += str;
}
catch (_a) {
logErr("unable to stringify input to composeErr");
}
}
return result;
}
// Establish a hashmap for matching queries to their responses by their nonces.
// nextNonce needs to start at '1' because '0' is reserved for the bridgeTest
// method performed at init.
const namespace = "libkernel-v0";
var nextNonce = 1;
var queries = new Object();
// postKernelQuery will send a postMessage to the kernel, handling details like
// the nonce and the resolve/reject upon receiving a response. The inputs are a
// resolve and reject function of a promise that should be resolved when the
// response is received, and the message that is going to the kernel itself.
export function postKernelQuery(kernelQuery) {
return new Promise((resolve, reject) => {
let nonce = nextNonce;
nextNonce++;
queries[nonce] = { resolve, reject };
window.postMessage({
method: "kernelQuery",
nonce,
kernelQuery,
}, window.location.origin);
});
}
// handleBridgeResponse will handle a response from the bridge indicating that
// the bridge is working.
function handleBridgeResponse(data) {
// the bridge is working. This needs to be declared before the remaining bridge
// variables because they need to reference it as a handler.
function handleBridgeResponse(resolve, reject, data) {
// Check whether the timeout for the bridge has already fired. If so,

@@ -47,2 +55,3 @@ // log that the bridge is available but late.

logErr("received late signal from bridge");
reject("received late signal from bridge");
return;

@@ -52,69 +61,119 @@ }

// Check whether the version is available in the data.
if (!("version" in data)) {
bridgeAvailable.resolve("bridge did not report a version");
if ("version" in data) {
resolve(data.version);
}
else {
bridgeAvailable.resolve(data.version);
resolve("bridge did not report a version");
}
}
// Establish a system to test if the bridge script is running. bridgeExists is
// a boolean which establishes whether or not we have already determinied if
// the bridge eixsts, bridgeAvailable is the {resolve, reject} object for a
// promise, and the blockForBridge will resolve/reject when we have determined
// if the bridge exists.
//
// The init() function will use a timeout to decide that the bridge does not
// exist, the hanelMessage function will look for a method called
// "bridgeTestResponse" to determine that the bridge does exist. The init
// script needs to send the bridge a test message so the bridge knows to
// respond.
var initialized;
var bridgeExists;
var bridgeAvailable;
var blockForBridge = new Promise((resolve, reject) => {
bridgeAvailable = { resolve, reject, update: null, handle: handleBridgeResponse };
});
// newKernelQuery will send a postMessage to the kernel, handling details like
// the nonce and the resolve/reject upon receiving a response.
//
// The first return value is a function that can be called to send a
// 'queryUpdate' to the kernel for that nonce.
export function newKernelQuery(data, update) {
let nonce = nextNonce;
nextNonce++;
let sendUpdate = function (data) {
queryUpdate(nonce, data);
};
let p = new Promise((resolve, reject) => {
queries[nonce] = { resolve, reject, update, handle: handleKernelResponse };
window.postMessage({
namespace,
method: "newKernelQuery",
nonce,
data,
});
});
return [sendUpdate, p];
}
// queryUpdate is a function that can be called to send a queryUpdate to an
// existing query.
function queryUpdate(nonce, data) {
window.postMessage({
namespace,
method: "queryUpdate",
nonce,
data,
});
}
// handleKernelResponse will parse the kernel's response from the bridge and
// resolve/reject the promise associated with the nonce.
function handleKernelResponse(event) {
// Check that we have a promise for the provided nonce.
if (!(event.data.nonce in queries)) {
logErr("nonce of kernelResponse not found\n", event, "\n", queries);
function handleKernelResponse(resolve, reject, data) {
resolve(data);
}
// handleMessage will handle a message from the kernel, using the response to
// resolve the appropriate promise in the set of queries.
function handleMessage(event) {
// Check the message source.
if (event.source !== window) {
return;
}
let result = queries[event.data.nonce];
delete queries[event.data.nonce];
// Check the status and then resolve or reject accordingly.
if (!("response" in event.data) || !("queryStatus" in event.data.response)) {
logErr("malformed kernel response\n", event);
// Check that this message is a response targeting libkernel.
if (event.data.namespace !== namespace) {
return;
}
if (event.data.response.queryStatus === "resolve") {
result.resolve(event.data.response);
if (!("method" in event.data) || typeof event.data.method !== "string") {
logErr("received message targeting our namespace with malformed method", event.data);
return;
}
else if (event.data.response.queryStatus === "reject") {
result.reject(event.data.response);
}
else {
logErr("malformed queryStatus");
}
}
// handleKernelResponseErr is a special handler for situations where the
// content script was unable to communicate with the background script.
function handleKernelResponseErr(event) {
let reject = queries[event.data.nonce];
delete queries[event.data.nonce];
if (!("err" in event.data) || typeof event.data.err !== "string") {
logErr("malformed error received from bridge");
if (event.data.method !== "response" && event.data.method !== "responseUpdate") {
// We don't log here because that would catch outbound messages
// from ourself.
return;
}
logErr(event.data.err);
reject(event.data.err);
}
// handleMessage will handle a message from the kernel, using the response to
// resolve the appropriate promise in the set of queries.
function handleMessage(event) {
// Check the message source.
if (event.source !== window) {
// Check that we have a nonce for this message.
if (!(event.data.nonce in queries)) {
logErr("message received with no matching nonce\n", event.data, "\n", queries);
return;
}
// Check that this message is a kernelResponse.
if (!("data" in event) || !("method" in event.data)) {
// If this is a responseUpdate, pass the data to the update handler.
if (event.data.method === "responseUpdate") {
let handler = queries[event.data.nonce];
if (!("update" in handler) || typeof handler.update !== "function") {
logErr("responseUpdate received, but no update method defined in handler");
return;
}
if (!("data" in event.data)) {
logErr("responseUpdate received, but no data provided: " + JSON.stringify(event.data));
return;
}
handler.update(event.data.data);
return;
}
if (event.data.method === "bridgeTestResponse") {
handleBridgeResponse(event.data);
// The method is "response", meaning the query is closed out can can be
// deleted.
let handler = queries[event.data.nonce];
delete queries[event.data.nonce];
if (!("err" in event.data)) {
handler.reject("no err field provided in response: " + JSON.stringify(event.data));
return;
}
if (event.data.method === "kernelResponse") {
handleKernelResponse(event);
if (event.data.err !== null) {
handler.reject(event.data.err);
return;
}
if (event.data.method === "kernelResponseErr") {
handleKernelResponseErr(event);
if (!("data" in event.data)) {
handler.reject("no data field provided in query: " + JSON.stringify(event.data));
return;
}
handler.handle(handler.resolve, handler.reject, event.data.data);
}

@@ -133,9 +192,10 @@ // init will add an event listener for messages from the kernel bridge. It is

window.addEventListener("message", handleMessage);
// Send a message checking if the bridge is alive and responding. We
// use a fake nonce because we don't care about the nonce for the
// bridge.
// Send a message checking if the bridge is alive and responding. The
// nonce '0' is kept available explicitly for this purpose.
window.postMessage({
method: "bridgeTestQuery",
namespace,
nonce: 0,
method: "test",
});
queries[0] = bridgeAvailable;
// After 2 seconds, check whether the bridge has responded. If not,

@@ -146,3 +206,3 @@ // fail the bridge.

bridgeExists = false;
bridgeAvailable.reject(new Error("bridge unavailable, need skynet extension"));
bridgeAvailable.reject("bridge unavailable, need skynet extension");
logErr("bridge did not respond after 2 seconds");

@@ -149,0 +209,0 @@ }

export declare function testMessage(): Promise<string>;
export declare function callModule(module: string, moduleMethod: string, moduleInput: any): Promise<any>;
export declare function callModule(module: string, method: string, data: any): Promise<any>;
export declare function upload(filename: string, fileData: Uint8Array): Promise<string>;
export declare function padAndEncrypt(filepath: string, fileData: Uint8Array): Promise<string>;

@@ -1,2 +0,3 @@

import { logErr, init, postKernelQuery } from './init';
import { logErr, composeErr, init, newKernelQuery } from './init';
const noBridge = "the bridge failed to initialize (do you have the Skynet browser extension?)";
// testMessage will send a test message to the kernel, ensuring that basic

@@ -17,12 +18,27 @@ // kernel communications are working. The promise will resolve to the version

.then(x => {
// Send a 'requestTest' message to the kernel. The
// request test message uniquely doesn't have any other
// parameters.
postKernelQuery({
kernelMethod: "requestTest",
})
// We use nested promises instead of promise chaining
// because promise chaining didn't provide enough
// control over handling the error.
.then(response => {
// Send a 'test' message to the kernel, which is a
// method with no parameters.
//
// The first return value of newKernelQuery is ignored
// because it is an update function that we can call if
// we wish to provide new information to the query via
// a 'queryUpdate'. The 'test' method does not support
// any queryUpdates.
//
// The second input of newKernelQuery is passed as
// null, it's usually a handler function to accept
// 'responseUpdate' messages from the kernel related to
// the query. The 'test' method doesn't have any
// 'responseUpdates', so there is no need to create an
// updateHandler.
let [_, query] = newKernelQuery({
method: "test",
}, null);
// We use nested promises instead of promise chaining
// because promise chaining didn't provide enough
// control over handling the error. We like wrapping
// our errors to help indicate exactly which part of
// the code has gone wrong, and that nuance gets lost
// with promise chaining.
query.then(response => {
if (!("version" in response)) {

@@ -34,9 +50,5 @@ resolve("kernel did not report a version");

})
.catch(response => {
if (!("err" in response) || typeof response.err !== "string") {
logErr("unrecognized response in postKernelQuery catch", response);
reject("unrecognized repsonse");
return;
}
reject(response.err);
.catch(err => {
let cErr = composeErr("newKernelQuery failed", err);
reject(cErr);
});

@@ -48,4 +60,5 @@ })

// the browser extension.
logErr("bridge is not initialized:", err);
reject(err);
let cErr = composeErr(noBridge, err);
logErr(cErr);
reject(cErr);
});

@@ -57,21 +70,30 @@ });

// handled for the user.
export function callModule(module, moduleMethod, moduleInput) {
//
// callModule can only be used for query-response communications, there is no
// support for sending queryUpdate messages or receiving responseUpdate
// messages. If you need those, use 'connectModule' instead.
export function callModule(module, method, data) {
return new Promise((resolve, reject) => {
init()
.then(x => {
return postKernelQuery({
kernelMethod: "moduleCall",
module,
moduleMethod,
moduleInput,
})
.then(response => {
let [_, query] = newKernelQuery({
method: "moduleCall",
data: {
module,
method,
data,
}
}, null);
query.then(response => {
resolve(response);
})
.catch(response => {
reject(response);
.catch(err => {
let cErr = composeErr("moduleCall query to kernel failed", err);
reject(err);
});
})
.catch(err => {
reject(err);
let cErr = composeErr(noBridge, err);
logErr(cErr);
reject(cErr);
});

@@ -93,16 +115,18 @@ });

.then(x => {
return postKernelQuery({
kernelMethod: "moduleCall",
module: "AQCS3RHbDlk00IdICFEI1rKZp-VNsnsKWC0n7K-taoAuog",
moduleMethod: "secureUpload",
moduleInput: {
filename,
fileData,
let [_, query] = newKernelQuery({
method: "moduleCall",
data: {
module: "AQD1kFeJJhRnkgWGD-ws6V1QITQrHd2WX5pQnU78MM_o3Q",
method: "secureUpload",
data: {
filename,
fileData,
},
},
})
.then(response => {
}, null);
query.then(response => {
resolve(response.output);
})
.catch(response => {
reject(response.err);
.catch(err => {
reject(err);
});

@@ -126,3 +150,3 @@ })

.then(x => {
postKernelQuery({
let [_, query] = newKernelQuery({
kernelMethod: "moduleCall",

@@ -135,4 +159,4 @@ module: "AQAs00kS6OKUd-FIWj9qdJLArCiEDMVgYBSkaetuTF-MsQ",

},
})
.then(response => {
}, null);
query.then(response => {
resolve(response.output);

@@ -139,0 +163,0 @@ })

{
"name": "libkernel",
"version": "0.0.49",
"version": "0.0.50",
"description": "helper library to interact with the skynet kernel",

@@ -5,0 +5,0 @@ "main": "dist/index.js",

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc