パッケージの詳細

clamscan

kylefarris605.6kMIT2.4.0

Use Node JS to scan files on your server with ClamAV's clamscan/clamdscan binary or via TCP to a remote server or local UNIX Domain socket. This is especially useful for scanning uploaded files provided by un-trusted sources.

clamav, virus, clamscan, upload

readme

NodeJS Clamscan Virus Scanning Utility

NPM Version NPM Downloads Node.js Version Test Suite

Use Node JS to scan files on your server with ClamAV's clamscan/clamdscan binary or via TCP to a remote server or local UNIX Domain socket. This is especially useful for scanning uploaded files provided by un-trusted sources.

!!IMPORTANT

If you are using a version prior to 1.2.0, please upgrade! There was a security vulnerability in previous versions that can cause false negative in some edge cases. Specific details on how the attack could be implemented will not be disclosed here. Please update to 1.2.0 or greater ASAP. No breaking changes are included, only the security patch.

All older versions in NPM have been deprecated.

Version 1.0.0 Information

If you are migrating from v0.8.5 or less to v1.0.0 or greater, please read the release notes as there are some breaking changes (but also some awesome new features!).

Table of Contents

Dependencies

To use local binary method of scanning

You will need to install ClamAV's clamscan binary and/or have clamdscan daemon running on your server. On linux, it's quite simple.

Fedora-based distros:

sudo yum install clamav

Debian-based distros:

sudo apt-get install clamav clamav-daemon

For OS X, you can install clamav with brew:

sudo brew install clamav

To use ClamAV using TCP sockets

You will need access to either:

  1. A local UNIX Domain socket for a local instance of clamd

  2. Follow instructions in To use local binary method of scanning.

  3. Socket file is usually: /var/run/clamd.scan/clamd.sock
  4. Make sure clamd is running on your local server

  5. A local/remote clamd daemon

  6. Must know the port the daemon is running on

  7. If running on remote server, you must have the IP address/domain name
  8. If running on remote server, it's firewall must have the appropriate TCP port(s) open
  9. Make sure clamd is running on your local/remote server

NOTE: This module is not intended to work on a Windows server. This would be a welcome addition if someone wants to add that feature (I may get around to it one day but have no urgent need for this).

How to Install

npm install clamscan

License Info

Licensed under the MIT License:

Getting Started

All of the values listed in the example below represent the default values for their respective configuration item.

You can simply do this:

const NodeClam = require('clamscan');
const ClamScan = new NodeClam().init();

And, you'll be good to go.

BUT: If you want more control, you can specify all sorts of options.

const NodeClam = require('clamscan');
const ClamScan = new NodeClam().init({
    removeInfected: false, // If true, removes infected files
    quarantineInfected: false, // False: Don't quarantine, Path: Moves files to this place.
    scanLog: null, // Path to a writeable log file to write scan results into
    debugMode: false, // Whether or not to log info/debug/error msgs to the console
    fileList: null, // path to file containing list of files to scan (for scanFiles method)
    scanRecursively: true, // If true, deep scan folders recursively
    clamscan: {
        path: '/usr/bin/clamscan', // Path to clamscan binary on your server
        db: null, // Path to a custom virus definition database
        scanArchives: true, // If true, scan archives (ex. zip, rar, tar, dmg, iso, etc...)
        active: true // If true, this module will consider using the clamscan binary
    },
    clamdscan: {
        socket: false, // Socket file for connecting via TCP
        host: false, // IP of host to connect to TCP interface
        port: false, // Port of host to use when connecting via TCP interface
        timeout: 60000, // Timeout for scanning files
        localFallback: true, // Use local preferred binary to scan if socket/tcp fails
        path: '/usr/bin/clamdscan', // Path to the clamdscan binary on your server
        configFile: null, // Specify config file if it's in an unusual place
        multiscan: true, // Scan using all available cores! Yay!
        reloadDb: false, // If true, will re-load the DB on every call (slow)
        active: true, // If true, this module will consider using the clamdscan binary
        bypassTest: false, // Check to see if socket is available when applicable
        tls: false, // Use plaintext TCP to connect to clamd
    },
    preference: 'clamdscan' // If clamdscan is found and active, it will be used by default
});

