Convert Figma logo to code with AI

nodejs logollhttp

Port of http_parser to llparse

1,686
187
1,686
16

Top Related Projects

http request/response parser for c

tiny HTTP parser written in C (used in HTTP::Parser::XS et al.)

Simple, secure & standards compliant web server for the most demanding of applications

6,411

An HTTP/1.1 client, written from scratch for Node.js

6,349

The JavaScript / Wasm runtime that powers Cloudflare Workers

Quick Overview

llhttp is a lightweight, fast, and secure HTTP parsing library written in C. It's designed to be used in high-performance server applications and is the HTTP parser used in Node.js. The project aims to provide a robust and efficient solution for parsing HTTP requests and responses.

Pros

  • High performance and low memory footprint
  • Extensively tested and battle-proven in production environments
  • Supports both HTTP/1.x and parts of HTTP/2
  • Easy to integrate into existing C/C++ projects

Cons

  • Limited documentation for advanced use cases
  • Requires careful memory management when used in C
  • May have a steeper learning curve compared to higher-level HTTP libraries
  • Not directly usable in higher-level languages without bindings

Code Examples

  1. Initializing the parser:
#include "llhttp.h"

llhttp_t parser;
llhttp_settings_t settings;

llhttp_settings_init(&settings);
llhttp_init(&parser, HTTP_REQUEST, &settings);
  1. Setting up callbacks:
settings.on_url = on_url_callback;
settings.on_header_field = on_header_field_callback;
settings.on_header_value = on_header_value_callback;
settings.on_body = on_body_callback;

static int on_url_callback(llhttp_t* parser, const char* at, size_t length) {
    // Handle URL parsing
    return 0;
}
  1. Parsing HTTP data:
const char* data = "GET /index.html HTTP/1.1\r\n"
                   "Host: example.com\r\n"
                   "\r\n";

enum llhttp_errno err = llhttp_execute(&parser, data, strlen(data));
if (err != HPE_OK) {
    fprintf(stderr, "Parse error: %s %s\n", llhttp_errno_name(err),
            parser.reason);
}

Getting Started

To use llhttp in your project:

  1. Clone the repository:

    git clone https://github.com/nodejs/llhttp.git
    
  2. Build the library:

    cd llhttp
    npm install
    make
    
  3. Include the header and link against the library in your C project:

    #include "llhttp.h"
    
    // Compile with:
    // gcc -I/path/to/llhttp/include -L/path/to/llhttp/lib -lllhttp your_file.c
    
  4. Initialize the parser and set up callbacks as shown in the code examples above.

Competitor Comparisons

http request/response parser for c

Pros of http-parser

  • Mature and well-established project with a longer history
  • Simpler implementation, potentially easier to understand and maintain
  • Written in C, which may be preferable for some use cases

Cons of http-parser

  • Lower performance compared to llhttp
  • Less actively maintained, with fewer recent updates
  • Limited support for newer HTTP features and protocols

Code Comparison

http-parser:

int http_parser_execute(http_parser *parser,
                        const http_parser_settings *settings,
                        const char *data,
                        size_t len)

llhttp:

enum llhttp_errno llhttp_execute(llhttp_t* parser,
                                 const char* data,
                                 size_t len)

Key Differences

  • llhttp is designed as a drop-in replacement for http-parser
  • llhttp offers improved performance and lower memory usage
  • llhttp provides better support for modern HTTP features
  • llhttp is generated from a TypeScript codebase, allowing for easier maintenance and updates
  • llhttp has more active development and frequent releases

Use Cases

  • http-parser may be preferred for legacy projects or those requiring a simpler C implementation
  • llhttp is recommended for new projects and those seeking better performance and modern HTTP support

Community and Support

  • Both projects are part of the Node.js organization on GitHub
  • llhttp has more recent activity and a growing community of contributors
  • http-parser still receives occasional updates and bug fixes

tiny HTTP parser written in C (used in HTTP::Parser::XS et al.)

Pros of picohttpparser

  • Extremely lightweight and fast, optimized for performance
  • Simple C implementation, easy to integrate into other projects
  • Designed for parsing HTTP/1 requests and headers efficiently

Cons of picohttpparser

  • Limited functionality compared to llhttp (focuses mainly on parsing)
  • Less actively maintained and updated
  • May require additional code for complete HTTP handling

