1.4.0 • Published 4 months ago

node-steg v1.4.0

Weekly downloads
1
License
ISC
Repository
github
Last release
4 months ago

node-steg

Usage

First import the CreateBuilder helper and constants. import { consts, CreateBuilder } from 'node-steg'; You can also import util if you want to control how verbose it is for debugging purposes. import { util } from 'node-steg';

Now, lets create a builder. let steg = CreateBuilder(); The arguments are CreateBuilder([major version, [minor version]]), if you want to use a specific version. At the time of writing, the default is v1.4.

For just packing a file in and calling it a day,

steg.inputImage('path/to/input/image.ext')
    .outputImage('path/to/output/image2.ext')
    .addFile('path/to/file')
    .save();

And to extract it. If reusing the same steg object, clear() must be called.

steg.clear()
    .inputImage('path/to/output/image2.ext')
    .load()
    .then((secs) => {
      // At this point, you're given a list of extractable items. If you don't care what's in it, or already know, there's a helper function for making it easier to extract everything
      return steg.extractAll(secs);
    })
    .then(() => console.log('Done'));

Supported image formats are PNG and WEBP (both static and animated). When it saves PNGs, it saves at compression 9 (highest). When it saves WEBPs, it saves as lossless+exact (save transparent pixels). For what I hope are obvious reasons, lossy formats can't work.

For animated WEBP images, the syntax is mostly the same. However, when both supplying paths for saving and loading (except for one exception below) you must provide them in the format { frame: <frame number, starting at 0>, path: <path> }.

steg.clear()
    .inputImage({ frame: 0, path: 'animated.webp' })
    .outputImage('out.webp') // This is the one exception where you don't need the special format
    // do things as normal
    .save();

steg.clear()
    .inputImage({ frame: 0, path: 'out.webp' })
    // etc
    .load()
    // etc

There are a number of storage modes available and can be applied separately for both alpha and non-alpha pixels. 3bpp: Stores 3 bits in the pixel, using the LSB of each of the RGB color channels 6bpp: Stores 6 bits using the lower 2 bits of each channel 9bpp: Ditto, but lower 3 bits 12bpp: Lower 4 15bpp: Lower 5 24bpp: Uses the full RGB values. This is handy if you don't care so much about the hiding but do want to use the storage 32bpp: This is a semi-special mode that forces overwriting the full RGBA data of every pixel. This is more implemented for completeness than anything.

There are also modifiers for what's considered alpha vs non-alpha, and which color channels to use if you want to leave one or more untouched.

Below is a full list of what the current steg builder can do. The term "out-of-band" is used to describe information that's needed but not stored in the image(s) and must be found by other means.

Helper or utility

Path Objects

Anywhere below where path is used, unless otherwise noted, can either be passed as a string or as the following object: For image paths:

{
  path: <path string>, // if you're loading a file
  buffer: <Buffer object>, // if you're loading a Buffer
  name: <file name>, // if using a Buffer, it needs to know what name to refer to it as
  map: <map object>, // optional, described below
  map: <path string>, // for ease and consistency
  frame: <frame number, starts at 0> // optional, only used with animated WebP images
}

For map paths:

{
  path: <path>, // if you're loading a file
  buffer: <Buffer object>, // if you're loading a Buffer
  name: <file name> // if you're loading a Buffer from a buffer map via .setBufferMap
  buffer: true // if you're saving to a Buffer
}

clear()

Resets the object for re-use.

dryrun(comp = false)

Switches to doing a dry run of the saving process. Everything is supported, but the save() call doesn't do the final saving. This does not create or modify any files. Any compression it would do is skipped and the full size is used instead. Set comp to true to enable compression. This does create temporary files unless useTempBuffers() is called and runs files through compression (where applicable) as it would during the normal saving process.

realrun()

Switches to doing a real run. This way, if a dry run succeeds, you can call this and do a proper run.

setPasswords(pws)

This is an out-of-band setting. More of a helper function. Pass it an array of passwords to pull from (in order) whenever it needs a password rather than prompting the user.

cliPasswordHandler()