Here is a non-default values example (to help you get an idea of what proper-looking values could be):

const NodeClam = require('clamscan');
const ClamScan = new NodeClam().init({
    removeInfected: true, // Removes files if they are infected
    quarantineInfected: '~/infected/', // Move file here. removeInfected must be FALSE, though.
    scanLog: '/var/log/node-clam', // You're a detail-oriented security professional.
    debugMode: true, // This will put some debug info in your js console
    fileList: '/home/webuser/scanFiles.txt', // path to file containing list of files to scan
    scanRecursively: false, // Choosing false here will save some CPU cycles
    clamscan: {
        path: '/usr/bin/clam', // I dunno, maybe your clamscan is just call "clam"
        scanArchives: false, // Choosing false here will save some CPU cycles
        db: '/usr/bin/better_clam_db', // Path to a custom virus definition database
        active: false // you don't want to use this at all because it's evil
    },
    clamdscan: {
        socket: '/var/run/clamd.scan/clamd.sock', // This is pretty typical
        host: '127.0.0.1', // If you want to connect locally but not through socket
        port: 12345, // Because, why not
        timeout: 300000, // 5 minutes
        localFallback: false, // Do no fail over to binary-method of scanning
        path: '/bin/clamdscan', // Special path to the clamdscan binary on your server
        configFile: '/etc/clamd.d/daemon.conf', // A fairly typical config location
        multiscan: false, // You hate speed and multi-threaded awesome-sauce
        reloadDb: true, // You want your scans to run slow like with clamscan
        active: false, // you don't want to use this at all because it's evil
        bypassTest: true, // Don't check to see if socket is available. You should probably never set this to true.
        tls: true, // Connect to clamd over TLS
    },
    preference: 'clamscan' // If clamscan is found and active, it will be used by default
});

NOTE: If a valid port is provided but no host value is provided, the clamscan will assume 'localhost' for host.

A note about using this module via sockets or TCP

As of version v1.0.0, this module supports communication with a local or remote ClamAV daemon through Unix Domain sockets or a TCP host/port combo. If you supply both in your configuration object, the UNIX Domain socket option will be used. The module will not not fallback to using the alternative Host/Port method. If you wish to connect via Host/Port and not a Socket, please either omit the socket property in the config object or use socket: null.

If you specify a valid clamscan/clamdscan binary in your config and you set clamdscan.localFallback: true in your config, this module will fallback to the traditional way this module has worked--using a binary directly/locally.

Also, there are some caveats to using the socket/tcp based approach:

  • The following configuration items are not honored (unless the module falls back to binary method):

    • removeInfected - remote clamd service config will dictate this
    • quarantineInfected - remote clamd service config will dictate this
    • scanLog - remote clamd service config will dictate this
    • fileList - this simply won't be available
    • clamscan.db - only available on fallback
    • clamscan.scanArchives - only available on fallback
    • clamscan.path - only available on fallback
    • clamdscan.configFile - only available on fallback
    • clamdscan.path - only available on fallback

Basic Usage Example

For the sake of brevity, all the examples in the API section will be shortened to just the relevant parts related specifically to that example. In those examples, we'll assume you already have an instance of the clamscan object. Since initializing the module returns a promise, you'll have to resolve that promise to get an instance of the clamscan object.

Below is the full example of how you could get that instance and run some methods:

const NodeClam = require('clamscan');
const ClamScan = new NodeClam().init(options);

// Get instance by resolving ClamScan promise object
ClamScan.then(async clamscan => {
    try {
        // You can re-use the `clamscan` object as many times as you want
        const version = await clamscan.getVersion();
        console.log(`ClamAV Version: ${version}`);

        const {isInfected, file, viruses} = await clamscan.isInfected('/some/file.zip');
        if (isInfected) console.log(`${file} is infected with ${viruses}!`);
    } catch (err) {
        // Handle any errors raised by the code in the try block
    }
}).catch(err => {
    // Handle errors that may have occurred during initialization
});

If you're writing your code within an async function, getting an instance can be one less step:

const NodeClam = require('clamscan');