Code Comparison

picohttpparser:

struct phr_header headers[100];
size_t num_headers = 100;
const char *method, *path;
int pret = phr_parse_request(buf, len, &method, &method_len, &path, &path_len,
                             &minor_version, headers, &num_headers, last_len);

llhttp:

llhttp_t parser;
llhttp_settings_t settings;
llhttp_init(&parser, HTTP_REQUEST, &settings);
enum llhttp_errno err = llhttp_execute(&parser, buf, len);
if (err == HPE_OK) {
    // Request fully parsed
}

Both libraries provide efficient HTTP parsing, but llhttp offers a more comprehensive solution for HTTP handling in Node.js environments, while picohttpparser excels in lightweight, performance-critical scenarios.

Simple, secure & standards compliant web server for the most demanding of applications

Pros of uWebSockets

  • Higher performance and lower latency for WebSocket connections
  • Built-in support for SSL/TLS and HTTP/HTTPS
  • More comprehensive WebSocket implementation with additional features

Cons of uWebSockets

  • Less focused on HTTP parsing compared to llhttp
  • Steeper learning curve due to more complex API
  • May require more manual configuration for certain use cases

Code Comparison

llhttp (HTTP parsing):

llhttp_settings_t settings;
llhttp_settings_init(&settings);
settings.on_message_complete = on_message_complete;

llhttp_t parser;
llhttp_init(&parser, HTTP_REQUEST, &settings);

uWebSockets (WebSocket server):

uWS::App().ws<PerSocketData>("/*", {
    .open = [](auto *ws, auto *req) {
        // Handle new WebSocket connection
    },
    .message = [](auto *ws, std::string_view message, uWS::OpCode opCode) {
        // Handle incoming WebSocket message
    }
}).listen(9001, [](auto *listen_socket) {
    if (listen_socket) {
        std::cout << "Listening on port 9001" << std::endl;
    }
});

Summary

llhttp is primarily focused on efficient HTTP parsing, while uWebSockets provides a more comprehensive solution for WebSocket and HTTP/HTTPS servers. uWebSockets offers higher performance for WebSocket connections but may have a steeper learning curve. The choice between the two depends on the specific requirements of your project, with llhttp being more suitable for pure HTTP parsing tasks and uWebSockets for full-featured WebSocket implementations.

6,411

An HTTP/1.1 client, written from scratch for Node.js

Pros of undici

  • Higher-level HTTP client with a more user-friendly API
  • Built-in connection pooling and automatic keep-alive handling
  • Support for modern HTTP features like HTTP/2 and WebSockets

Cons of undici

  • Larger package size and potentially higher memory footprint
  • May have a steeper learning curve for developers familiar with Node.js core HTTP module

Code comparison

undici:

import { request } from 'undici'

const { statusCode, headers, body } = await request('http://example.com')
console.log(statusCode, headers)
for await (const data of body) {
  console.log(data)
}

llhttp:

const http = require('http')

http.get('http://example.com', (res) => {
  console.log(res.statusCode, res.headers)
  res.on('data', (chunk) => console.log(chunk))
})

Key differences

  • undici provides a more modern, Promise-based API
  • llhttp is a lower-level HTTP parser used internally by Node.js
  • undici offers more features out-of-the-box, while llhttp focuses on parsing efficiency

Use cases

  • undici: Ideal for applications requiring a feature-rich HTTP client
  • llhttp: Better suited for projects needing low-level HTTP parsing or custom implementations

Both libraries have their merits, and the choice depends on specific project requirements and developer preferences.

6,349

The JavaScript / Wasm runtime that powers Cloudflare Workers

Pros of workerd

  • Designed for edge computing, offering a more comprehensive serverless platform
  • Supports multiple languages and runtimes, not limited to JavaScript
  • Provides built-in security features and isolation mechanisms

Cons of workerd

  • More complex setup and configuration compared to llhttp
  • Potentially higher resource usage due to its broader feature set
  • Less focused on HTTP parsing, as it's a full-fledged runtime environment

Code Comparison

llhttp (HTTP parsing):

llhttp_settings_t settings;
llhttp_settings_init(&settings);
settings.on_message_complete = on_message_complete;

