node-github
Advanced tools
Comparing version 0.0.3 to 0.0.4
530
index.js
@@ -1,529 +0,1 @@ | ||
/** | ||
* Copyright 2012 Cloud9 IDE, Inc. | ||
* | ||
* This product includes software developed by | ||
* Cloud9 IDE, Inc (http://c9.io). | ||
* | ||
* Author: Mike de Boer <mike@c9.io> | ||
*/ | ||
"use strict"; | ||
var error = require("./error"); | ||
var Util = require("./util"); | ||
/** section: github | ||
* class Client | ||
* | ||
* [[Client]] can load any version of the [[github]] client API, with the | ||
* requirement that a valid routes.json definition file is present in the | ||
* `api/[VERSION]/client` directory and that the routes found in this file are | ||
* implemented as well. | ||
* | ||
* Upon instantiation of the [[Client]] class, the routes.json file is loaded | ||
* from the API version specified in the configuration and, parsed and from it | ||
* the routes for HTTP requests are extracted. For each HTTP endpoint to the | ||
* HTTP server, a method is generated which accepts a Javascript Object | ||
* with parameters and an optional callback to be invoked when the API request | ||
* returns from the server or when the parameters could not be validated. | ||
* | ||
* When an HTTP endpoint is processed and a method is generated as described | ||
* above, [[Client]] also sets up parameter validation with the rules as | ||
* defined in the routes.json. A full example that illustrates how this works | ||
* is shown below: | ||
* | ||
* ##### Example | ||
* | ||
* First, we look at a listing of a sample routes.json routes definition file: | ||
* | ||
* { | ||
* "incoming": { | ||
* "room": { | ||
* "addmember": { | ||
* "url": "/api/room/addmember", | ||
* "method": "POST", | ||
* "params": { | ||
* "roomid": { | ||
* "type": "String", | ||
* "required": true, | ||
* "validation": "^[0-9a-z]{1}[0-9a-z-_\\|\\/]{1,61}$", | ||
* "invalidmsg": "Room name invalid. Please only use letters, numbers, underscores (_), dashes (-), pipes (|), slashes (/) and more than one character", | ||
* "description": "Name of the chat room" | ||
* }, | ||
* "uid": { | ||
* "type": "Number", | ||
* "required": true, | ||
* "validation": "^[0-9]+$", | ||
* "invalidmsg": "Invalid User ID encountered", | ||
* "description": "A uid is a numeric User ID, with which the user can be retrieved from the database" | ||
* }, | ||
* "workspaceId": { | ||
* "type": "String", | ||
* "required": true, | ||
* "validation": "", | ||
* "invalidmessage": "workspaceId is a required field", | ||
* "description": "Identifier of the active workspace. Example: 'user/bob/node_chat'" | ||
* } | ||
* } | ||
* } | ||
* }, | ||
* | ||
* "outgoing": { | ||
* "room": { | ||
* "addmessage": { | ||
* "url": "/api/room/addmessage", | ||
* "params": { | ||
* "roomid": { | ||
* "type": "String", | ||
* "required": true, | ||
* "validation": "^[0-9a-z]{1}[0-9a-z-_\\|\\/]{1,61}$", | ||
* "invalidmsg": "Room name invalid. Please only use letters, numbers, underscores (_), dashes (-), pipes (|), slashes (/) and more than one character", | ||
* "description": "Name of the chat room" | ||
* }, | ||
* "from": { | ||
* "type": "Number", | ||
* "required": true, | ||
* "validation": "^[0-9]+$", | ||
* "invalidmsg": "Invalid User ID encountered", | ||
* "description": "from is numeric User ID, with which the user can be retrieved from the database" | ||
* }, | ||
* "content": { | ||
* "type": "String", | ||
* "required": true, | ||
* "validation": "", | ||
* "invalidmsg": "No or empty message, which is not allowed", | ||
* "description": "Content of the message sent" | ||
* } | ||
* } | ||
* } | ||
* } | ||
* } | ||
* } | ||
* | ||
* You probably noticed that the definition is quite verbose and the decision | ||
* for its design was made to be verbose rather than allowing for variables and | ||
* reuse, as is possible with Javascript, but not with JSON definitions. | ||
* | ||
* There are two sections; 'incoming' and 'outgoing'. | ||
* NOTE: if it is deemed more practical or common sense, the terms 'incoming' and | ||
* 'outgoing' may be swapped. This is up for discussion. | ||
* | ||
* The `incoming` section defines the endpoints for calls to the [[CollabServer]]. | ||
* These definitions are parsed and methods are created that the client can call | ||
* to make an HTTP request to the server. | ||
* In this example, there is one endpoint defined: `incoming/room/addmember`. | ||
* This endpoint will be exposed as a member on the [[Client]] object and may | ||
* be invoked with | ||
* | ||
* client["incoming/room/addmember"]({ | ||
* "roomid": "ry/node_chat", | ||
* "uid": "1234", | ||
* "workspaceId": "user/bob/node_chat" | ||
* }, function(err, ret) { | ||
* // do something with the result here. | ||
* }); | ||
* | ||
* The callback is optional. In the case of an error or when the `ret` argument | ||
* holds a value, it will also be broadcasted to all the connected Socket.IO clients | ||
* that are connected to the workspace called `user/bob/node_chat`. | ||
* All the parameters as specified in the Object that is passed to the function | ||
* as first argument, will be validated according to the rules in the `params` | ||
* block of the route definition. | ||
* Thus, in the case of the `roomid` parameter, according to the definition in | ||
* the `params` block, it is a required parameter (needs to hold a value) and its | ||
* value validated with the RegExp `^[0-9a-z]{1}[0-9a-z-_\|\/]{1,61}$`. In other | ||
* words, if the validation requirements are not met, an HTTP error is passed as | ||
* first argument of the callback. | ||
* | ||
* Implementation Notes: the `method` is NOT case sensitive, whereas `url` is. | ||
* The `url` parameter also supports denoting parameters inside it as follows: | ||
* | ||
* "addmember": { | ||
* "url": "/api/room/:roomid/addmember", | ||
* "method": "POST", | ||
* ... | ||
* } | ||
* | ||
* The `outgoing` section defines the endpoints for events that come in through | ||
* [Redis pubsub](http://redis.io/topics/pubsub). | ||
* Each time that the [[CollabServer]] makes a change in the Redis database, this | ||
* may trigger an event to be published to a Redis pubsub channel. [[Client]] | ||
* has subscribed to quite bunch of channels; room, workspace, project and user. | ||
* Whenever a message comes in from one of these channels, [[Client]] first | ||
* checks if an endpoint is registered for that outgoing message. | ||
* In this example, there is one endpoint defined: `outgoing/room/addmessage`. | ||
* This is enpoint will be exposed as a member on the [[Client]] object and may | ||
* be invoked with | ||
* | ||
* client["outgoing/room/addmessage"]({ | ||
* "roomid": "ry/node_chat", | ||
* "from": "1234", | ||
* "content": "my first message!" | ||
* }); | ||
* | ||
* This means that when a message from Redis pubsub channel `room/addmessage` | ||
* arrives, this method is invoked with the JSON passed directly as first argument. | ||
* Important to point out here is that this message is only broadcasted from the | ||
* server when a message was added to a room object in the Redis database; everything | ||
* is persisted in the database, which allows both the server and client | ||
* to operate without keeping any state in memory. | ||
* The endpoint handler in this example then, upon invocation, retrieves all the | ||
* active workspaces that are connected with the room `ry/node_chat` and broadcasts | ||
* the message as-is to all Socket.IO clients that are connected to each workspace. | ||
**/ | ||
var Client = module.exports = function(config) { | ||
this.config = config; | ||
this.debug = Util.isTrue(config.debug); | ||
this.version = config.version; | ||
var cls = require("./api/v" + this.version); | ||
this[this.version] = new cls(this); | ||
this.setupRoutes(); | ||
}; | ||
(function() { | ||
/** | ||
* Client#setupRoutes() -> null | ||
* | ||
* Configures the routes as defined in a routes.json file of an API version | ||
* | ||
* [[Client#setupRoutes]] is invoked by the constructor, takes the | ||
* contents of the JSON document that contains the definitions of all the | ||
* available API routes and iterates over them. | ||
* | ||
* It first recurses through each definition block until it reaches an API | ||
* endpoint. It knows that an endpoint is found when the `url` and `param` | ||
* definitions are found as a direct member of a definition block. | ||
* Then the availability of an implementation by the API is checked; if it's | ||
* not present, this means that a portion of the API as defined in the routes.json | ||
* file is not implemented properly, thus an exception is thrown. | ||
* After this check, a method is attached to the [[Client]] instance | ||
* and becomes available for use. Inside this method, the parameter validation | ||
* and typecasting is done, according to the definition of the parameters in | ||
* the `params` block, upon invocation. | ||
* | ||
* This mechanism ensures that the handlers ALWAYS receive normalized data | ||
* that is of the correct format and type. JSON parameters are parsed, Strings | ||
* are trimmed, Numbers and Floats are casted and checked for NaN after that. | ||
* | ||
* Note: Query escaping for usage with SQL products is something that can be | ||
* implemented additionally by adding an additional parameter type. | ||
**/ | ||
this.setupRoutes = function() { | ||
var self = this; | ||
var api = this[this.version]; | ||
var routes = api.routes; | ||
var defines = routes.defines; | ||
this.constants = defines.constants; | ||
delete routes.defines; | ||
function trim(s) { | ||
if (typeof s != "string") | ||
return s; | ||
return s.replace(/^[\s\t\r\n]+/, "").replace(/[\s\t\r\n]+$/, ""); | ||
} | ||
function parseParams(msg, paramsStruct) { | ||
var params = Object.keys(paramsStruct); | ||
var paramName, def, value, type; | ||
for (var i = 0, l = params.length; i < l; ++i) { | ||
paramName = params[i]; | ||
if (paramName.charAt(0) == "$") { | ||
paramName = paramName.substr(1); | ||
if (!defines.params[paramName]) { | ||
throw new error.BadRequest("Invalid variable parameter name substitution; param '" + | ||
paramName + "' not found in defines block", "fatal"); | ||
} | ||
else | ||
def = defines.params[paramName]; | ||
} | ||
else | ||
def = paramsStruct[paramName]; | ||
value = trim(msg[paramName]); | ||
if (!value) { | ||
// we don't need to validation for undefined parameter values | ||
// that are not required. | ||
if (!def.required) | ||
continue; | ||
throw new error.BadRequest("Empty value for parameter '" + | ||
paramName + "': " + value); | ||
} | ||
// validate the value and type of parameter: | ||
if (def.validation) { | ||
if (!new RegExp(def.validation).test(value)) { | ||
throw new error.BadRequest("Invalid value for parameter '" + | ||
paramName + "': " + value); | ||
} | ||
} | ||
if (def.type) { | ||
type = def.type.toLowerCase(); | ||
if (type == "number") { | ||
value = parseInt(value, 10); | ||
if (isNaN(value)) { | ||
throw new error.BadRequest("Invalid value for parameter '" + | ||
paramName + "': " + msg[paramName] + " is NaN"); | ||
} | ||
} | ||
else if (type == "float") { | ||
value = parseFloat(value); | ||
if (isNaN(value)) { | ||
throw new error.BadRequest("Invalid value for parameter '" + | ||
paramName + "': " + msg[paramName] + " is NaN"); | ||
} | ||
} | ||
else if (type == "json") { | ||
if (typeof value == "string") { | ||
try { | ||
value = JSON.parse(value); | ||
} | ||
catch(ex) { | ||
throw new error.BadRequest("JSON parse error of value for parameter '" + | ||
paramName + "': " + value); | ||
} | ||
} | ||
} | ||
else if (type == "date") { | ||
value = new Date(value); | ||
} | ||
} | ||
msg[paramName] = value; | ||
} | ||
} | ||
function prepareApi(struct, baseType) { | ||
if (!baseType) | ||
baseType = ""; | ||
Object.keys(struct).forEach(function(routePart) { | ||
var block = struct[routePart]; | ||
if (!block) | ||
return; | ||
var messageType = baseType + "/" + routePart; | ||
if (block.url && block.params) { | ||
// we ended up at an API definition part! | ||
var endPoint = messageType.replace(/^[\/]+/g, ""); | ||
var parts = messageType.split("/"); | ||
var section = Util.toCamelCase(parts[1].toLowerCase()); | ||
parts.splice(0, 2); | ||
var funcName = Util.toCamelCase(parts.join("-")); | ||
if (!api[section]) { | ||
throw new Error("Unsupported route section, not implemented in version " + | ||
self.version + " for route '" + endPoint + "' and block: " + | ||
JSON.stringify(block)); | ||
} | ||
if (!api[section][funcName]) { | ||
if (self.debug) | ||
Util.log("Tried to call " + funcName); | ||
throw new Error("Unsupported route, not implemented in version " + | ||
self.version + " for route '" + endPoint + "' and block: " + | ||
JSON.stringify(block)); | ||
} | ||
if (!self[section]) { | ||
self[section] = {}; | ||
// add a utility function 'getFooApi()', which returns the | ||
// section to which functions are attached. | ||
self[Util.toCamelCase("get-" + section + "-api")] = function() { | ||
return self[section]; | ||
}; | ||
} | ||
self[section][funcName] = function(msg, callback) { | ||
try { | ||
parseParams(msg, block.params); | ||
} | ||
catch (ex) { | ||
// when the message was sent to the client, we can | ||
// reply with the error directly. | ||
api.sendError(ex, block, msg, callback); | ||
if (self.debug) | ||
Util.log(ex.message, "fatal"); | ||
// on error, there's no need to continue. | ||
return; | ||
} | ||
api[section][funcName].call(api, msg, block, callback); | ||
}; | ||
} | ||
else { | ||
// recurse into this block next: | ||
prepareApi(block, messageType); | ||
} | ||
}); | ||
} | ||
prepareApi(routes); | ||
}; | ||
this.authenticate = function(options) { | ||
if (!options) { | ||
this.auth = false; | ||
return; | ||
} | ||
if (!options.type || "token|basic|oauth".indexOf(options.type) === -1) | ||
throw new Error("Invalid authentication type, must be 'token', 'basic' or 'oauth'"); | ||
if (options.type == "token" && (!options.username || !options.token)) | ||
throw new Error("Token based authentication requires both a username and token to be set"); | ||
if (options.type == "basic" && (!options.username || !options.password)) | ||
throw new Error("Basic authentication requires both a username and password to be set"); | ||
if (options.type == "oauth" && !options.token) | ||
throw new Error("OAuth2 authentication requires both a token to be set"); | ||
this.auth = options; | ||
}; | ||
function getQuery(msg, def, format) { | ||
var query = format == "json" ? {} : []; | ||
if (!def || !def.params) | ||
return query; | ||
var url = def.url; | ||
Object.keys(def.params).forEach(function(paramName) { | ||
paramName = paramName.replace(/^[$]+/, ""); | ||
if (!msg[paramName]) | ||
return; | ||
var isUrlParam = url.indexOf(":" + paramName) !== -1; | ||
var valFormat = isUrlParam || format != "json" ? "query" : format; | ||
var val; | ||
if (valFormat != "json" && typeof msg[paramName] == "object") { | ||
try { | ||
msg[paramName] = JSON.stringify(msg[paramName]); | ||
val = encodeURIComponent(msg[paramName]); | ||
} | ||
catch (ex) { | ||
return Util.log("httpSend: Error while converting object to JSON: " | ||
+ (ex.message || ex), "error"); | ||
} | ||
} | ||
else | ||
val = valFormat == "json" ? msg[paramName] : encodeURIComponent(msg[paramName]); | ||
if (isUrlParam) | ||
url = url.replace(":" + paramName, val); | ||
else { | ||
if (format == "json") | ||
query[paramName] = val; | ||
else | ||
query.push(paramName + "=" + val); | ||
} | ||
}); | ||
def.url = url; | ||
return query; | ||
} | ||
/** | ||
* Client#httpSend(msg, block, callback) -> null | ||
* - msg (Object): parameters to send as the request body | ||
* - block (Object): parameter definition from the `routes.json` file that | ||
* contains validation rules | ||
* - callback (Function): function to be called when the request returns. | ||
* If the the request returns with an error, the error is passed to | ||
* the callback as its first argument (NodeJS-style). | ||
* | ||
* Send an HTTP request to the server and pass the result to a callback. | ||
**/ | ||
this.httpSend = function(msg, block, callback) { | ||
var method = block.method.toLowerCase(); | ||
var hasBody = ("head|get|delete".indexOf(method) === -1); | ||
var format = hasBody && this.constants.requestFormat | ||
? this.constants.requestFormat | ||
: "query"; | ||
var query = getQuery(msg, block, format); | ||
var path = (!hasBody && query.length) | ||
? block.url + "?" + query.join("&") | ||
: block.url; | ||
var protocol = this.constants.protocol || "http"; | ||
var host = this.constants.host; | ||
var port = this.constants.port || (protocol == "https" ? 443 : 80); | ||
if (this.config.proxy) { | ||
host = this.config.proxy.host; | ||
port = this.config.proxy.port || 3128; | ||
} | ||
var headers = { | ||
"host": host, | ||
"user-agent": "NodeJS HTTP Client", | ||
"content-length": "0" | ||
}; | ||
if (hasBody) { | ||
if (format == "json") | ||
query = JSON.stringify(query); | ||
else | ||
query = query.join("&"); | ||
headers["content-length"] = query.length; | ||
headers["content-type"] = format == "json" | ||
? "application/json" | ||
: "application/x-www-form-urlencoded"; | ||
} | ||
if (this.auth) { | ||
var basic; | ||
switch (this.auth.type) { | ||
case "oauth": | ||
path += (path.indexOf("?") === -1 ? "?" : "&") + | ||
"access_token=" + encodeURIComponent(this.auth.token); | ||
break; | ||
case "token": | ||
basic = new Buffer(this.auth.username + "/token:" + this.auth.token, "ascii").toString("base64"); | ||
headers.authorization = "Basic " + basic; | ||
break; | ||
case "basic": | ||
basic = new Buffer(this.auth.username + ":" + this.auth.password, "ascii").toString("base64"); | ||
headers.authorization = "Basic " + basic; | ||
break; | ||
default: | ||
break; | ||
} | ||
} | ||
var options = { | ||
host: host, | ||
port: port, | ||
path: path, | ||
method: method, | ||
headers: headers | ||
}; | ||
if (this.debug) | ||
console.log("REQUEST: ", options); | ||
var self = this; | ||
var req = require(protocol).request(options, function(res) { | ||
if (self.debug) { | ||
console.log("STATUS: " + res.statusCode); | ||
console.log("HEADERS: " + JSON.stringify(res.headers)); | ||
} | ||
res.setEncoding("utf8"); | ||
var data = ""; | ||
res.on("data", function(chunk) { | ||
data += chunk; | ||
}); | ||
res.on("end", function() { | ||
if (res.statusCode >= 400 && res.statusCode < 600 || res.statusCode < 10) { | ||
callback(new error.HttpError(data, res.statusCode)); | ||
} | ||
else { | ||
res.data = data; | ||
callback(null, res); | ||
} | ||
}); | ||
}); | ||
req.on("error", function(e) { | ||
if (self.debug) | ||
console.log("problem with request: " + e.message); | ||
callback(e.message); | ||
}); | ||
// write data to request body | ||
if (hasBody && query.length) | ||
req.write(query + "\n"); | ||
req.end(); | ||
}; | ||
}).call(Client.prototype); | ||
throw new Error(`'node-github' is deprecated. Install 'github' instead`) |
{ | ||
"name" : "node-github", | ||
"version" : "0.0.3", | ||
"description" : "Wrapper for the GitHub API", | ||
"author": "Mike de Boer <mike@c9.io>", | ||
"homepage": "http://github.com/c9/node-github3", | ||
"repository" : { | ||
"type" : "git", | ||
"url" : "http://github.com/c9/node-github3.git" | ||
}, | ||
"engine" : { | ||
"node": ">=0.4.0" | ||
}, | ||
"devDependencies": { | ||
"asyncjs": ">=0.0.x", | ||
"oauth": ">=0.8.x" | ||
}, | ||
"main" : ".", | ||
"scripts": { | ||
"test": "node ./test/all.js" | ||
}, | ||
"licenses": [{ | ||
"type": "The MIT License", | ||
"url": "http://www.opensource.org/licenses/mit-license.php"} | ||
] | ||
"deprecated": false, | ||
"description": "DEPRECATED: use 'github' package", | ||
"name": "node-github", | ||
"scripts": { | ||
"preinstall": "node index.js" | ||
}, | ||
"version": "0.0.4" | ||
} |
115
README.md
@@ -1,114 +0,3 @@ | ||
# JavaScript GitHub API for Node.JS | ||
# DEPRECATED | ||
A Node.JS module, which provides an object oriented wrapper for the GitHub v3 API. | ||
## Installation | ||
Install with the Node.JS package manager [npm](http://npmjs.org/): | ||
$ npm install node-github | ||
or | ||
Install via git clone: | ||
$ git clone git://github.com/c9/node-github3.git | ||
$ cd node-github3 | ||
$ npm install | ||
## Documentation | ||
You can find the docs for the API of this client at [http://c9.github.com/node-github3/](http://c9.github.com/node-github3/) | ||
Additionally, the [official Github documentation](http://developer.github.com/) | ||
is a very useful resource. | ||
## Example | ||
Print all followers of the user "mikedeboer" to the console. | ||
var GitHubApi = require("node-github"); | ||
var github = new GitHubApi({ | ||
version: "3.0.0" | ||
}); | ||
github.user.getFollowingFromUser({ | ||
user: "mikedeboer" | ||
}, function(err, res) { | ||
console.log(JSON.stringify(res)); | ||
}); | ||
First the _GitHubApi_ class is imported from the _node-github_ module. This class provides | ||
access to all of GitHub's APIs (e.g. user, issues or repo APIs). The _getFollowingFromUser_ | ||
method lists all followers of a given GitHub user. Is is part of the user API. It | ||
takes the user name as first argument and a callback as last argument. Once the | ||
follower list is returned from the server, the callback is called. | ||
Like in Node.JS, callbacks are always the last argument. If the functions fails an | ||
error object is passed as first argument to the callback. | ||
## Authentication | ||
Most GitHub API calls don't require authentication. As a rule of thumb: If you | ||
can see the information by visiting the site without being logged in, you don't | ||
have to be authenticated to retrieve the same information through the API. Of | ||
course calls, which change data or read sensitive information have to be authenticated. | ||
You need the GitHub user name and the API key for authentication. The API key can | ||
be found in the user's _Account Settings_ page. | ||
This example shows how to authenticate and then change _location_ field of the | ||
account settings to _Argentina_: | ||
github.authenticate({ | ||
type: "basic", | ||
username: username, | ||
password: password | ||
}); | ||
github.user.update({ | ||
location: "Argentina" | ||
}, function(err) { | ||
console.log("done!"); | ||
}); | ||
Note that the _authenticate_ method is synchronous because it only stores the | ||
credentials for the next request. | ||
## Implemented GitHub APIs | ||
* Gists: 100% | ||
* Git Data: 100% | ||
* Issues: 100% | ||
* Orgs: 100% | ||
* Pull Requests: 100% | ||
* Repos: 100% | ||
* Users: 100% | ||
* Events: 100% | ||
## Running the Tests | ||
The unit tests are based on the [ayncjs](https://github.com/ajaxorg/async.js) | ||
module, which is provided as an npm dependency. To run the tests make sure that the | ||
npm dependencies are installed by running `npm install` from the project directory. | ||
Running all unit tests: | ||
npm test | ||
or | ||
node test/all.js | ||
The test classes can also be run separately. This will e.g. run the UserApi test: | ||
node api/v3.0.0/userTest.js | ||
Note that a connection to the internet is required to run the tests. | ||
## TODO | ||
* generate Client documentation | ||
## LICENSE | ||
MIT license. See the LICENSE file for details. | ||
`node-github` is deprecated. Install [github](https://www.npmjs.com/package/github) instead. |
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Install scripts
Supply chain riskInstall scripts are run when the package is installed. The majority of malware in npm is hidden in install scripts.
Found 1 instance in 1 package
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
New author
Supply chain riskA new npm collaborator published a version of the package for the first time. New collaborators are usually benign additions to a project, but do indicate a change to the security surface area of a package.
Found 1 instance in 1 package
Trivial Package
Supply chain riskPackages less than 10 lines of code are easily copied into your own project and may not warrant the additional supply chain risk of an external dependency.
Found 1 instance in 1 package
No contributors or author data
MaintenancePackage does not specify a list of contributors or an author in package.json.
Found 1 instance in 1 package
No License Found
License(Experimental) License information could not be found.
Found 1 instance in 1 package
No repository
Supply chain riskPackage does not have a linked source code repository. Without this field, a package will have no reference to the location of the source code use to generate the package.
Found 1 instance in 1 package
No website
QualityPackage does not have a website.
Found 1 instance in 1 package
License Policy Violation
LicenseThis package is not allowed per your license policy. Review the package's license to ensure compliance.
Found 1 instance in 1 package
Deprecated
MaintenanceThe maintainer of the package marked it as deprecated. This could indicate that a single version should not be used, or that the package is no longer maintained and any new vulnerabilities will not be fixed.
Found 1 instance in 1 package
Network access
Supply chain riskThis module accesses the network.
Found 1 instance in 1 package
Dynamic require
Supply chain riskDynamic require can indicate the package is performing dangerous or unsafe dynamic code execution.
Found 1 instance in 1 package
Filesystem access
Supply chain riskAccesses the file system, and could potentially read sensitive data.
Found 1 instance in 1 package
0
0
1
360
3
1
1
3
4
1
3