npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

@geode/genepi

v1.2.0

Published

Automatic generation of N-API wrapper from a C++ library

Downloads

7

Readme


Introduction

genepi is a C++11 library providing a complete set of macros to generate a Node.js addon from your C++ code using N-API.

genepi works with cmake-js (a Node.js native addon of CMake) as build system. CMake is largely used in the C++ community, here are some arguments to why CMake is a great build system for Node.js addons: link.

genepi is MIT licensed and based on templates and macros inspired by nbind but using N-API.

Features

genepi allows you to:

  • Use your C++ API from JavaScript without any extra effort.
    • From Node.js and Electron,
    • On Linux, macOS and Windows,
    • Without changes to your C++ code. Simply add a separate short description at the end.
  • Distribute native code binary.

In more detail:

  • Export C++ classes, even ones not visible from other files.
  • Export multiple C++ inheritances, even between several libraries.
  • Export C++ methods simply by mentioning their names.
  • Auto-detect argument and return types from C++ declarations.
  • Automatically convert types and data structures between languages.
  • Call C++ methods from JavaScript with type checking.
  • Pass instances of compatible classes by value between languages (through the C++ stack).

Requirements

You need Node.js (at least v10.x) and one of the following C++ compilers:

Quick start

  1. Use your already existing C++ code in JavaScript
// My C++ code in hello.cpp
#include <iostream>
#include <string>
 
struct Greeter {
    static void sayHello( const std::string& name ) 
    {
        std::cout << "Hello, " << name << std::endl;
    }
};
  1. Install genepi and add some scripts to the package.json
npm install @geode/genepi
{
  "scripts": {
    "build": "cmake-js compile",
    "build:debug": "cmake-js compile -D"
  }
}
  1. Add JavaScript binding
// Add this to the file (or in another file)
#include <genepi/genepi.h>
 
GENEPI_CLASS( Greeter )
{
    GENEPI_METHOD( sayHello );
}
GENEPI_MODULE( hello )
  1. Configure your project by creating a CMakeLists.txt
cmake_minimum_required(VERSION 3.5)

project(my_project)

find_package(genepi REQUIRED PATHS ${PROJECT_SOURCE_DIR}/node_modules/@geode/genepi/build)

add_genepi_library(my_project "hello.cpp")
  1. Compile your addon
npm run build
  1. Use it!
var myProject = require('my_project.node');
myProject.Greeter.sayHello('you');

User guide

Creating your project

Create your repository using the provided Github template: genepi-template. Here is how to use a Github template: link.

Calling from Node.js

Each genepi module (i.e. each Node.js addon generated) needs to be registered using the GENEPI_MODULE macro:

// My C++ library

GENEPI_MODULE( my_addon );

This name my_addon is only used by N-API. The name of the addon is set in the CMakeLists.txt using the add_genepi_library macro. See Quick start.

// My JavaScript file
var example = require('my-genepi-addon.node');

// Use the binding

This require will only work if the module can be found by Node.js. To ease the import, you can use the package bindings. It will try several possible paths to find the module.

// My JavaScript file
var example = require('bindings')('my-genepi-addon');

// Use the binding

Functions

Functions not belonging to any class can be exported inside a named or an anonymous namespace. The C++ function gets exported to JavaScript with the same name using GENEPI_FUNCTION, or it can be renamed by adding a second argument (without quotation marks) using NAMED_GENEPI_FUNCTION. Addind double underscore in the new name will result in adding the function in nested objects on the js side.

If the C++ function is overloaded, GENEPI_MULTIFUNCTION macro must be used instead. See overloaded functions.

Example from C++: functions.cpp

#include <iostream>
#include <string>

void sayHello( const std::string& name )
{
    std::cout << "Hello, " << name << std::endl;
}

void sayBye( const std::string& name )
{
    std::cout << "Bye, " << name << std::endl;
}

void sayByeAgain( const std::string& name )
{
    std::cout << "Bye again, " << name << std::endl;
}