llhttp_t parser;
llhttp_init(&parser, HTTP_REQUEST, &settings);

workerd (Worker setup):

export default {
  async fetch(request, env, ctx) {
    return new Response("Hello World!");
  }
};

While llhttp focuses on low-level HTTP parsing, workerd provides a higher-level abstraction for building serverless applications. llhttp is more suitable for projects requiring fine-grained control over HTTP parsing, while workerd is better suited for developing edge computing applications with built-in security and isolation features.

Convert Figma logo designs to code with AI

Visual Copilot

Introducing Visual Copilot: A new AI model to turn Figma designs to high quality code using your components.

Try Visual Copilot

README

llhttp

CI

Port of http_parser to llparse.

Why?

Let's face it, http_parser is practically unmaintainable. Even introduction of a single new method results in a significant code churn.

This project aims to:

  • Make it maintainable
  • Verifiable
  • Improving benchmarks where possible

More details in Fedor Indutny's talk at JSConf EU 2019

How?

Over time, different approaches for improving http_parser's code base were tried. However, all of them failed due to resulting significant performance degradation.

This project is a port of http_parser to TypeScript. llparse is used to generate the output C source file, which could be compiled and linked with the embedder's program (like Node.js).

Performance

So far llhttp outperforms http_parser:

input sizebandwidthreqs/sectime
llhttp8192.00 mb1777.24 mb/s3583799.39 req/sec4.61 s
http_parser8192.00 mb694.66 mb/s1406180.33 req/sec11.79 s

llhttp is faster by approximately 156%.

Maintenance

llhttp project has about 1400 lines of TypeScript code describing the parser itself and around 450 lines of C code and headers providing the helper methods. The whole http_parser is implemented in approximately 2500 lines of C, and 436 lines of headers.

All optimizations and multi-character matching in llhttp are generated automatically, and thus doesn't add any extra maintenance cost. On the contrary, most of http_parser's code is hand-optimized and unrolled. Instead describing "how" it should parse the HTTP requests/responses, a maintainer should implement the new features in http_parser cautiously, considering possible performance degradation and manually optimizing the new code.

Verification

The state machine graph is encoded explicitly in llhttp. The llparse automatically checks the graph for absence of loops and correct reporting of the input ranges (spans) like header names and values. In the future, additional checks could be performed to get even stricter verification of the llhttp.

Usage

#include "stdio.h"
#include "llhttp.h"
#include "string.h"

int handle_on_message_complete(llhttp_t* parser) {
	fprintf(stdout, "Message completed!\n");
	return 0;
}

int main() {
	llhttp_t parser;
	llhttp_settings_t settings;

	/*Initialize user callbacks and settings */
	llhttp_settings_init(&settings);

	/*Set user callback */
	settings.on_message_complete = handle_on_message_complete;

	/*Initialize the parser in HTTP_BOTH mode, meaning that it will select between
	*HTTP_REQUEST and HTTP_RESPONSE parsing automatically while reading the first
	*input.
	*/
	llhttp_init(&parser, HTTP_BOTH, &settings);

	/*Parse request! */
	const char* request = "GET / HTTP/1.1\r\n\r\n";
	int request_len = strlen(request);

	enum llhttp_errno err = llhttp_execute(&parser, request, request_len);
	if (err == HPE_OK) {
		fprintf(stdout, "Successfully parsed!\n");
	} else {
		fprintf(stderr, "Parse error: %s %s\n", llhttp_errno_name(err), parser.reason);
	}
}

For more information on API usage, please refer to src/native/api.h.

API

llhttp_settings_t

The settings object contains a list of callbacks that the parser will invoke.

The following callbacks can return 0 (proceed normally), -1 (error) or HPE_PAUSED (pause the parser):

  • on_message_begin: Invoked when a new request/response starts.
  • on_message_complete: Invoked when a request/response has been completedly parsed.
  • on_url_complete: Invoked after the URL has been parsed.
  • on_method_complete: Invoked after the HTTP method has been parsed.
  • on_protocol_complete: Invoked after the HTTP version has been parsed.
  • on_version_complete: Invoked after the HTTP version has been parsed.
  • on_status_complete: Invoked after the status code has been parsed.
  • on_header_field_complete: Invoked after a header name has been parsed.
  • on_header_value_complete: Invoked after a header value has been parsed.
  • on_chunk_header: Invoked after a new chunk is started. The current chunk length is stored in parser->content_length.
  • on_chunk_extension_name_complete: Invoked after a chunk extension name is started.
  • on_chunk_extension_value_complete: Invoked after a chunk extension value is started.
  • on_chunk_complete: Invoked after a new chunk is received.
  • on_reset: Invoked after on_message_complete and before on_message_begin when a new message is received on the same parser. This is not invoked for the first message of the parser.

