short-duid-js

1.2.1 • Public • Published

Short Distributed ID generator module

This module was inspired by icicle and snowflake. The ideas is to be able to generate non-colliding, URL friendly, and relatively short IDs that could be used in any application that requires to create URIs for arbitrary resources.

Looking around for what is available, I failed to find anything that would be simple and easy to implement. As a result, this module was born.

The id is a 64bit unsigned integer with 42 bits used for current timestamp in milliseconds, 10 bits used for shard id, and final 12 bits are used for revolving sequence.

timestamp_ms shard_id sequence
42bit 10bit 12bit

short-duid-js

Codacy Badge Quality Gate Status npm version npm downloads

Changelog

  • 1.2.1 Bugfix for the drifting time, and optimizations
  • 1.2.0 Depndencies' versions bumps and removed bn.js dependency
  • 1.1.6 Depndencies' versions bumps and tidyup of variables
  • 1.1.5 package.json clean-up and corrections
  • 1.1.4 General clean-up and updates (no new functionality)
  • 1.1.3 Updated dependencies
  • 1.1.1 No changes to the code, fixing CI pipeline
  • 1.1.0 Major update of dependencies
  • 1.0.2 Minor fixes
  • 1.0.0 Initial release

Requirements

  • nodejs 12.X
  • nodejs 14.X
  • nodejs 16.X
  • nodejs 18.X

Features

  • Written in pure JavaScript, no native code
  • Time and sequence based numeric unique ID generation
  • Time and sequence based alphanumeric URL-safe unique ID generation
  • Designed to be distributed among 1024 shards, no need to synchronize runtime or after setup
  • Can generate 4096 unique IDs per millisecond per shard (theoretically)
  • Can generate unique IDs for 139 years without overflow or collision
  • Resilient to time drift or sequence overflow, does not delay ID generation
  • Allows to set custom epoch, prolong unique ID generation and shorten the ID
  • Simple to use

Installation

npm install short-duid-js --save

How to use

This module is very simple to use, first you will need to initialize it and then start using its instance.

API

short_duid.init(shard_id, salt, epoch_start)

Instantiates short-duid and sets parameters for the life of instance.

Returns
  • Short Distributed ID module instance or Javascript Object type
    • Initializes new instance of the module with the given parameters
Parameters
  • shard_id - ID of this instance of short-duid, should be unique and not shared with other instances in the cluster; from 0 to 1023. This parameter will be converted into signed 32 bit integer and masked to fit in 12 bits.
  • salt - Salt that is used by hashid encoder/decoder, should be constant and shared across all nodes in the cluster. Do not change this parameter once used in production, or you will have collisions in the alphanumeric IDs. Good way to generate salt on Linux: dd if=/dev/random bs=1 count=102400 2>/dev/null| sha256sum
  • epoch_start - Number of milliseconds since unix epoch (1970, Jan 1 00:00:00 GMT). This should be some date in the near past and should never be changed further into the future once in production. Example: 1433116800000; //Mon, 01 Jun 2015 00:00:00 GMT. This parameter will be converted to unsigned 64bit integer.

instance.getDUID(count)

Method to retrieve array of DUIDs in alphanumeric form. Length of the array is specified by count parameter.

Returns
  • Javascript array object of variable length, depending on count parameter.
    • Example: [ "XLz0E3MvkEL" ]
Parameters
  • count - Number of alphanumeric DUIDs to return, from 0 to 8192.

instance.getDUIDInt(count)

Essential same method as _instance_.getDUID but instead of hashid converted integer, will return unique ID in a numeric form as string.

Returns
  • Javascript array object of variable length, depending on count parameter.
    • Example: [ "12534941854212112" ]
Parameters
  • count - Number of numeric DUIDs to return, from 0 to 8192.

instance.getShardID()

Method to get currently set shard ID of ShortDUID _instance_

Returns
  • number current shard ID of ShortDUID _instance_
    • Example: 0
Parameters
  • N/A

instance.getEpochStart()

Method to get currently set custom epoch starting time in milliseconds of ShortDUID _instance_