namespace foo
{
    void sayNamespacedHello( const std::string& name )
    {
        std::cout << "Hello, " << name << std::endl;
    }
}

#include <genepi/genepi.h>

namespace
{
    GENEPI_FUNCTION( sayHello );
    NAMED_GENEPI_FUNCTION( sayBye, sayGoodbye );
    NAMED_GENEPI_FUNCTION( sayByeAgain, say__Goodbye );
}

namespace foo
{
    GENEPI_FUNCTION( sayNamespacedHello );
}

GENEPI_MODULE( functions );

Example from JavaScript: functions.js

var functions = require('genepi-functions.node');

functions.sayHello('you'); // Output: Hello, you
functions.sayGoodbye('you'); // Output: Bye, you
functions.say.Goodbye('you'); // Output: Bye again, you
functions.sayNamespacedHello('you'); // Output: Hello, you

Overloaded functions

The GENEPI_FUNCTION() macro cannot distinguish between several overloaded versions of the same function, causing an error. In this case the GENEPI_MULTIFUNCTION() macro must be used.

The second parameter of the macro is the return type. For calling from JavaScript, each overload needs to have a distinct name, given in the third parameter (without quotation marks). The remaining parameters are the parameter types of the C++ function.

Example from C++: overloaded-functions.cpp

#include <iostream>
#include <string>

void test( const std::string& number )
{
    std::cout << "Number " << number << std::endl;
}

void test( int number )
{
    std::cout << "Number " << number << std::endl;
}

void test( int number, int another_number )
{
    std::cout << "Number " << number + another_number << std::endl;
}

#include <genepi/genepi.h>

namespace
{
    GENEPI_MULTIFUNCTION( test, void, test_string, const std::string& );
    GENEPI_MULTIFUNCTION( test, void, test_int, int );
    GENEPI_MULTIFUNCTION( test, void, test_int2, int, int );
}

GENEPI_MODULE( overloaded_functions );

Example from JavaScript: overloaded-functions.js

var overloadedFunctions = require('genepi-overloaded-functions.node');

overloadedFunctions.test_string('42'); // Output: Number 42
overloadedFunctions.test_int(42); // Output: Number 42
overloadedFunctions.test_int2(20, 22); // Output: Number 42

Classes and constructors

The GENEPI_CLASS(className) macro takes the name of your C++ class as an argument (without any quotation marks), and exports it to JavaScript using the same name. It's followed by a curly brace enclosed block of method exports, as if it was a function definition.

The class can be renamed on the JavaScript side by using the NAMED_GENEPI_CLASS macro and passing a string as a second argument. This is especially useful for binding a template class specialization with a more reasonable name: NAMED_GENEPI_CLASS(Data<int>, IntData).

Constructors are exported with a macro call GENEPI_CONSTRUCTOR(types...); where types is a comma-separated list of arguments to the constructor, such as int, int. Calling GENEPI_CONSTRUCTOR multiple times allows overloading it.

Constructor arguments are the only types that genepi cannot detect automatically.

Example from C++: classes.cpp

#include <iostream>
#include <string>

class ClassExample
{
public:
    ClassExample()
    {
        std::cout << "No arguments" << std::endl;
    }

    ClassExample( int a, int b )
    {
        std::cout << "Ints: " << a << " " << b << std::endl;
    }

    ClassExample( const std::string& msg )
    {
        std::cout << "String: " << msg << std::endl;
    }
};

#include <genepi/genepi.h>

GENEPI_CLASS( ClassExample )
{
    GENEPI_CONSTRUCTOR();
    GENEPI_CONSTRUCTOR( int, int );
    GENEPI_CONSTRUCTOR( const std::string& );
}

GENEPI_MODULE( classes );

Example from JavaScript: classes.js

var classes = require('genepi-classes.node');

var a = new classes.ClassExample();  // Output: No arguments
var b = new classes.ClassExample(42, 54); // Output: Ints: 42 54
var c = new classes.ClassExample("Don't panic"); // Output: String: Don't panic