The following callbacks can return 0 (proceed normally), -1 (error) or HPE_USER (error from the callback):

  • on_url: Invoked when another character of the URL is received.
  • on_status: Invoked when another character of the status is received.
  • on_method: Invoked when another character of the method is received. When parser is created with HTTP_BOTH and the input is a response, this also invoked for the sequence HTTP/ of the first message.
  • on_protocol: Invoked when another character of the protocol is received.
  • on_version: Invoked when another character of the version is received.
  • on_header_field: Invoked when another character of a header name is received.
  • on_header_value: Invoked when another character of a header value is received.
  • on_chunk_extension_name: Invoked when another character of a chunk extension name is received.
  • on_chunk_extension_value: Invoked when another character of a extension value is received.

The callback on_headers_complete, invoked when headers are completed, can return:

  • 0: Proceed normally.
  • 1: Assume that request/response has no body, and proceed to parsing the next message.
  • 2: Assume absence of body (as above) and make llhttp_execute() return HPE_PAUSED_UPGRADE.
  • -1: Error
  • HPE_PAUSED: Pause the parser.

void llhttp_init(llhttp_t* parser, llhttp_type_t type, const llhttp_settings_t* settings)

Initialize the parser with specific type and user settings.

uint8_t llhttp_get_type(llhttp_t* parser)

Returns the type of the parser.

uint8_t llhttp_get_http_major(llhttp_t* parser)

Returns the major version of the HTTP protocol of the current request/response.

uint8_t llhttp_get_http_minor(llhttp_t* parser)

Returns the minor version of the HTTP protocol of the current request/response.

uint8_t llhttp_get_method(llhttp_t* parser)

Returns the method of the current request.

int llhttp_get_status_code(llhttp_t* parser)

Returns the method of the current response.

uint8_t llhttp_get_upgrade(llhttp_t* parser)

Returns 1 if request includes the Connection: upgrade header.

void llhttp_reset(llhttp_t* parser)

Reset an already initialized parser back to the start state, preserving the existing parser type, callback settings, user data, and lenient flags.

void llhttp_settings_init(llhttp_settings_t* settings)

Initialize the settings object.

llhttp_errno_t llhttp_execute(llhttp_t* parser, const char* data, size_t len)

Parse full or partial request/response, invoking user callbacks along the way.

If any of llhttp_data_cb returns errno not equal to HPE_OK - the parsing interrupts, and such errno is returned from llhttp_execute(). If HPE_PAUSED was used as a errno, the execution can be resumed with llhttp_resume() call.

In a special case of CONNECT/Upgrade request/response HPE_PAUSED_UPGRADE is returned after fully parsing the request/response. If the user wishes to continue parsing, they need to invoke llhttp_resume_after_upgrade().

if this function ever returns a non-pause type error, it will continue to return the same error upon each successive call up until llhttp_init() is called.

llhttp_errno_t llhttp_finish(llhttp_t* parser)

This method should be called when the other side has no further bytes to send (e.g. shutdown of readable side of the TCP connection.)

Requests without Content-Length and other messages might require treating all incoming bytes as the part of the body, up to the last byte of the connection.

This method will invoke on_message_complete() callback if the request was terminated safely. Otherwise a error code would be returned.

int llhttp_message_needs_eof(const llhttp_t* parser)

Returns 1 if the incoming message is parsed until the last byte, and has to be completed by calling llhttp_finish() on EOF.

int llhttp_should_keep_alive(const llhttp_t* parser)

Returns 1 if there might be any other messages following the last that was successfully parsed.

void llhttp_pause(llhttp_t* parser)

Make further calls of llhttp_execute() return HPE_PAUSED and set appropriate error reason.

Do not call this from user callbacks! User callbacks must return HPE_PAUSED if pausing is required.