async some_function() {
    try {
        // Get instance by resolving ClamScan promise object
        const clamscan = await new NodeClam().init(options);
        const {goodFiles, badFiles} = await clamscan.scanDir('/foo/bar');
    } catch (err) {
        // Handle any errors raised by the code in the try block
    }
}

some_function();

API

Complete/functional examples for various use-cases can be found in the examples folder.

.getVersion([callback])

This method allows you to determine the version of ClamAV you are interfacing with. It supports a callback and Promise API. If no callback is supplied, a Promise will be returned.

Parameters

  • callback (function) (optional) Will be called when the scan is complete. It receives 2 parameters:

    • err (object or null) A standard javascript Error object (null if no error)
    • version (string) The version of the clamav server you're interfacing with

Returns

  • Promise

    • Promise resolution returns: version (string) The version of the clamav server you're interfacing with

Callback Example

clamscan.getVersion((err, version) => {
    if (err) return console.error(err);
    console.log(`ClamAV Version: ${version}`);
});

Promise Example

clamscan.getVersion().then(version => {
    console.log(`ClamAV Version: ${version}`);
}).catch(err => {
    console.error(err);
});

.isInfected(filePath[,callback])

This method allows you to scan a single file. It supports a callback and Promise API. If no callback is supplied, a Promise will be returned. This method will likely be the most common use-case for this module.

Alias

.scan_file

Parameters

  • filePath (string) Represents a path to the file to be scanned.
  • callback (function) (optional) Will be called when the scan is complete. It takes 3 parameters:

    • err (object or null) A standard javascript Error object (null if no error)
    • file (string) The original filePath passed into the isInfected method.
    • isInfected (boolean) True: File is infected; False: File is clean. NULL: Unable to scan.
    • viruses (array) An array of any viruses found in the scanned file.

Returns

  • Promise

    • Promise resolution returns: result (object):

      • file (string) The original filePath passed into the isInfected method.
      • isInfected (boolean) True: File is infected; False: File is clean. NULL: Unable to scan.
      • viruses (array) An array of any viruses found in the scanned file.

Callback Example

clamscan.isInfected('/a/picture/for_example.jpg', (err, file, isInfected, viruses) => {
    if (err) return console.error(err);

    if (isInfected) {
        console.log(`${file} is infected with ${viruses.join(', ')}.`);
    }
});

Promise Example

clamscan.isInfected('/a/picture/for_example.jpg').then(result => {
    const {file, isInfected, viruses} =  result;
    if (isInfected) console.log(`${file} is infected with ${viruses.join(', ')}.`);
}).then(err => {
    console.error(err);
})

Async/Await Example

const {file, isInfected, viruses} = await clamscan.isInfected('/a/picture/for_example.jpg');

.scanDir(dirPath[,endCallback[,fileCallback]])

Allows you to scan an entire directory for infected files. This obeys your recursive option even for clamdscan which does not have a native way to turn this feature off. If you have multiple paths, send them in an array to scanFiles.

TL;DR: For maximum speed, don't supply a fileCallback.

If you choose to supply a fileCallback, the scan will run a little bit slower (depending on number of files to be scanned) for clamdscan. If you are using clamscan, while it will work, I'd highly advise you to NOT pass a fileCallback... it will run incredibly slow.

NOTE

The goodFiles parameter of the endCallback callback in this method will only contain the directory that was scanned in all but the following scenarios:

  • A fileCallback callback is provided, and scanRecursively is set to true.
  • The scanner is set to clamdscan and scanRecursively is set to false.
  • The scanned directory contains 1 or more viruses. In this case, the goodFiles array will be empty.

There will, however, be a total count of the good files which is calculated by determining the total number of files scanned and subtracting the number of bad files from that count. We simply can't provide a list of all good files due to the potential large memory usage implications of scanning a directory with, for example, millions of files.

Parameters

  • dirPath (string) (required) Full path to the directory to scan.
  • endCallback (function) (optional) Will be called when the entire directory has been completely scanned. This callback takes 3 parameters:

    • err (object) A standard javascript Error object (null if no error)
    • goodFiles (array) An empty array if path is infected. An array containing the directory name that was passed in if clean.
    • badFiles (array) List of the full paths to all files that are infected.
    • viruses (array) List of all the viruses found (feature request: associate to the bad files).
    • numGoodFiles (number) Number of files that were found to be clean.
  • fileCallback (function) (optional) Will be called after each file in the directory has been scanned. This is useful for keeping track of the progress of the scan. This callback takes 3 parameters:

    • err (object or null) A standard Javascript Error object (null if no error)
    • file (string) Path to the file that just got scanned.
    • isInfected (boolean) True: File is infected; False: File is clean. NULL: Unable to scan file.