Methods

Methods are exported inside a GENEPI_CLASS or a NAMED_GENEPI_CLASS block with a macro call GENEPI_METHOD which takes the name of the method as an argument (without any quotation marks). The C++ method gets exported to JavaScript with the same name.

If the C++ method is overloaded, GENEPI_MULTIMETHOD macro must be used instead. See overloaded methods.

If the method is static, it becomes a property of the JavaScript constructor function and can be accessed like className.methodName(). Otherwise it becomes a property of the prototype and can be accessed like obj = new className(); obj.methodName();

Example from C++: methods.cpp

#include <iostream>

class MethodExample
{
public:
    void add( int a, int b )
    {
        sum_ += a + b;
        std::cout << "Sum = " << sum_ << std::endl;
    }

    static void static_add( int a, int b )
    {
        MethodExample example;
        example.add( a, b );
    }

private:
    int sum_{ 0 };
};

#include <genepi/genepi.h>

GENEPI_CLASS( MethodExample )
{
    GENEPI_CONSTRUCTOR();
    GENEPI_METHOD( add );
    GENEPI_METHOD( static_add );
}

GENEPI_MODULE( methods );

Example from JavaScript: methods.js


var methods = require('genepi-methods.node');

var example = new methods.MethodExample();
example.add(12, 24); // Output: Sum = 36

methods.MethodExample.static_add(12,24); // Output: Sum = 36

Overloaded methods

The GENEPI_METHOD() macro, like GENEPI_FUNCTION macro, cannot distinguish between several overloaded versions of the same method. In this case the GENEPI_MULTIMETHOD() macro must be used.

The second parameter of the macro is the return type. For calling from JavaScript, each overload needs to have a distinct name, given in the third parameter (WITH quotation marks). The remaining parameters are the parameter types of the C++ method.

Example from C++: overloaded-methods.cpp

#include <iostream>
#include <string>

class OverloadMethod
{
public:
    void test( const std::string& number )
    {
        std::cout << "Number " << number << std::endl;
    }

    void test( int number )
    {
        std::cout << "Number " << number << std::endl;
    }

    void test( int number, int another_number )
    {
        std::cout << "Number " << number + another_number << std::endl;
    }
};

#include <genepi/genepi.h>

GENEPI_CLASS( OverloadMethod )
{
    GENEPI_CONSTRUCTOR();
    GENEPI_MULTIMETHOD( test, void, "test_string", const std::string& );
    GENEPI_MULTIMETHOD( test, void, "test_int", int );
    GENEPI_MULTIMETHOD( test, void, "test_int2", int, int );
}

GENEPI_MODULE( overloaded_methods );

Example from JavaScript: overloaded-functions.js

var overloadedMethods = require('genepi-overloaded-methods.node');

var a = new overloadedMethods.OverloadMethod();
a.test_string('42'); // Ouput: Number 42
a.test_int(42); // Ouput: Number 42
a.test_int2(20, 22); // Ouput: Number 42

Inheritance

When a C++ class inherits another, the GENEPI_INHERIT macro can be used to allow calling parent class methods on the child class, or passing child class instances to C++ methods expecting parent class instances.

Internally JavaScript only has prototype-based single inheritance while C++ supports multiple inheritance. To simulate it, genepi will copy the contents of the parents to the prototype. This has otherwise the same effect, except the JavaScript instanceof operator will return false for the parent classes.

Example from C++: inherit.cpp

#include <iostream>

class FirstParent
{
public:
    FirstParent()
    {
        std::cout << "FirstParent" << std::endl;
    }

    void from_first_parent()
    {
        std::cout << "from first parent" << std::endl;
    }
};

class SecondParent
{
public:
    SecondParent()
    {
        std::cout << "SecondParent" << std::endl;
    }

    void from_second_parent()
    {
        std::cout << "from second parent" << std::endl;
    }
};

class Child: public FirstParent, public SecondParent
{
public:
    Child()
    {
        std::cout << "Child" << std::endl;
    }
};