void llhttp_resume(llhttp_t* parser)

Might be called to resume the execution after the pause in user's callback.

See llhttp_execute() above for details.

Call this only if llhttp_execute() returns HPE_PAUSED.

void llhttp_resume_after_upgrade(llhttp_t* parser)

Might be called to resume the execution after the pause in user's callback. See llhttp_execute() above for details.

Call this only if llhttp_execute() returns HPE_PAUSED_UPGRADE

llhttp_errno_t llhttp_get_errno(const llhttp_t* parser)

Returns the latest error.

const char* llhttp_get_error_reason(const llhttp_t* parser)

Returns the verbal explanation of the latest returned error.

User callback should set error reason when returning the error. See llhttp_set_error_reason() for details.

void llhttp_set_error_reason(llhttp_t* parser, const char* reason)

Assign verbal description to the returned error. Must be called in user callbacks right before returning the errno.

HPE_USER error code might be useful in user callbacks.

const char* llhttp_get_error_pos(const llhttp_t* parser)

Returns the pointer to the last parsed byte before the returned error. The pointer is relative to the data argument of llhttp_execute().

This method might be useful for counting the number of parsed bytes.

const char* llhttp_errno_name(llhttp_errno_t err)

Returns textual name of error code.

const char* llhttp_method_name(llhttp_method_t method)

Returns textual name of HTTP method.

const char* llhttp_status_name(llhttp_status_t status)

Returns textual name of HTTP status.

void llhttp_set_lenient_headers(llhttp_t* parser, int enabled)

Enables/disables lenient header value parsing (disabled by default). Lenient parsing disables header value token checks, extending llhttp's protocol support to highly non-compliant clients/server.

No HPE_INVALID_HEADER_TOKEN will be raised for incorrect header values when lenient parsing is "on".

Enabling this flag can pose a security issue since you will be exposed to request smuggling attacks. USE WITH CAUTION!

void llhttp_set_lenient_chunked_length(llhttp_t* parser, int enabled)

Enables/disables lenient handling of conflicting Transfer-Encoding and Content-Length headers (disabled by default).

Normally llhttp would error when Transfer-Encoding is present in conjunction with Content-Length.

This error is important to prevent HTTP request smuggling, but may be less desirable for small number of cases involving legacy servers.

Enabling this flag can pose a security issue since you will be exposed to request smuggling attacks. USE WITH CAUTION!

void llhttp_set_lenient_keep_alive(llhttp_t* parser, int enabled)

Enables/disables lenient handling of Connection: close and HTTP/1.0 requests responses.

Normally llhttp would error the HTTP request/response after the request/response with Connection: close and Content-Length.

This is important to prevent cache poisoning attacks, but might interact badly with outdated and insecure clients.

With this flag the extra request/response will be parsed normally.

Enabling this flag can pose a security issue since you will be exposed to poisoning attacks. USE WITH CAUTION!

void llhttp_set_lenient_transfer_encoding(llhttp_t* parser, int enabled)

Enables/disables lenient handling of Transfer-Encoding header.

Normally llhttp would error when a Transfer-Encoding has chunked value and another value after it (either in a single header or in multiple headers whose value are internally joined using , ).

This is mandated by the spec to reliably determine request body size and thus avoid request smuggling.

With this flag the extra value will be parsed normally.

Enabling this flag can pose a security issue since you will be exposed to request smuggling attacks. USE WITH CAUTION!

void llhttp_set_lenient_version(llhttp_t* parser, int enabled)

Enables/disables lenient handling of HTTP version.

Normally llhttp would error when the HTTP version in the request or status line is not 0.9, 1.0, 1.1 or 2.0. With this flag the extra value will be parsed normally.

Enabling this flag can pose a security issue since you will allow unsupported HTTP versions. USE WITH CAUTION!

void llhttp_set_lenient_data_after_close(llhttp_t* parser, int enabled)

Enables/disables lenient handling of additional data received after a message ends and keep-alive is disabled.

Normally llhttp would error when additional unexpected data is received if the message contains the Connection header with close value. With this flag the extra data will discarded without throwing an error.

Enabling this flag can pose a security issue since you will be exposed to poisoning attacks. USE WITH CAUTION!

