Security News
Research
Data Theft Repackaged: A Case Study in Malicious Wrapper Packages on npm
The Socket Research Team breaks down a malicious wrapper package that uses obfuscation to harvest credentials and exfiltrate sensitive data.
JS Stacks are different from queues because they are LIFO (last in first out) unlike a queue which is FIFO (first in first out). While a Queue executes in order, a stack executes whatever was most recently added to the stack, much like a reading a stack of papers on your desk. If you read half the papers and someone puts more on the top you start with the new ones first.
Works great in node.js, webpack, browserify, or any other commonjs loader or compiler. To use in plain old vanilla browser javascript without common js just replace the requires in the examples with script tags. We show that below too. Any time you need a JS stack easy-stack is there for you.
require('easy-stack');
for ES6 node.
require('easy-stack/es5.js');
for ES5 node and browser.
npm install easy-stack
npm info : See npm trends and stats for easy-stack
GitHub info :
Package details websites :
This work is licenced via the DBAD Public Licence.
key | type | parameters | default | description |
---|---|---|---|---|
add | function | any number of functions | adds all parameter functions to stack and starts execution if autoRun is true, stack is not already running and stack is not forcibly stopped | |
next | function | executes next item in stack if stack is not forcibly stopped | ||
clear | function | removes remaining items in the stack | ||
contents | Array | stack instance contents | ||
autoRun | Bool | true | should autoRun stack when new item added | |
stop | Bool | false | setting this to true will forcibly prevent the stack from executing |
var Stack=require('easy-stack');
//create a new Stack instance
var stack=new Stack;
for(var i=0; i<50; i++){
//add a bunch of stuff to the stack
stack.add(makeRequest);
}
function makeRequest(){
//do stuff
console.log('making some request');
this.next();
}
The only difference is including via a script tag instead of using require.
<html>
<head>
<!-- this is the only difference -->
<script src='./es5.js'></script>
<script>
console.log('my awesome app script');
var stack=new Stack;
for(var i=0; i<50; i++){
stack.add(makeRequest);
}
function makeRequest(){
console.log('making some request');
this.next();
}
</script>
</head>
<body>
</body>
</html>
This allows you to start adding requests immediately and only execute if the websocket is connected. To use in plain browser based JS without webpack or browserify just replace the requires with the script tag.
var Stack=require('easy-stack');
//ws-share just makes it easier to share websocket code and ensure you don't open a websocket more than once
var WS=require('ws-share');
//js-message makes it easy to create and parse normalized JSON messages.
var Message=require('js-message');
//create a new Stack instance
var stack=new Stack;
//force stop until websocket opened
stack.stop=true;
var ws=null;
function startWS(){
//websocket.org rocks
ws=new WS('wss://echo.websocket.org/?encoding=text');
ws.on(
'open',
function(){
ws.on(
'message',
handleResponse
);
//now that websocket is opened allow auto execution
stack.stop=false;
stack.next();
}
);
ws.on(
'error',
function(err){
//stop execution of stack if there is an error because the websocket is likely closed
stack.stop=true;
//remove remaining items in the stack
stack.clear();
throw(err);
}
);
ws.on(
'close',
function(){
//stop execution of stack when the websocket closed
stack.stop=true;
}
);
}
//simulate a lot of requests being stackd up for the websocket
for(var i=0; i<50; i++){
stack.add(makeRequest);
}
var messageID=0;
function handleResponse(e){
var message=new Message;
message.load(e.data);
console.log(message.type,message.data);
}
function makeRequest(){
messageID++;
var message=new Message;
message.type='testMessage';
message.data=messageID;
ws.send(message.JSON);
this.next();
}
startWS();
const Stack=require('easy-stack');
class MyAwesomestack extends Stack{
isStopped(){
return this.stop;
}
removeThirdItem(){
this.contents.splice(2,1);
return this.contents;
}
};
var Stack=require('easy-stack');
//MyAwesomestack inherits from stack
MyAwesomestack.prototype = new Stack;
//Constructor will extend stack
MyAwesomestack.prototype.constructor = MyAwesomestack;
function MyAwesomestack(){
//extend with some stuff your app needs,
//maybe npm publish your extention with easy-stack as a dependancy?
Object.defineProperties(
this,
{
isStopped:{
enumerable:true,
get:checkStopped,
set:checkStopped
},
removeThirdItem:{
enumerable:true,
writable:false,
value:removeThird
}
}
);
//enforce Object.assign for extending by locking down Class structure
//no willy nilly cowboy coding
Object.seal(this);
function checkStopped(){
return this.stop;
}
function removeThird(){
//get the stack content
var list=this.contents;
//modify the stack content
list.splice(2,1);
//save the modified stack content
this.contents=list;
return this.contents;
}
}
FAQs
Simple JS stack with auto run for node and browsers
The npm package easy-stack receives a total of 826,347 weekly downloads. As such, easy-stack popularity was classified as popular.
We found that easy-stack demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer 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
The Socket Research Team breaks down a malicious wrapper package that uses obfuscation to harvest credentials and exfiltrate sensitive data.
Research
Security News
Attackers used a malicious npm package typosquatting a popular ESLint plugin to steal sensitive data, execute commands, and exploit developer systems.
Security News
The Ultralytics' PyPI Package was compromised four times in one weekend through GitHub Actions cache poisoning and failure to rotate previously compromised API tokens.