#include <genepi/genepi.h>

GENEPI_CLASS( FirstParent )
{
    GENEPI_CONSTRUCTOR();
    GENEPI_METHOD( from_first_parent );
}

GENEPI_CLASS( SecondParent )
{
    GENEPI_CONSTRUCTOR();
    GENEPI_METHOD( from_second_parent );
}

GENEPI_CLASS( Child )
{
    GENEPI_CONSTRUCTOR();
    GENEPI_INHERIT( FirstParent );
    GENEPI_INHERIT( SecondParent );
}

GENEPI_MODULE( inherit );

Example from JavaScript: overloaded-methods.js

var inherit = require('genepi-inherit.node');

var a = new inherit.Child(); // Ouput: FirstParent / SecondParent / Child
a.from_first_parent(); // Output: from first parent
a.from_second_parent(); // Output: from second parent

Passing data structures

genepi supports automatically converting between JavaScript arrays and C++ std::vector or std::array types. Just use them as arguments or return values in C++ methods.

Note that data structures don't use the same memory layout in both languages, so the data always gets copied which takes more time for more data. For example the strings in an array of strings also get copied, one character at a time.

Using objects

C++ objects can be passed to and from JavaScript using different parameter and return types in C++ code:

  • by reference using pointers or references
  • by value

Constness of objects is not ensured by genepi. So you could call a non const method on a const object.

Note: using pointers and references is particularly dangerous because the pointer may become invalid without JavaScript noticing it.

Example from C++: objects.cpp

#include <iostream>

class Coord
{
public:
    Coord( int x, int y ) : x_( x ), y_( y ) {}

    int getX()
    {
        return x_;
    }
    int getY()
    {
        return y_;
    }

private:
    int x_, y_;
};

class ObjectExample
{
public:
    static void showByValue( Coord coord )
    {
        std::cout << "C++ value " << coord.getX() << ", " << coord.getY()
                  << std::endl;
    }

    static void showByRef( Coord* coord )
    {
        std::cout << "C++ ref " << coord->getX() << ", " << coord->getY()
                  << std::endl;
    }

    static Coord getValue()
    {
        return Coord{ 12, 34 };
    }

    static Coord* getRef()
    {
        static Coord coord{ 56, 78 };
        return &coord;
    }
};

#include <genepi/genepi.h>

GENEPI_CLASS( Coord )
{
    GENEPI_CONSTRUCTOR( int, int );
    GENEPI_METHOD( getX );
    GENEPI_METHOD( getY );
}

GENEPI_CLASS( ObjectExample )
{
    GENEPI_METHOD( showByValue );
    GENEPI_METHOD( showByRef );
    GENEPI_METHOD( getValue );
    GENEPI_METHOD( getRef );
}

GENEPI_MODULE( objects );

Example from JavaScript: objects.js

var objects = require('genepi-objects.node');

var value1 = new objects.Coord(123, 456);
var value2 = objects.ObjectExample.getValue();
objects.ObjectExample.showByValue(value1); // Output: C++ value 123, 456
objects.ObjectExample.showByValue(value2); // Output: C++ value 12, 34

var ref = objects.ObjectExample.getRef();
objects.ObjectExample.showByRef(ref); // Output: C++ ref 56, 78

Type conversion

Parameters and return values of function calls between languages are automatically converted between equivalent types:

| JavaScript | C++ | | ---------- | ------------------------------------------- | | number | (un)signed char, short, int, long | | number | float, double | | boolean | bool | | string | const (unsigned) char * | | string | std::string | | Array | std::vector<type> | | Array | std::array<type, size> | | genepi-wrapped pointer | Pointer or reference to an instance of any bound classSee Using objects |

Alternatives

Questions

For questions and support please use the official slack and go to the channel #genepi. The issue list of this repo is exclusively for bug reports and feature requests.

Changelog

Detailed changes for each release are documented in the release notes.

License

MIT

Copyright (c) 2019-present, Geode-solutions