void llhttp_set_lenient_optional_lf_after_cr(llhttp_t* parser, int enabled)

Enables/disables lenient handling of incomplete CRLF sequences.

Normally llhttp would error when a CR is not followed by LF when terminating the request line, the status line, the headers or a chunk header. With this flag only a CR is required to terminate such sections.

Enabling this flag can pose a security issue since you will be exposed to request smuggling attacks. USE WITH CAUTION!

void llhttp_set_lenient_optional_cr_before_lf(llhttp_t* parser, int enabled)

Enables/disables lenient handling of line separators.

Normally llhttp would error when a LF is not preceded by CR when terminating the request line, the status line, the headers, a chunk header or a chunk data. With this flag only a LF is required to terminate such sections.

Enabling this flag can pose a security issue since you will be exposed to request smuggling attacks. USE WITH CAUTION!

void llhttp_set_lenient_optional_crlf_after_chunk(llhttp_t* parser, int enabled)

Enables/disables lenient handling of chunks not separated via CRLF.

Normally llhttp would error when after a chunk data a CRLF is missing before starting a new chunk. With this flag the new chunk can start immediately after the previous one.

Enabling this flag can pose a security issue since you will be exposed to request smuggling attacks. USE WITH CAUTION!

void llhttp_set_lenient_spaces_after_chunk_size(llhttp_t* parser, int enabled)

Enables/disables lenient handling of spaces after chunk size.

Normally llhttp would error when after a chunk size is followed by one or more spaces are present instead of a CRLF or ;. With this flag this check is disabled.

Enabling this flag can pose a security issue since you will be exposed to request smuggling attacks. USE WITH CAUTION!

Build Instructions

Make sure you have Node.js, npm and npx installed. Then under project directory run:

npm ci
make

Bindings to other languages

Using with CMake

If you want to use this library in a CMake project as a shared library, you can use the snippet below.

FetchContent_Declare(llhttp
  URL "https://github.com/nodejs/llhttp/archive/refs/tags/release/v8.1.0.tar.gz")

FetchContent_MakeAvailable(llhttp)

# Link with the llhttp_shared target
target_link_libraries(${EXAMPLE_PROJECT_NAME} ${PROJECT_LIBRARIES} llhttp_shared ${PROJECT_NAME})

If you want to use this library in a CMake project as a static library, you can set some cache variables first.

FetchContent_Declare(llhttp
  URL "https://github.com/nodejs/llhttp/archive/refs/tags/release/v8.1.0.tar.gz")

set(BUILD_SHARED_LIBS OFF CACHE INTERNAL "")
set(BUILD_STATIC_LIBS ON CACHE INTERNAL "")
FetchContent_MakeAvailable(llhttp)

# Link with the llhttp_static target
target_link_libraries(${EXAMPLE_PROJECT_NAME} ${PROJECT_LIBRARIES} llhttp_static ${PROJECT_NAME})

Note that using the git repo directly (e.g., via a git repo url and tag) will not work with FetchContent_Declare because CMakeLists.txt requires string replacements (e.g., _RELEASE_) before it will build.

Building on Windows

Installation

  • choco install git
  • choco install node
  • choco install llvm (or install the C++ Clang tools for Windows optional package from the Visual Studio 2019 installer)
  • choco install make (or if you have MinGW, it comes bundled)
  1. Ensure that Clang and make are in your system path.
  2. Using Git Bash, clone the repo to your preferred location.
  3. Cd into the cloned directory and run npm ci
  4. Run make
  5. Your repo/build directory should now have libllhttp.a and libllhttp.so static and dynamic libraries.
  6. When building your executable, you can link to these libraries. Make sure to set the build folder as an include path when building so you can reference the declarations in repo/build/llhttp.h.

A simple example on linking with the library:

Assuming you have an executable main.cpp in your current working directory, you would run: clang++ -Os -g3 -Wall -Wextra -Wno-unused-parameter -I/path/to/llhttp/build main.cpp /path/to/llhttp/build/libllhttp.a -o main.exe.

If you are getting unresolved external symbol linker errors you are likely attempting to build llhttp.c without linking it with object files from api.c and http.c.

LICENSE

This software is licensed under the MIT License.

Copyright Fedor Indutny, 2018.

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

NPM DownloadsLast 30 Days