Asks the user for missing passwords via the command line and a silent 'Enter password:' prompt.

setPasswordHandler(f)

Sets f as the password handler. Must return the password to use, either directly or via a Promise.

getPasswordHandler()

Returns the previously-set handler wrapped in a function. The function will throw an error if no handler was set.

setBufferMap(map)

Use map, a map of name/Buffer pairs, for images and maps. Needed when using a Buffer for a map defined in an image table. Otherwise it's mainly for convenience. Example: steg.clear().setBufferMap({ 'image.png': pngBuffer, 'image.map': mapBuffer }).inputImage({ name: 'image.png', map: 'image.map' })... You can alternatively do .inputImage({ name: 'image.png', buffer: pngBuffer, map: { name: 'image.map', buffer: mapBuffer } }) in the above example.

useThreads(b = true)

Enable or disable the use of threads (Workers) when saving. Can greatly speed up using WebP images.

Input/output

inputImage(path)

The input image for both saving and loading. path is a Path Object. If a map is supplied, it's automatically loaded.

outputImage(path)

The output image for saving. path is a Path Object. If a map is supplied, it's saved at the end.

async save()

Saves the image(s).

async load()

Parses the image(s) and returns a list of data sections (see Classes section below).

async extractAll(secs = this.#secs, path = './extracted')

Extract all the sections in secs or, if null/undefined, extract all sections found, to the directory path. If path is a string, this function will return an array with, in order, the contents of each text section. If path is null, this function will return an array with, in order, the contents of each file and text section.

async getLoadOpts(packed = false, enc = false, salt = false, raw = false)

If packed is false, this returns the object representing the options required to load the current Steg instance (such as header mode, global seed, salt, etc). If packed is true, it returns the object as a JSON string with byte 0 being a flag for if it's encrypted. If enc is also true, you'll be prompted for a password to use to encrypt via AES256 and a salt unique to this pair of functions. If salt is true, it uses the current salt provided by .setSalt(), if any. raw is ignored. If raw is false, then salt is a string that is hashed using SHA256. If raw is true, then salt is a hex-encoded 32-byte value that is directly used as the salt. Does NOT support generating a random salt, as the salt must be known. Does not save passwords set by setPasswords().

async setLoadOpts(blob, packed = false, enc = false, salt = false, raw = false)

Loads the appropriate settings defined in blob into this Steg instance so that all that must be supplied are any passwords required and the input path before load() can be used. packed, enc, salt, and raw function the way they do with getLoadOpts().

useTempBuffers()

Use Buffers instead of temp files when handling encrypted/compressed files.

Out-of-band

setHeaderMode(mode)

This sets the mode used to store the first half of the header. It defaults to MODE_A3BPP | MODE_3BPP.

setHeaderModeMask(mask)

This changes which channels are used to store the first half of the header. It defaults to MODEMASK_RGB.

setGlobalSeed(seed)

This uses seed to randomly distribute the header and data around the image. It defaults to disabled. seed is an arbitrary-length string consisting of a-z, A-Z, 0-9, and spaces.

setGlobalShuffleSeed(seed)

This uses seed to randomly shuffle any data written to the image. It defaults to disabled. seed is an arbitrary-length string consisting of a-z, A-Z, 0-9, and spaces.

setInitialCursor(x, y)

This sets the cursor to x, y rather than the default 0, 0. Has no effect if a global seed is enabled.

setSalt(salt, raw = false)

This overrides the internally-defined default salt when using encryption. If raw is false, then salt is a string that is hashed using SHA256. If raw is true, then salt is a hex-encoded 32-byte value that is directly used as the salt. If salt is undefined, then a crypto-safe 32-byte PRNG value is generated and used. The downside to this last option is the only way to obtain the salt is via getLoadOpts().

Header

setGlobalMode(mode)

This sets the mode used to store the second half of the header, as well as the rest of the data in general. It defaults to MODE_A3BPP | MODE_3BPP.

setGlobalModeMask(mask)

This changes which channels are used to store the second half of the header, as well as the rest of the data in general. It defaults to MODEMASK_RGB.

setGlobalAlphaBounds(bounds)

This changes what alpha value is considered alpha vs non-alpha. It supports 8 steps, each roughly 36 apart. Defaults to ALPHA_255.

Sections

setAlphaBounds(bounds)

This changes what alpha value is considered alpha vs non-alpha from what is set globally until another setAlphaBounds() is called or is cleared.

clearAlphaBounds()

Removes the active setAlphaBounds() and returns the alpha value to the global one.

setRect(x, y, width, height)

Bounds all operations within the defined rectangle until another setRect() called or is cleared.

clearRect()

Removes the active setRect().

setMode(mode)

Override the global mode until another setMode() is called or is cleared.

clearMode()

Reset the mode back to the global mode.

setModeMask(mask)

Override the global mode mask until another setModeMask() is called or is cleared.

clearModeMask()

Reset the mode mask back to the global mode mask.

setSeed(seed)

Override the global seed until another setSeed() is called or is cleared.

clearSeed()

Reset the seed back to the global seed. If there was no global seed, this disables the randomness.

setShuffleSeed(seed)

Override the global shuffle seed until another setShuffleSeed() is called or is cleared.

clearShuffleSeed()

Reset the seed back to the global shuffle seed. If there was no global shuffle seed, this disables shuffling.

pushCursor()/popCursor()

Save/load the image index and x, y position of the cursor.

moveCursor(x, y, index = 0)

Move the cursor to x, y in the current image or the one specified by index.

moveImage(index = 0)

Move the cursor to the one specified by index. Doesn't touch the cursor position of the target image. Does nothing if index is already the current image.

setImageTable(inputFiles, outputFiles)

This sets up a table of images you can jump around between with moveCursor(). Both arguments are arrays and must be the same length. Both inputFiles and outputFiles are arrays of Path Objects. It is, however, currently unsupported to mix anim and non-anim WEBP, or mix frames. Example: Each assuming .inputImage({ frame: 0, path: 'in.webp' }).outputImage({ frame: 0, path: 'out.webp' })

  • .setImageTable([ { frame: 1, path: 'in.webp' } ], [ { frame: 1, path: 'out.webp' } ]) This is valid.
  • .setImageTable([ { frame: 4, path: 'in.webp' } ], [ { frame: 1, path: 'out.webp' } ]) This is unsupported. Frame indexes cannot be mismatched.
  • .setImageTable([ 'random.png' ], [ { frame: 1, path: 'out.webp' } ]) This is also unsupported, as is using random.webp for the left side. Cannot mix animated and non-animated.
  • .setImageTable([ { frame: 1, path: 'in.webp' } ], [ 'random.webp' ]) This is also unsupported. Doesn't matter which side, they cannot be mixed.
  • .setImageTable([ { frame: 2, path: 'in.webp' } ], [ { frame: 2, path: 'different.webp' } ]) This is also unsupported, if 'in.webp' is already mapped to another output name. In this case, it's trying to save frame 2 of 'in.webp' (which is already mapped to 'out.webp') to another file. This technically works, but will result in 'out.webp' being duplicated as 'different.webp', rather than the frame copied over.

    The short version is that it only supports modifying frames in the same animation, not replacing or extracting them. See node-webpmux or the official webpmux tool if you need that (I'd recommend node-webpmux as I've got a more-complete toolset than webpmux does in it).

clearImageTable()

Disables the active table and moves the cursor back to the main image. Any images from any previously-active tables will still be written to properly.

setCompression(type, level = 0, text = false)

Set the active compression algorithm to run files/text through. Currently, only COMP_GZIP and COMP_BROTLI are supported.

For GZIP:

  • level must range 0 - 9
  • text is unused

    For BROTLI:

  • level must range 0 - 11

  • text enables BROTLI's special text-mode compression

clearCompression()

Clear an active setCompression().

setEncryption(type, pw = undefined, kdf = KDF_ARGON2ID, { adv = false, memoryCost = 65536, timeCost = 50, parallelism = 8, iterations = 1000000 } = {})

Set the active encryption algorithm to run files/text through. Currently supported algorithms for type:

  • CRYPT_AES256 (AES-256-CBC).
  • CRYPT_CAMELLIA256 (CAMELLIA-256-CBC).
  • CRYPT_ARIA256 (ARIA-256-CBC).
  • CRYPT_CHACHA20 (ChaCha20).
  • CRYPT_BLOWFISH (BF-CBC). kdf sets which KDF (Key Derivative Function) to use. Currently supported KDFs are:
  • KDF_PBKDF2 The old default. Does 1000000 iterations and sha256.
  • KDF_ARGON2I
  • KDF_ARGON2D
  • KDF_ARGON2ID Variants of Argon2. Defaults to Argon2id, memory cost 64MiB, time cost 50, parallelism 8. If adv is set to true...
  • memoryCost, timeCost, and parallelism only affect Argon2
  • iterations only affects PBKDF2

clearEncryption()

Clear an active setEncryption().

Files/text

addFile(source, name, compressed = false)

Add the file at source to the image under the name name. Set compressed to true if the file is already compressed via the active compression mode. If source is a Buffer, that Buffer's data is used as the contents of the file.

addDirectory(path, full = false, recursive = false, compressed = false)

Add the contents of the directory at path to the image. File names are preserved as-is and the basename of path is used as the base path. Example, addDirectory('a/b/c') will add the contents of that directory under c/. Set full to true to add the path names as-is rather than the basename. Example, addDirectory('a/b/c') will then add the contents of that directory under a/b/c/. Set recursive to true to recursively add any other directories under the path. Set compressed to true if ALL files under path are already compressed via the active compression mode.

addPartialFile(source, name, index, compressed = false)

Add the file at source you intend to store in pieces under the name name and index index. Set compressed to true if the file is already compressed via the active compression mode. index can be any integer 0 <= n <= 255 and is used solely for your own reference in addPartialFilePiece(). If source is a Buffer, that Buffer's data is used as the contents of the file.

addPartialFilePiece(index, size = 0, last = false)

Add a piece of file index. If size is 0 or greater than the remaining size of the file, the rest of the file is written and last is assumed true. Set last to true to flag that this is the last piece you intend to write. You can use this if you don't intend to write the entire file.

addText(text, honor = TEXT_HONOR_NONE)

Adds a simple block of text to the image. More simple than a text file. honor is a mask of TEXT_HONOR_ENCRYPTION and TEXT_HONOR_COMPRESSION to control which, if any, are desired to apply to this text block.

Classes

StegFile

  • name: Name of the file.
  • size: Size of the file as it was stored (after compression/encryption).
  • realSize: Uncompressed/decrypted size of the file (only computed after extracting).
  • compressed: Boolean flag for if the file was compressed.
  • encrypted: Boolean flag for if the file was encrypted.
  • async extract(path = './extracted'): Extract the file to path. If path is null, the file is extracted to a Buffer instead, and that Buffer is returned.

StegPartialFile

  • name: Name of the file.
  • size: Size of the file as it was stored (after compression/encryption).
  • realSize: Uncompressed/decrypted size of the file (only computed after extracting).
  • compressed: Boolean flag for if the file was compressed.
  • encrypted: Boolean flag for if the file was encrypted.
  • count: The number of pieces this file is in.
  • async extract(path = './extracted'): Extract the file to path. If path is null, the file is extracted to a Buffer instead, and that Buffer is returned.

StegText

  • size: Size of the text as it was stored (after compression/encryption).
  • realSize: Uncompressed/decrypted size of the text (only computed after extracting).
  • compressed: Boolean flag for if the text was compressed.
  • encrypted: Boolean flag for if the text was encrypted.
  • async extract(): Extracts and returns the text.

Util

util has many things, but for controlling verbosity, only util.Channels, util.debug, and util.setChannel() are important.

debug(v)

Set v to true to enable debug mode, false to disable it, or pass nothing to get the current debug state. This mostly only disables the file extraction progress messages ("Saved x of size"). Does NOT set channel to DEBUG.

Channels

  • SILENT: Outputs nothing at all.
  • NORMAL: Default; Outputs basic information during saving/extracting, such as the number of pixels changed per image and extraction progress during exracting.
  • VERBOSE: Ouputs more detailed information about what it's doing to the image. Mostly useless.
  • VVERBOSE: Outputs even more information about what it's doing, but mostly during loading.
  • DEBUG: Outputs each and every modified pixel of every image it touches.

setChannel(channel)

Sets the output channel to one of the above.

For a full (more technical) description of the format things are stored in the image(s), see the file steg/specs/v<major>.<minor>.spec (like steg/specs/v1.2.spec). Also see test.mjs for more examples in a very ugly file.

Command-line tool

In bin/ is steg.mjs. This is a somewhat simple CLI tool for packing/unpacking.

For packing:

  • -pack (must be first argument) Set to packing mode.
  • -silent Suppress all status and result output.
  • -v Set to VERBOSE output. Outputs extra status messages.
  • -vv Set to VVERBOSE output. Currently identical to -v as packing outputs nothing to the VVERBOSE channel.
  • -debug Set to DEBUG output with debug mode enabled. Outputs everything -vv does, as well as every pixel modified.
  • -version/-ver <version> Set the version wanted in the format <major>.<minor> like 1.2.
  • -headmode/-hm <mode> Set the header mode in the format [non-alpha]/[alpha] like 9/24. Values are the bits-per-pixel (3, 6, 9, 12, 15, 24, 32).
  • -headmodemask/-hmm <mask> Set the header mode mask in the format [r][g][b] like rb.
  • -mode/-m <mode> Set the global mode. Same format as -headmode.
  • -modemask/-mm <mask> Set the global mode mask. Same format as -headmodemask.
  • -salt <salt> [raw] Override the salt with the SHA256 hash of <salt>. If [raw] is provided, <salt> is considered raw.
  • -alpha <threshhold> Set the global alpha threshhold where <threshhold> is a value between 0 and 7 inclusive. The meanings are as follows: 0: alpha 255, 1: 220, 2: 184, 3: 148, 4: 112, 5: 76, 6: 40, 7: 0 This is in line with the ALPHA_* constants.
  • -rand [seed] Set the global seed to the value of [seed] if provided, otherwise generate one to use.
  • -shuffle [seed] Set the global shuffle seed to the value of [seed] if provided, otherwise generate one to use.
  • -dryrun [comp] Set to dryrun mode. If [comp] is set, compress files/text blocks during the dry run.
  • -in <path> [frame] [map] Use <path> as the input image. If [frame] is provided, use that frame of <path>. If [map] is provided, load and use [map] when loading <path>.
  • -out <path> [frame] [map] Use <path> as the output image. If [frame] is provided, use that frame of <path>. If [map] is provided, save the map for <path> as [map].
  • -cursor <x> <y> Set the initial cursor to <x>, <y>.
  • -getloadopts/-glo <path> [enc] Save the load opts to <path>. If [enc] is provided, encrypt the opts.
  • -newsec/-ns <sec> <opts...> Define a new section. and their options are defined below:
    • file <path> [name] [comp] Argument order is important. Save <path> under the name [name] if provided, or the base filename if not. If [comp] is provided, consider this file already compressed.
    • dir <path> [full] [recurse] [comp] If [full] is provided, use the full pathname (minus <path>) as the file name. Otherwise use only the base file name. If [recurse] is provided, add this directory recursively rather than only the files. If [comp] is provided, consider all files to already be compressed.
    • rand [seed] Use the seed [seed] if provided, otherwise generate a new random seed to use.
    • shuffle [seed] Use the seed [seed] if provided, otherwise generate a new random seed to use.
    • imagetable <in1> <out1> [<in2> <out2> [...]] Create an image table using the provided <in> <out> pairs. Each <in> and <out> are in the format [-frame <index>] [-map <map>] <path>
    • rect <x> <y> <w> <h> Limit to the rect defined by x, y, w, h.
    • cursor <cmd> <args...> <cmd> is one of.. push: Push the cursor onto the stack pop: Pop the cursor off of the stack and use it
    • move <x> <y> [index] Move the cursor to x, y of the image at [index] if provided, otherwise use the current image.
    • image [index] Move to the image at [index] and use whatever the cursor last was on it. Does nothing if [index] is the current image. If [index] isn't provided, it returns to the primary image.
    • compress <type> <args...> <type> can be one of.. gzip <level>: Use gzip. <level> is between 0 and 9. brotli <level> [text]: Use Brotli. <level> is between 0 and 11. If [text] is provided, set Brotli into text compression mode.
    • encrypt <type> [ [kdf] [ <adv> [<memory cost> <time cost> <parallelism>] [<iterations>] ] ] <type> can be one of.. aes256: Use AES 256. camellia256: Use CAMELLIA 256. aria256: Use ARIA 256. chacha20: Use ChaCha20. blowfish: Use Blowfish. [kdf] can be one of.. pbkdf2 argon2i argon2d argon2id (default) If [kdf] and <adv> are supplied.. <memory cost>, <time cost>, and <parallelism>: Settings for Argon2. <iterations>: Setting for PBKDF2.
    • partialfile <path> <index> [name] [comp] Define a file at <path> that is to be saved in discreet chunks. <index> is an arbitrary integer to use to refer to it in partialfilepiece blocks. If [name] is defined, use [name] as the filename rather than the base filename. If [comp] is provided, consider the file already compressed.
    • partialfilepiece <index> <size> [final] Define a piece of the partial file <index> and of size <size> bytes. If [final] is provided, this is the final piece that is going to be defined and the file is considered complete.
    • mode <mode> Set a new mode. <mode> is the same format as -headmode.
    • modemask <mask> Set a new mode mask. <mask> is the same format as -headmodemask.
    • alpha <threshhold> Set a new alpha threshhold. <threshhold> is the same format as -alpha.
    • text <text> [honor] Save the text block <text>. If [honor] is defined, set whether compression/encryption should be honored. Format is <encrypt/compress>[/<encrypt/compress>] like encrypt or compress/encrypt.
  • -clearsec/-cs <sec> Clear a section's effects. Valid <sec> are defined below:
    • rand Disable the seed. If a global seed was previously in effect, return to using it.
    • shuffle Disable the seed. If a global shuffle seed was previously in effect, return to using it.
    • imagetable Disable the image table. Any changes are kept and if a new table is defined using any of the previous images, their existing data is preserved.
    • rect Disable the rect limitation and return to using the whole image's bounds.
    • compress Disable compression.
    • encrypt Disable encryption.
    • mode Return to using the global mode.
    • modemask Return to using the global mode mask.
    • alpha Return to using the global alpha threshhold.
  • -save Actually perform the actions defined. Omitting this is useful if you're only interested in using -getloadopts.

For unpacking:

  • -unpack (must be the first argument) Set to unpacking mode.
  • -silent Suppress all status and result output.
  • -v Set to VERBOSE output. Outputs extra status messages.
  • -vv Set to VVERBOSE output. Outputs everything -v does as well as what values were read.
  • -debug Set to DEBUG output with debug mode enabled. Outputs everything -vv does, as well as the values of every pixel read.
  • -headmode/-hm <mode> Same as -headmode of packing.
  • -headmodemask/-hmm <mask> Same as -headmodemask of packing.
  • -image <path> Use <path> as the input image.
  • -rand [seed] Same as -rand of packing.
  • -shuffle [seed] Same as -shuffle of packing.
  • -cursor <x> <y> Same as -cursor of packing.
  • -salt <salt> [raw] Same as -salt of packing.
  • -setloadopts/-slo <path> [enc] Load the load opts from <path>. If [enc] is provided, then treat it as encrypted.
  • -extract <path> Extract the contents to the directory <path> and print any text blocks in full. Omitting this gives a summary of the contents and any text blocks under 100 bytes in size.
1.4.0

4 months ago

1.3.4

3 years ago

1.3.3

3 years ago

1.3.2

3 years ago

1.3.1

3 years ago

1.2.0

3 years ago

1.1.0

4 years ago