Returns

  • Promise

    • Promise resolution returns: result (object):

      • path (string) The original dir_path passed into the scanDir method.
      • isInfected (boolean) True: File is infected; False: File is clean. NULL: Unable to scan.
      • goodFiles (array) An empty array if path is infected. An array containing the directory name that was passed in if clean.
      • badFiles (array) List of the full paths to all files that are infected.
      • viruses (array) List of all the viruses found (feature request: associate to the bad files).
      • numGoodFiles (number) Number of files that were found to be clean.

Callback Example

clamscan.scanDir('/some/path/to/scan', (err, goodFiles, badFiles, viruses, numGoodFiles) {
    if (err) return console.error(err);

    if (badFiles.length > 0) {
        console.log(`${path} was infected. The offending files (${badFiles.join (', ')}) have been quarantined.`);
        console.log(`Viruses Found: ${viruses.join(', ')}`);
    } else {
        console.log(`${goodFiles[0]} looks good! ${numGoodFiles} file scanned and no problems found!.`);
    }
});

Promise Example

clamscan.scanDir('/some/path/to/scan').then(results => {
    const { path, isInfected, goodFiles, badFiles, viruses, numGoodFiles } = results;
    //...
}).catch(err => {
    return console.error(err);
});

Async/Await Example

const { path, isInfected, goodFiles, badFiles, viruses, numGoodFiles } = await clamscan.scanDir('/some/path/to/scan');

.scanFiles(files[,endCallback[,fileCallback]])

This allows you to scan many files that might be in different directories or maybe only certain files of a single directory. This is essentially a wrapper for isInfected that simplifies the process of scanning many files or directories.

Parameters

  • files (array) (optional) A list of strings representing full paths to files you want scanned. If not supplied, the module will check for a fileList config option. If neither is found, the method will throw an error.
  • endCallback (function) (optional) Will be called when the entire list of files has been completely scanned. This callback takes 3 parameters:

    • err (object or null) A standard JavaScript Error object (null if no error)
    • goodFiles (array) List of the full paths to all files that are clean.
    • badFiles (array) List of the full paths to all files that are infected.
  • fileCallback (function) (optional) Will be called after each file in the list has been scanned. This is useful for keeping track of the progress of the scan. This callback takes 3 parameters:

    • err (object or null) A standard JavaScript Error object (null if no error)
    • file (string) Path to the file that just got scanned.
    • isInfected (boolean) True: File is infected; False: File is clean. NULL: Unable to scan file.

Returns

  • Promise

    • Promise resolution returns: result (object):

      • goodFiles (array) List of the full paths to all files that are clean.
      • badFiles (array) List of the full paths to all files that are infected.
      • errors (object) Per-file errors keyed by the filename in which the error happened. (ex. {'foo.txt': Error})
      • viruses (array) List of all the viruses found (feature request: associate to the bad files).

Callback Example

const scan_status = { good: 0, bad: 0 };
const files = [
    '/path/to/file/1.jpg',
    '/path/to/file/2.mov',
    '/path/to/file/3.rb'
];
clamscan.scanFiles(files, (err, goodFiles, badFiles, viruses) => {
    if (err) return console.error(err);
    if (badFiles.length > 0) {
        console.log({
            msg: `${goodFiles.length} files were OK. ${badFiles.length} were infected!`,
            badFiles,
            goodFiles,
            viruses,
        });
    } else {
        res.send({msg: "Everything looks good! No problems here!."});
    }
}, (err, file, isInfected, viruses) => {
    ;(isInfected ? scan_status.bad++ : scan_status.good++);
    console.log(`${file} is ${(isInfected ? `infected with ${viruses}` : 'ok')}.`);
    console.log('Scan Status: ', `${(scan_status.bad + scan_status.good)}/${files.length}`);
});