Returns
  • string currently set custom epoch of ShortDUID _instance_, since it is unsigned 64bit integer, we return it as string.
    • Example: "0"
Parameters
  • N/A

instance.getSalt()

Method to get current salt of ShortDUID _instance_. Salt is used to generate alphanumeric DUIDs and also in hashidEncode/hashidDecode methods.

Returns
  • string currently set salt that is used in hashid conversion of ShortDUID _instance_
    • Example: "this is my salt"
Parameters
  • N/A

instance.hashidEncode(number_array)

Method to hash(encode) array of unsigned 64bit integers (in Javascript string format).

Returns
  • string hashid of array of unsigned 64bit integers
    • Example: "3nMMYV0PvMl"
Parameters
  • number_array - Array of unsigned 64bit integers in javascript number or string (if does not fit in Javascript 58bit integer data type) form.

instance.hashidDecode(hashid_string)

Decode previously encoded array of numbers with hashid method.

Returns
  • Javascript array array of unsigned 64bit integers in a string form
    • Example: ['1', '2', '3']
Parameters
  • hashid_string - Hashid in a string form. Example: 3nMMYV0PvMl

Advanced API

This API is mainly used by unit tests and should not be required for normal usage of the module. Use it at your own risk.

instance.getCurrentTimeMs()

Method to get current time since unix epoch in milliseconds as seen by the module, not adjusted for custom epoch. This method can be useful in testing and also in capturing reference time to ensure timer stability across restarts.

Returns
  • string of numbers, current time since unix epoch in milliseconds as seen by the module.
Parameters
  • N/A

instance.driftTime(milliseconds)

Method to help simulate system_clock drift, can accept positive or negative integers.

Returns
  • string number of milliseconds to drift ShortDUID's internal clock
Parameters
  • milliseconds (optional) number of milliseconds to drift system_clock by, can be a positive or negative integer.

Example #1

Simplest example to execute all of the major methods of the module.

var duid = require("short-duid-js");
var duid_instance = duid.init(0, "my salt", 0);
console.log(duid_instance.getDUID(10));
console.log(duid_instance.getDUIDInt(10));

Example #2

More complete example that will create API server with help of fastify and reply to queries.

package.json
{
  "name": "ShortDUIDAPIServer",
  "description": "ShortDUID Example API Service",
  "version": "0.0.1",
  "private": true,
  "dependencies": {
    "fastify": "^4.0.1",
    "short-duid-js": "^1.2.0"
  },
  "scripts": {
    "start": "node api_server.js"
  }
}
api_server.js
"use strict";
const duid = require("short-duid-js");
const fastify = require("fastify")({
  logger: true,
});

// Define ShortDUID parameters
const shardId = 0;
const epochStart = 1655012000n * 1000n; // Not so old unix epoch timestamp
const salt = "this is my super secret salt";

// Instantiate short-duid
const duidInstance = new duid.init(shardId, salt, epochStart);
console.log("Node with shard_id #" + shardId + " started.");

// Main route
fastify
  .get("/", async (request, reply) => {
    return { name: "ShortDUID API Server" };
  })
  .get("/nduid/:count?", async (request, reply) => {
    return duidInstance.getDUIDInt(request.params.count || 1);
  })
  .get("/duid/:count?", async (request, reply) => {
    return duidInstance.getDUID(request.params.count || 1);
  });

/**
 * Run the server!
 */
const start = async () => {
  try {
    await fastify.listen({ port: 3000 });
  } catch (err) {
    fastify.log.error(err);
    process.exit(1);
  }
};
start();

And then you should install application with npm install --save and start the server by npm start. Getting fresh id can be done by curl: curl http://localhost:3000/duid/.

More examples

For more examples please see examples folder, which I plan to keep adding to. You are free to contribute more examples.

Projects using ShortDUID

So far I know of none, if you are using it in your project and do not mind sharing this information, please drop me a note at ian@phpb.com, and I will add you to this list.

  • DOSID - is a sister project that implements similar ID generation mechanism but on top of Cloudflare edge network.

Testing