Promise Example

Note: There is currently no way to get per-file notifications with the Promise API.

clamscan.scanFiles(files).then(results => {
    const { goodFiles, badFiles, errors, viruses } = results;
    // ...
}).catch(err => {
    console.error(err);
})

Async/Await Example

const { goodFiles, badFiles, errors, viruses } = await clamscan.scanFiles(files);

Scanning files listed in fileList

If this modules is configured with a valid path to a file containing a newline-delimited list of files, it will use the list in that file when scanning if the first paramter passed is falsy.

Files List Document:

/some/path/to/file.zip
/some/other/path/to/file.exe
/one/more/file/to/scan.rb

Script:

const ClamScan = new NodeClam().init({
    fileList: '/path/to/fileList.txt'
});

ClamScan.then(async clamscan => {
    // Supply nothing to first parameter to use `fileList`
    const { goodFiles, badFiles, errors, viruses } = await clamscan.scanFiles();
});

.scanStream(stream[,callback])

This method allows you to scan a binary stream. NOTE: This method will only work if you've configured the module to allow the use of a TCP or UNIX Domain socket. In other words, this will not work if you only have access to a local ClamAV binary.

Parameters

  • stream (stream) A readable stream object
  • callback (function) (optional) Will be called after the stream has been scanned (or attempted to be scanned):

    • err (object or null) A standard JavaScript Error object (null if no error)
    • isInfected (boolean) True: Stream is infected; False: Stream is clean. NULL: Unable to scan file.

Returns

  • Promise

    • Promise resolution returns: result (object):

      • file (string) NULL as no file path can be provided with the stream
      • isInfected (boolean) True: File is infected; False: File is clean. NULL: Unable to scan.
      • viruses (array) An array of any viruses found in the scanned file.

Examples

Callback Example:

const NodeClam = require('clamscan');

// You'll need to specify your socket or TCP connection info
const clamscan = new NodeClam().init({
    clamdscan: {
        socket: '/var/run/clamd.scan/clamd.sock',
        host: '127.0.0.1',
        port: 3310,
    }
});
const Readable = require('stream').Readable;
const rs = Readable();

rs.push('foooooo');
rs.push('barrrrr');
rs.push(null);

clamscan.scanStream(stream, (err, { isInfected. viruses }) => {
    if (err) return console.error(err);
    if (isInfected) return console.log('Stream is infected! Booo!', viruses);
    console.log('Stream is not infected! Yay!');
});

Promise Example:

clamscan.scanStream(stream).then(({isInfected}) => {
    if (isInfected) return console.log("Stream is infected! Booo!");
    console.log("Stream is not infected! Yay!");
}).catch(err => {
    console.error(err);
};

Promise Example:

const { isInfected, viruses } = await clamscan.scanStream(stream);

.passthrough()

The passthrough method returns a PassthroughStream object which allows you pipe a ReadbleStream through it and on to another output. In the case of this module's passthrough implementation, it's actually forking the data to also go to ClamAV via TCP or Domain Sockets. Each data chunk is only passed on to the output if that chunk was successfully sent to and received by ClamAV. The PassthroughStream object returned from this method has a special event that is emitted when ClamAV finishes scanning the streamed data so that you can decide if there's anything you need to do with the final output destination (ex. delete a file or S3 object).

In typical, non-passthrough setups, a file is uploaded to the local filesytem and then subsequently scanned. With that setup, you have to wait for the upload to complete and then wait again for the scan to complete. Using this module's passthrough method, you could theoretically speed up user uploads intended to be scanned by up to 2x because the files are simultaneously scanned and written to any WriteableStream output (examples: filesystem, S3, gzip, etc...).

As for these theoretical gains, your mileage my vary and I'd love to hear feedback on this to see where things can still be improved.

Please note that this method is different than all the others in that it returns a PassthroughStream object and does not support a Promise or Callback API. This makes sense once you see the example below (a practical working example can be found in the examples directory of this module):

Example

const NodeClam = require('clamscan');

// You'll need to specify your socket or TCP connection info
const clamscan = new NodeClam().init({
    clamdscan: {
        socket: '/var/run/clamd.scan/clamd.sock',
        host: '127.0.0.1',
        port: 3310,
    }
});

// For example's sake, we're using the Axios module
const axios = require('Axios');

// Get a readable stream for a URL request
const input = axios.get(some_url);

// Create a writable stream to a local file
const output = fs.createWriteStream(some_local_file);

// Get instance of this module's PassthroughStream object
const av = clamscan.passthrough();

// Send output of Axios stream to ClamAV.
// Send output of Axios to `some_local_file` if ClamAV receives data successfully
input.pipe(av).pipe(output);

// What happens when scan is completed
av.on('scan-complete', result => {
   const { isInfected, viruses } = result;
   // Do stuff if you want
});

// What happens when data has been fully written to `output`
output.on('finish', () => {
    // Do stuff if you want
});

// NOTE: no errors (or other events) are being handled in this example but standard errors will be emitted according to NodeJS's Stream specifications

.ping()

This method checks to see if the remote/local socket is working. It supports a callback and Promise API. If no callback is supplied, a Promise will be returned. This method can be used for healthcheck purposes and is already implicitly used during scan.

Parameters

  • callback (function) (optional) Will be called after the ping:

    • err (object or null) A standard JavaScript Error object (null if no error)
    • client (object) A copy of the Socket/TCP client

Returns

  • Promise

    • Promise resolution returns: client (object): A copy of the Socket/TCP client

Examples

Callback Example:

const NodeClam = require('clamscan');

// You'll need to specify your socket or TCP connection info
const clamscan = new NodeClam().init({
    clamdscan: {
        socket: '/var/run/clamd.scan/clamd.sock',
        host: '127.0.0.1',
        port: 3310,
    }
});

clamscan.ping((err, client) => {
    if (err) return console.error(err);
    console.log('ClamAV is still working!');
    client.end();
});

Promise Example:

clamscan.ping().then((client) => {
    console.log('ClamAV is still working!');
    client.end();
}).catch(err => {
    console.error(err);
};

Promise Example:

const client = await clamscan.ping();
client.end();

Contribute

Got a missing feature you'd like to use? Found a bug? Go ahead and fork this repo, build the feature and issue a pull request.

Resources used to help develop this module

更新履歴

Changes

This file is a manually maintained list of changes for each release. Feel free to add your changes here when sending pull requests. Also send corrections if you spot any mistakes.

0.2.1

  • ClamAV returns an exit code 1 when it detects a virus but exec was interpreting that response as an error. Checking the response with type-sensitive equivalence resolves this bug.

0.2.2

  • Fixed documentation

0.4.0 (2014-11-19)

  • Corrected the installation instructions for clamav. Thank you @jshamley!
  • Fixed major bug preventing the scan_dir method from working properly
  • Corrected documentation describing how to instantiate this module.

0.5.0 (2014-12-19)

  • Deprecated the quarantine_path option. Please only use quarantine_infected for now on.
  • Updated documentation to reflect above change.

0.6.0 (2015-01-02)

NOTE: There are some breaking changes on this release. Since this is still a pre-version 1 release, I decided to only do a minor bump to 0.4.0

  • The ability to run "forked" instances of clamscan has been removed because of irregularities with different systems--namely if you had max_forks set to 3, it would sometimes only scan the first or last file in the group... not good.
  • Added the ability to use clamdscan. This ultimately negates the downside of removing the forking capability mentioned in item one. This is a really big improvement (many orders of magnitude) if your system has access to the clamdscan daemon.
  • Added a file_list option allowing one to specify a text file that lists (one per line) paths to files to be scanned. This is great if you need to scan hundreds or thousands of random files.
  • clam_path option has been moved to clam.path
  • db option has been moved to clam.db
  • scan_archives option has been moved to clam.scan_archives
  • scan_files now supports directories as well and will obey your scan_recursively option.

0.6.1 (2015-01-05)

  • Updated description in package.json file.

0.6.2 (2015-01-05)

  • Fixed major bug in the scan_files method that was causing it to only scan half the files passed to it.

0.6.3 (2015-01-05)

  • Removed the unnecessary "index_old.js" file put there for reference during the 0.5.0 -> 0.6.0 semi-rewrite.

0.6.4 (2015-01-26)

  • Fixed error messages

0.7.0 (2015-06-01)

  • Fixed a bug caused by not passing a file_cb paramter to the scan_file method. Thanks nicolaspeixoto!
  • Added tests
  • Fixed poor validation of method parameters
  • Changed API of scan_dir such that the paramaters passed to the end_cb are different in certain defined situations. See the "NOTE" section of the scan_dir documentation for details.
  • Changed err paramter in all callbacks from a simple string to a proper javascript Error object.
  • Added documentation for how to use a file_list file for scanning.

0.7.1 (2015-06-05)

  • Added node dependency of > 0.12 to package.json file

0.8.0 (2015-06-05)

  • Removed item causing node > 0.12 dependency.
  • Removed dependency of node > 0.12 in package.json file.

0.8.1 (2015-06-09)

  • Fixed check for database file. Issue #6

0.8.2 (2015-08-14)

  • Updated to execFile instead of exec
  • Improved test suite

0.9.0-beta (2015-07-01) - Never Released

  • Added support for TCP/UNIX Domain socket communication to local or remote clamav services.
  • Added a get_version method.
  • NULL is now returned to the third parameter of the is_infected when file is neither infected or clean (i.e. on unexpected response)
  • Created alias: scan_file for is_infected.
  • Created a scan_stream method.
  • Minor code clean-up

1.0.0 (2019-05-02)

This is a huge major release in which this module was essentially completely re-written. This version introduces some breaking changes and major new features. Please read the release notes below carefully.

  • Now requires at least Node v10.0.0
  • Code re-written in ES2018 code
  • Now supports a hybrid Promise/Callback API (supports async/await)
  • Now properly supports TCP/UNIX Domain socket communication to local or remote clamav services (with optional fallback to local binary via child process).
  • Added new scan_stream method which allows you to pass an input stream.
  • Added new get_version method which allows you to check the version of ClamAV that you'll be communicating with.
  • Added new passthrough method which allows you to pipe a stream "through" the clamscan module and on to another destination (ex. S3).
  • Added new alias scan_file that points to is_infected.
  • In order to provide the name of any viruses found, a new standard viruses array is now be provided to the callback for:

    • is_infected & scan_file methods (callback format: (err, file, is_infected, viruses) => { ... }).
    • scan_files method (callback format: (err, good_files, bad_files, error_files, viruses) => { ... }).
    • scan_dir method (callback format: (err, good_files, bad_files, viruses) => { ... }).
  • In all cases, the viruses parameter will be an empty array on error or when no viruses are found.

  • scan_files now has another additional parameter in its callback:

    • error_files: An object keyed by the filenames that presented errors while scanning. The value of those keys will be the error message for that file.
  • Introduces new API to instantiate the module (NOTE: The old way will no longer work! See below for more info).

API Changes with 1.0.0:

For some full-fledged examples of how the new API works, checkout the /examples directory in the module root directory.

Module Initialization

Pre-1.0.0
const clamscan = require('clamscan')(options);
1.0.0

NOTE: Due to the new asynchronous nature of the checks that are performed upon initialization of the module, the initialization method now returns a Promise instead of the actual instantiated object. Resolving the Promise with then will return the object like before.

const NodeClam = require('clamscan');
const ClamScan = new NodeClam().init(options);

Making Method Calls

Pre-1.0.0
clamscan.is_infected('/path/to/file.txt', (err, file, is_infected) => {
    // Do stuff
});
1.0.0
ClamScan.then(clamscan => {
    clamscan.is_infected('/path/to/file.txt', (err, file, is_infected, viruses) => {
        // Do stuff
    });
});

If you prefer the async/await style of coding:

;(async () => {
    const clamscan = await new NodeClam().init(options);
    clamscan.is_infected('/path/to/file.txt', (err, file, is_infected, viruses) => {
        // Do stuff
    });
})();

New Way to Get Results

Pre-1.0.0

The only way to get results/errors in pre-1.0.0 was through callbacks.

const clamscan = require('clamscan')(options);
clamscan.scan_dir('/path/to/directory', (err, good_files, bad_files) => {
    // Do stuff inside callback
});
1.0.0

In version 1.0.0 and beyond, you will now be able to use Promises as well (and, of course, async/await).

Promises
const ClamScan = new NodeClam().init(options);
ClamScan.then(clamscan =>
    clamscan.scan_dir('/path/to/directory').then(result => {
        const {good_files, bad_files} = result;
        // Do stuff
    }).catch(err => {
        // Handle scan error
    });
}).catch(err => {
    // Handle initialization error
});
Async/Await
;(async () => {
    try {
        const clamscan = await new NodeClam().init(options);
        const {good_files, bad_files} = await clamscan.scan_dir('/path/to/directory');
        // Do stuff
    } catch (err) {
        // Handle any error
    }
})();

New Methods

scan_stream

The scan_stream method allows you supply a readable stream to have it scanned. Theoretically any stream can be scanned this way. Like all methods, it supports callback and Promise response styles (full documentation is in README).

Basic Promise (async/await) Example:
;(async () => {
    try {
        const clamscan = await new NodeClam().init(options);
        const stream = new Readable();
        rs.push('foooooo');
        rs.push('barrrrr');
        rs.push(null);

        const {is_infected, viruses} = await clamscan.scan_stream(stream);

        // Do stuff
    } catch (err) {
        // Handle any error
    }
})();
Basic Callback Example:
;(async () => {
    try {
        const clamscan = await new NodeClam().init(options);
        const stream = new Readable();
        rs.push('foooooo');
        rs.push('barrrrr');
        rs.push(null);

        clamscan.scan_stream(stream, (err, results)  => {
            if (err) {
                // Handle error
            } else {
                const {is_infected, viruses} = results;
                // Do stuff
            }
        });

        // Do stuff
    } catch (err) {
        // Handle any error
    }
})();
passthrough

The passthrough method allows you supply a readable stream that will be "passed-through" the clamscan module and onto another destination. In reality, the passthrough method works more like a fork stream whereby the input stream is simultaneously streamed to ClamAV and whatever is the next destination. Events are created when ClamAV is done and/or when viruses are detected so that you can decide what to do with the data on the next destination (delete if virus detected, for instance). Data is only passed through to the next generation if the data has been successfully received by ClamAV. If anything halts the data going to ClamAV (including issues caused by ClamAV), the entire pipeline is halted and events are fired.

Normally, a file is uploaded and then scanned. This method should theoretically speed up user uploads intended to be scanned by up to 2x because the files are simultaneously scanned and written to disk. Your mileage my vary.

This method is different than all the others in that it returns a PassthroughStream object and does not support a Promise or Callback API. This makes sense once you see the example below (full documentation is in README).

Basic Example:
;(async () => {
    try {
        const clamscan = await new NodeClam().init(options);
        const request = require('request');
        const input = request.get(some_url);
        const output = fs.createWriteStream(some_local_file);
        const av = clamscan.passthrough();

        // Send output of RequestJS stream to ClamAV.
        // Send output of RequestJS to `some_local_file` if ClamAV receives data successfully
        input.pipe(av).pipe(output);

        // What happens when scan is completed
        av.on('scan-complete', result => {
            const {is_infected, viruses} = result;
            // Do stuff if you want
        });

        // What happens when data has been fully written to `output`
        output.on('finish', () => {
            // Do stuff if you want
        });
    } catch (err) {
        // Handle any error
    }
})();

1.2.0

SECURITY PATCH

An important security patch was released in this version which fixes a bug causing false negatives in specific edge cases. Please upgrade immediately and only use this version from this point on.

All older versions of this package have been deprecated on NPM.

1.3.0

This just has some bug fixes and updates to dependencies. Technically, a new 'timeout' event was added to the passthrough stream method, but, its not fully fleshed out and doesn't seem to work so it will remain undocumented for now.

1.4.0

  • Updated Mocha to v8.1.1. Subsequently, the oldest version of NodeJS allowed for this module is now v10.12.0.
  • Fixed issue with the method not throwing errors when testing existence and viability of remote/local socket.

1.4.1

All sockets clients should now close when they are done being used, fail, or timeout.

1.4.2

  • Fixed initialization to pass a config-file option during clamav version check
  • Added new contributor
  • Fixed tests

Newer Versions

Please see the GitHub Release page for this project to see changelog info starting with v2.0.0.