git clone https://github.com/phpb-com/short-duid-js.git && cd short-duid-js && npm install --save-dev
npm test && npm run bench

> short-duid-js@1.2.0 test
> npx mocha --harmony --reporter spec ./test/



  Short DUID
    #getEpochStart()
      ✔ should return set epoch start, for instance #1: 1433116800000
      ✔ should return set epoch start, for instance #2: 1433116800000
      ✔ instance #1 and instance #2 should return same epoch start: 1433116800000
      ✔ should reset custom epoch to zero if given one larger than real epoch
      ✔ should accept custom epoch that is even 1 millisecond in the past
    #getSalt()
      ✔ should return set salt, for instance #1: 39622feb2b3e7aa7208f50f45ec36fd513baadad6977b53295a3b28aeaed4a54
      ✔ should return set salt, for instance #2: 39622feb2b3e7aa7208f50f45ec36fd513baadad6977b53295a3b28aeaed4a54
      ✔ instance #1 and instance #2 should return same salt: 39622feb2b3e7aa7208f50f45ec36fd513baadad6977b53295a3b28aeaed4a54
    #getShardID()
      ✔ should overflow if shard id is set to integer that does not fit in 10 bits: 1024 --> 0
      ✔ should overflow if shard id is set to integer that does not fit in 10 bits: 1025 --> 1
      ✔ should return set shard id for id that fits within 10 bits: 1023 --> 1023
      ✔ should return set shard id for id that fits within 10 bits: 0 --> 0
      ✔ should return set shard id for instance #1: 123
      ✔ should return set shard id for instance #2: 12
      ✔ should return different shard ids for instance #1 and instance #2
    #getDUID()
      ✔ Asked for 1 DUIDs, correctly returns 1 DUIDs
      ✔ Asked for 0 DUIDs, correctly returns 0 DUIDs
      ✔ Asked for 8192 DUIDs, correctly returns 8192 DUIDs
      ✔ Asked for 8193 DUIDs, correctly returns 1 DUIDs
      ✔ should have no duplicates in the returned arrays, 8192 IDs each, and combined. (96ms)
    #getDUIDInt()
      ✔ Asked for 1 Int DUIDs, correctly returns 1 Integer DUIDs
      ✔ Asked for 0 Int DUIDs, correctly returns 0 Integer DUIDs
      ✔ Asked for 8192 Int DUIDs, correctly returns 8192 Integer DUIDs
      ✔ Asked for 8193 Int DUIDs, correctly returns 1 Integer DUIDs
      ✔ should have no duplicates in the returned arrays, 8192 IDs each, and combined.
    DUID with drifting time
      ✔ should return same drift time as given as parameter
      ✔ should generate ID with -7851 millisecond drift into the past from now( 221886083479 ), 930657687468224513 should be numerically smaller than 930657720410288129
      ✔ should consistently generate unique IDs even when time is drifting backwards constantly (68ms)


  28 passing (258ms)


> short-duid-js@1.2.0 bench
> /usr/bin/env node benchmarks/test.js

single DUIDInt generation x 1,950,543 ops/sec ±0.66% (90 runs sampled)
batch of 10 DUIDInt generation x 204,988 ops/sec ±4.84% (96 runs sampled)
single DUID generation x 316,785 ops/sec ±8.43% (90 runs sampled)
batch of 10 DUID generation x 34,976 ops/sec ±0.23% (101 runs sampled)
single DUID generation (1 character salt) x 347,631 ops/sec ±0.28% (96 runs sampled)
batch of 10 DUID generation (1 character salt) x 34,766 ops/sec ±0.34% (97 runs sampled)

TODO

  • [ ] Improve error handling, at the moment most of them are silent and prefer overflow
  • [ ] Add more examples

Contributing

All are welcome to submit issues, ideas, pull requests and patches

License

The MIT License (MIT)

Copyright (c) 2015 Ian Matyssik ian@phpb.com

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.

Package Sidebar

Install

npm i short-duid-js

Weekly Downloads

63

Version

1.2.1

License

MIT

Unpacked Size

78.1 kB

Total Files

20

Last publish

Collaborators

  • phpb_com