diff --git a/.codeclimate.yml b/.codeclimate.yml index 1fd0569..0b9e04c 100644 --- a/.codeclimate.yml +++ b/.codeclimate.yml @@ -1,15 +1,15 @@ engines: eslint: enabled: true - channel: "eslint-8" + channel: 'eslint-8' config: - config: ".eslintrc.yaml" + config: '.eslintrc.yaml' ratings: - paths: - - "**.js" + paths: + - '**.js' checks: method-complexity: config: - threshold: 10 \ No newline at end of file + threshold: 10 diff --git a/.eslintrc.yaml b/.eslintrc.yaml index fe947ea..9fe535c 100644 --- a/.eslintrc.yaml +++ b/.eslintrc.yaml @@ -2,24 +2,10 @@ env: node: true es6: true mocha: true - es2020: true - -plugins: - - haraka + es2022: true extends: - - eslint:recommended - - plugin:haraka/recommended + - '@haraka' rules: - indent: [2, 2, {"SwitchCase": 1}] - -root: true - -globals: - OK: true - CONT: true - DENY: true - DENYSOFT: true - DENYDISCONNECT: true - DENYSOFTDISCONNECT: true + no-unused-vars: 1 diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 3c40dcc..80cf255 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -2,6 +2,7 @@ name: CI on: push: + pull_request: env: CI: true @@ -14,23 +15,8 @@ jobs: # uses: haraka/.github/.github/workflows/coverage.yml@master # secrets: inherit - test: - needs: lint - runs-on: ${{ matrix.os }} - strategy: - matrix: - os: [ ubuntu-latest, windows-latest ] - node-version: [ 14, 16, 18 ] - fail-fast: false + ubuntu: + uses: haraka/.github/.github/workflows/ubuntu.yml@master - steps: - - uses: actions/checkout@v3 - - - uses: actions/setup-node@v3 - name: Node ${{ matrix.node-version }} on ${{ matrix.os }} - with: - node-version: ${{ matrix.node-version }} - - - run: npm install - - - run: npm test + windows: + uses: haraka/.github/.github/workflows/windows.yml@master diff --git a/.github/workflows/codeql.yml b/.github/workflows/codeql.yml index 3627451..8314a66 100644 --- a/.github/workflows/codeql.yml +++ b/.github/workflows/codeql.yml @@ -2,10 +2,10 @@ name: CodeQL on: push: - branches: [ master ] + branches: [master] pull_request: # The branches below must be a subset of the branches above - branches: [ master ] + branches: [master] schedule: - cron: '18 7 * * 4' diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml index d489fbd..e81c15f 100644 --- a/.github/workflows/publish.yml +++ b/.github/workflows/publish.yml @@ -13,4 +13,4 @@ env: jobs: publish: uses: haraka/.github/.github/workflows/publish.yml@master - secrets: inherit \ No newline at end of file + secrets: inherit diff --git a/.prettierrc b/.prettierrc new file mode 100644 index 0000000..8ded5e0 --- /dev/null +++ b/.prettierrc @@ -0,0 +1,2 @@ +singleQuote: true +semi: false diff --git a/.release b/.release index 0890e94..afb1db8 160000 --- a/.release +++ b/.release @@ -1 +1 @@ -Subproject commit 0890e945e4e061c96c7b2ab45017525904c17728 +Subproject commit afb1db801607dda5e859f39b600f0dd0111e4651 diff --git a/Changes.md b/CHANGELOG.md similarity index 61% rename from Changes.md rename to CHANGELOG.md index 22e6776..1b95b3e 100644 --- a/Changes.md +++ b/CHANGELOG.md @@ -1,18 +1,28 @@ +# Changelog + +The format is based on [Keep a Changelog](https://keepachangelog.com/). + ### Unreleased +### [2.0.3] - 2023-12-13 + +- ci: use shared ci workflows +- confirm maxmind db has loaded before registering hook #23 +- populate [files] in package.json +- dep: eslint-plugin-haraka -> @haraka/eslint-config +- lint: remove duplicate / stale rules from .eslintrc +- doc: renamed Changes.md -> CHANGELOG.md ### 2.0.2 - 2023-02-07 - fix: catching DNS timeout exception - ### 2.0.1 - 2022-05-27 - fix: when adding headers, assure ASN is string - fix: when adding header, look in correct location for asn.org - when create conn note, only assign properties with values - ### 2.0.0 - 2022-05-23 - style: replace most callbacks with async/await @@ -20,7 +30,6 @@ - asn.ini: switch default dns provider to rspamd - dep: remove async - ### 1.0.9 - 2022-05-22 - ci: add GitHub Actions CI, #17 @@ -30,7 +39,6 @@ - style: more es6/7 - dep(async): bump version to 3.2 - ### 1.0.8 - 2018-01-22 - parse maxmind ASN w/o Org data @@ -39,18 +47,15 @@ - added parse_rspamd test - emit rspamd DNS provider results (when enabled) - ### 1.0.7 - 2017-02-06 - updated eslint to use eslint-plugin-haraka - aggregate results before emitting - ### 1.0.6 - 2016-10-20 -* when protocols[setting]=false, don't enable that protocol - * ie, do what the config implies - +- when protocols[setting]=false, don't enable that protocol + - ie, do what the config implies ### 1.0.5 - 2016-10-08 @@ -62,5 +67,8 @@ ### 1.0.0 - 2016-07-21 - -[2.0.2]: https://github.com/haraka/haraka-plugin-asn/releases/tag/2.0.2 +[1.0.9]: https://github.com/haraka/haraka-plugin-asn/releases/tag/v1.0.9 +[2.0.0]: https://github.com/haraka/haraka-plugin-asn/releases/tag/v2.0.0 +[2.0.1]: https://github.com/haraka/haraka-plugin-asn/releases/tag/2.0.1 +[2.0.2]: https://github.com/haraka/haraka-plugin-asn/releases/tag/v2.0.2 +[2.0.3]: https://github.com/haraka/haraka-plugin-asn/releases/tag/2.0.3 diff --git a/CONTRIBUTORS.md b/CONTRIBUTORS.md new file mode 100644 index 0000000..028df07 --- /dev/null +++ b/CONTRIBUTORS.md @@ -0,0 +1,8 @@ +# Contributors + +This handcrafted artisinal software is brought to you by: + +|
msimerson (26) |
analogic (1) |
lnedry (1) | +| :------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | + +this file is maintained by [.release](https://github.com/msimerson/.release) diff --git a/README.md b/README.md index a3cc681..cfd26db 100644 --- a/README.md +++ b/README.md @@ -6,21 +6,18 @@ look up ASN from local GeoIP databases and/or DNS based providers. - ### DNS providers -* origin.asn.cymru.com -* asn.routeviews.org -* asn.rspamd.com - +- origin.asn.cymru.com +- asn.routeviews.org +- asn.rspamd.com ### Databases -* MaxMind ASN database +- MaxMind ASN database PS: Run something like [maxmind-geolite-mirror](https://www.npmjs.com/package/maxmind-geolite-mirror) weekly to keep your database files up-to-date. - [ci-img]: https://github.com/haraka/haraka-plugin-asn/actions/workflows/ci.yml/badge.svg [ci-url]: https://github.com/haraka/haraka-plugin-asn/actions/workflows/ci.yml [clim-img]: https://codeclimate.com/github/haraka/haraka-plugin-asn/badges/gpa.svg diff --git a/index.js b/index.js index e25f317..44e5344 100644 --- a/index.js +++ b/index.js @@ -1,232 +1,237 @@ // determine the ASN of the connecting IP -// node built-ins -const dns = require('dns').promises; -const fs = require('fs/promises'); -const path = require('path') - -let test_ip = '66.128.51.163'; -const providers = []; -let conf_providers = [ 'origin.asn.cymru.com', 'asn.routeviews.org', 'asn.rspamd.com' ]; +const dns = require('node:dns').promises +const fs = require('node:fs/promises') +const path = require('node:path') + +let test_ip = '66.128.51.163' +const providers = [] +let conf_providers = [ + 'origin.asn.cymru.com', + 'asn.routeviews.org', + 'asn.rspamd.com', +] exports.register = async function () { - this.registered = false; + this.registered = false - this.load_asn_ini(); + this.load_asn_ini() - await this.test_and_register_dns_providers(); - await this.test_and_register_geoip(); + await this.test_and_register_dns_providers() + await this.test_and_register_geoip() if (this.cfg.header.asn) { - this.register_hook('data_post', 'add_header_asn'); + this.register_hook('data_post', 'add_header_asn') } if (this.cfg.header.provider) { - this.register_hook('data_post', 'add_header_provider'); + this.register_hook('data_post', 'add_header_provider') } } exports.test_and_register_dns_providers = async function () { - if (!this.cfg.protocols.dns) return; // disabled in config + if (!this.cfg.protocols.dns) return // disabled in config for (const zone of conf_providers) { try { const res = await this.get_dns_results(zone, test_ip) if (!res) { - this.logerror(this, `${zone} failed`); - continue; + this.logerror(this, `${zone} failed`) + continue } - this.logdebug(this, `${zone} succeeded`); + this.logdebug(this, `${zone} succeeded`) - if (!providers.includes(zone)) providers.push(zone); - if (this.registered) continue; - this.registered = true; - this.register_hook('lookup_rdns', 'lookup_via_dns'); - } - catch (err) { - this.logerror(this, `zone ${zone} encountered ${err.message}`); + if (!providers.includes(zone)) providers.push(zone) + if (this.registered) continue + this.registered = true + this.register_hook('lookup_rdns', 'lookup_via_dns') + } catch (err) { + this.logerror(this, `zone ${zone} encountered ${err.message}`) } } return providers } exports.load_asn_ini = function () { - const plugin = this; - plugin.cfg = plugin.config.get('asn.ini', { - booleans: [ - '+header.asn', - '-header.provider', - '+protocols.dns', - '+protocols.geoip', - ] - }, - function () { - plugin.load_asn_ini(); - }); - - const c = plugin.cfg; + const plugin = this + plugin.cfg = plugin.config.get( + 'asn.ini', + { + booleans: [ + '+header.asn', + '-header.provider', + '+protocols.dns', + '+protocols.geoip', + ], + }, + function () { + plugin.load_asn_ini() + }, + ) + + const c = plugin.cfg if (c.main.providers !== undefined) { if (c.main.providers === '') { - conf_providers = []; - } - else { - conf_providers = c.main.providers.split(/[\s,;]+/); + conf_providers = [] + } else { + conf_providers = c.main.providers.split(/[\s,;]+/) } } - if (c.main.test_ip) test_ip = c.main.test_ip; - - // backwards compat with old config settings (Sunset 3.0) - if (c.main.asn_header !== undefined) c.header.asn = c.main.asn_header; - if (c.main.provider_header !== undefined) c.header.provider = c.main.provider_header; + if (c.main.test_ip) test_ip = c.main.test_ip } exports.get_dns_results = async function (zone, ip) { - const query = `${ip.split('.').reverse().join('.')}.${zone}`; + const query = `${ip.split('.').reverse().join('.')}.${zone}` const timeout = (prom, time, exception) => { - let timer; + let timer return Promise.race([ prom, - new Promise((_r, rej) => timer = setTimeout(rej, time, exception)) - ]).finally(() => clearTimeout(timer)); + new Promise((_r, rej) => (timer = setTimeout(rej, time, exception))), + ]).finally(() => clearTimeout(timer)) } try { const addrs = await timeout( dns.resolveTxt(query), (this.cfg.main.timeout || 4) * 1000, - new Error(`${zone} timeout`) - ); + new Error(`${zone} timeout`), + ) if (!addrs || !addrs[0]) { - this.logerror(this, `no results for ${query}`); + this.logerror(this, `no results for ${query}`) return } - const first = addrs[0]; + const first = addrs[0] - this.logdebug(this, `${zone} answers: ${first}`); + this.logdebug(this, `${zone} answers: ${first}`) - return this.get_result(zone, first); - } - catch (err) { - this.logerror(this, `error: ${err} running: ${query}`); + return this.get_result(zone, first) + } catch (err) { + this.logerror(this, `error: ${err} running: ${query}`) } - } exports.get_result = function (zone, first) { - switch (zone) { - case 'origin.asn.cymru.com' : return this.parse_cymru(first.join('')); - case 'asn.routeviews.org' : return this.parse_routeviews(first); - case 'asn.rspamd.com' : return this.parse_rspamd(first.join('')); - case 'origin.asn.spameatingmonkey.net': return this.parse_monkey(first.join('')); + case 'origin.asn.cymru.com': + return this.parse_cymru(first.join('')) + case 'asn.routeviews.org': + return this.parse_routeviews(first) + case 'asn.rspamd.com': + return this.parse_rspamd(first.join('')) + case 'origin.asn.spameatingmonkey.net': + return this.parse_monkey(first.join('')) } - this.logerror(this, `unrecognized ASN provider: ${zone}`); - return; + this.logerror(this, `unrecognized ASN provider: ${zone}`) + return } exports.lookup_via_dns = function (next, connection) { - const plugin = this; - if (connection.remote.is_private) return next(); + if (connection.remote.is_private) return next() const promises = [] for (const zone of providers) { - promises.push(new Promise(resolve => { - // connection.logdebug(plugin, `zone: ${zone}`); - - try { - - plugin.get_dns_results(zone, connection.remote.ip).then(r => { - if (!r) return resolve(); - - const results = { emit: true }; - - // store asn & net from any source - if (r.asn) results.asn = r.asn; - if (r.net) results.net = r.net; - - // store provider specific results - switch (zone) { - case 'origin.asn.cymru.com': - results.cymru = r; - break; - case 'asn.routeviews.org': - results.routeviews = r; - break; - case 'origin.asn.spameatingmonkey.net': - results.monkey = r; - break; - case 'asn.rspamd.com': - results.rspamd = r; - break; - } - - connection.results.add(plugin, results); - resolve(results); - }) - } - catch (err) { - connection.results.add(plugin, { err }) - resolve(); - } - })) + promises.push( + new Promise((resolve) => { + // connection.logdebug(plugin, `zone: ${zone}`); + + try { + this.get_dns_results(zone, connection.remote.ip).then((r) => { + if (!r) return resolve() + + const results = { emit: true } + + // store asn & net from any source + if (r.asn) results.asn = r.asn + if (r.net) results.net = r.net + + // store provider specific results + switch (zone) { + case 'origin.asn.cymru.com': + results.cymru = r + break + case 'asn.routeviews.org': + results.routeviews = r + break + case 'origin.asn.spameatingmonkey.net': + results.monkey = r + break + case 'asn.rspamd.com': + results.rspamd = r + break + } + + connection.results.add(this, results) + resolve(results) + }) + } catch (err) { + connection.results.add(this, { err }) + resolve() + } + }), + ) } Promise.all(promises).then(next) } exports.parse_routeviews = function (thing) { - let labels; + let labels if (typeof thing === 'string' && /,/.test(thing)) { - labels = thing.split(','); - return { asn: labels[0], net: `${labels[1]}/${labels[2]}` }; + labels = thing.split(',') + return { asn: labels[0], net: `${labels[1]}/${labels[2]}` } } // this is a correct result (node >= 0.10.26) // 99.177.75.208.asn.routeviews.org. IN TXT "40431" "208.75.176.0" "21" if (Array.isArray(thing)) { - labels = thing; - } - else { + labels = thing + } else { // this is what node (< 0.10.26) returns // 99.177.75.208.asn.routeviews.org. IN TXT "40431208.75.176.021" - labels = thing.split(/ /); + labels = thing.split(/ /) } if (labels.length !== 3) { - this.logerror(this, `result length not 3: ${labels.length} string="${thing}"`); - return; + this.logerror( + this, + `result length not 3: ${labels.length} string="${thing}"`, + ) + return } - return { asn: labels[0], net: `${labels[1]}/${labels[2]}` }; + return { asn: labels[0], net: `${labels[1]}/${labels[2]}` } } exports.parse_cymru = function (str) { - const r = str.split(/\s+\|\s*/); + const r = str.split(/\s+\|\s*/) // 99.177.75.208.origin.asn.cymru.com. 14350 IN TXT // "40431 | 208.75.176.0/21 | US | arin | 2007-03-02" // "10290 | 12.129.48.0/24 | US | arin |" if (r.length < 4) { - this.logerror(this, `cymru: bad result length ${r.length} string="${str}"`); - return; + this.logerror(this, `cymru: bad result length ${r.length} string="${str}"`) + return } - return { asn: r[0], net: r[1], country: r[2], assignor: r[3], date: r[4] }; + return { asn: r[0], net: r[1], country: r[2], assignor: r[3], date: r[4] } } exports.parse_monkey = function (str) { - const plugin = this; - const r = str.split(/\s+\|\s+/); + const plugin = this + const r = str.split(/\s+\|\s+/) // "74.125.44.0/23 | AS15169 | Google Inc. | 2000-03-30" // "74.125.0.0/16 | AS15169 | Google Inc. | 2000-03-30 | US" if (r.length < 3) { - plugin.logerror(plugin, `monkey: bad result length ${r.length} string="${str}"`); - return; + plugin.logerror( + plugin, + `monkey: bad result length ${r.length} string="${str}"`, + ) + return } return { asn: r[1].substring(2), @@ -234,81 +239,81 @@ exports.parse_monkey = function (str) { org: r[2], date: r[3], country: r[4], - }; + } } exports.parse_rspamd = function (str) { - const plugin = this; - const r = str.split(/\s*\|\s*/); + const plugin = this + const r = str.split(/\s*\|\s*/) // 8.8.8.8.asn.rspamd.com. 14350 IN TXT // "15169|8.8.8.0/24|US|arin|" if (r.length < 4) { - plugin.logerror(plugin, `rspamd: bad result length ${r.length} string="${str}"`); - return; + plugin.logerror( + plugin, + `rspamd: bad result length ${r.length} string="${str}"`, + ) + return } - return { asn: r[0], net: r[1], country: r[2], assignor: r[3], date: r[4] }; + return { asn: r[0], net: r[1], country: r[2], assignor: r[3], date: r[4] } } exports.add_header_asn = function (next, connection) { - - const asn = connection.results.get('asn'); - if (!asn?.asn) return next(); - if (!connection.transaction) return next(); + const asn = connection.results.get('asn') + if (!asn?.asn) return next() + if (!connection.transaction) return next() if (asn.net) { - connection.transaction.add_header('X-Haraka-ASN', `${asn.asn} ${asn.net}`); - } - else { - connection.transaction.add_header('X-Haraka-ASN', `${asn.asn}`); + connection.transaction.add_header('X-Haraka-ASN', `${asn.asn} ${asn.net}`) + } else { + connection.transaction.add_header('X-Haraka-ASN', `${asn.asn}`) } if (asn.org) { - connection.transaction.add_header('X-Haraka-ASN-Org', `${asn.org}`); + connection.transaction.add_header('X-Haraka-ASN-Org', `${asn.org}`) } - next(); + next() } exports.add_header_provider = function (next, connection) { - - const asn = connection.results.get('asn'); - if (!asn?.asn) return next(); + const asn = connection.results.get('asn') + if (!asn?.asn) return next() for (const p in asn) { - if (!asn[p].asn) continue; // ignore non-object results + if (!asn[p].asn) continue // ignore non-object results - const name = `X-Haraka-ASN-${p.toUpperCase()}`; - const values = []; + const name = `X-Haraka-ASN-${p.toUpperCase()}` + const values = [] for (const k in asn[p]) { - values.push(`${k}=${asn[p][k]}`); + values.push(`${k}=${asn[p][k]}`) } - if (values.length === 0) continue; - connection.transaction.add_header(name, values.join(' ')); + if (values.length === 0) continue + connection.transaction.add_header(name, values.join(' ')) } - next(); + next() } exports.test_and_register_geoip = async function () { - if (!this.cfg.protocols.geoip) return; // disabled in config + if (!this.cfg.protocols.geoip) return // disabled in config try { - this.maxmind = require('maxmind'); + this.maxmind = require('maxmind') if (await this.load_dbs()) { - this.register_hook('connect', 'lookup_via_maxmind'); + this.register_hook('connect', 'lookup_via_maxmind') } - } - catch (e) { - this.logerror(e); - this.logerror("unable to load maxmind, try\n\n\t'npm install -g maxmind@0.6'\n\n"); + } catch (e) { + this.logerror(e) + this.logerror( + "unable to load maxmind, try\n\n\t'npm install -g maxmind@0.6'\n\n", + ) } } exports.load_dbs = async function () { - - this.dbsLoaded = 0; - const dbdir = this.cfg.main.dbdir || '/usr/local/share/GeoIP/'; - const dbPath = path.join(dbdir, `GeoLite2-ASN.mmdb`); + this.dbsLoaded = 0 + const dbdir = this.cfg.main.dbdir || '/usr/local/share/GeoIP/' + const dbPath = path.join(dbdir, `GeoLite2-ASN.mmdb`) try { await fs.access(dbPath) @@ -318,14 +323,13 @@ exports.load_dbs = async function () { watchForUpdates: true, cache: { max: 1000, // max items in cache - maxAge: 1000 * 60 * 60 // life time in milliseconds - } + maxAge: 1000 * 60 * 60, // life time in milliseconds + }, }) - this.loginfo(`loaded maxmind db ${dbPath}`); - this.dbsLoaded++; - } - catch (e) { + this.loginfo(`loaded maxmind db ${dbPath}`) + this.dbsLoaded++ + } catch (e) { console.error(e) this.loginfo(`missing [access to] DB ${dbPath}`) } @@ -334,15 +338,19 @@ exports.load_dbs = async function () { } exports.lookup_via_maxmind = function (next, connection) { - if (!this.maxmind || !this.dbsLoaded) return next(); + if (!this.maxmind || !this.dbsLoaded) return next() const asn = this.lookup.get(connection.remote.ip) if (asn?.autonomous_system_number || asn?.autonomous_system_organization) { connection.results.add(this, { - ...(asn.autonomous_system_number ? { asn: asn.autonomous_system_number } : {}), - ...(asn.autonomous_system_organization ? { org: asn.autonomous_system_organization } : {}), - }); + ...(asn.autonomous_system_number + ? { asn: asn.autonomous_system_number } + : {}), + ...(asn.autonomous_system_organization + ? { org: asn.autonomous_system_organization } + : {}), + }) } - next(); + next() } diff --git a/package.json b/package.json index 425d9cd..49e297a 100644 --- a/package.json +++ b/package.json @@ -1,13 +1,21 @@ { "name": "haraka-plugin-asn", - "version": "2.0.2", + "version": "2.0.3", "description": "look up ASN of remote MTA", "main": "index.js", + "files": [ + "CHANGELOG.md", + "config" + ], "scripts": { - "test": "npx mocha --exit", - "lint": "npx eslint *.js test", - "lintfix": "npx eslint --fix *.js test", - "versions": "npx dependency-version-checker check" + "format": "npm run prettier:fix && npm run lint:fix", + "lint": "npx eslint@^8 *.js test", + "lint:fix": "npx eslint@^8 *.js test --fix", + "prettier": "npx prettier . --check", + "prettier:fix": "npx prettier . --write --log-level=warn", + "test": "npx mocha@^10 --exit", + "versions": "npx dependency-version-checker check", + "versions:fix": "npx dependency-version-checker update" }, "repository": { "type": "git", @@ -26,12 +34,10 @@ }, "homepage": "https://github.com/haraka/haraka-plugin-asn#readme", "dependencies": { - "maxmind": "^4.3.6" + "maxmind": "^4.3.19" }, "devDependencies": { - "eslint": ">=8", - "eslint-plugin-haraka": ">=1.0.14", - "haraka-test-fixtures": ">=1.0.34", - "mocha": ">=9" + "@haraka/eslint-config": "^1.1.5", + "haraka-test-fixtures": "^1.3.7" } } diff --git a/test/asn.js b/test/asn.js index 28f7f83..874209b 100644 --- a/test/asn.js +++ b/test/asn.js @@ -1,191 +1,194 @@ -'use strict'; +'use strict' -// node build-in modules -const assert = require('assert'); +const assert = require('node:assert') // npm installed modules -const fixtures = require('haraka-test-fixtures'); +const fixtures = require('haraka-test-fixtures') describe('parse_monkey', function () { - - const asn = new fixtures.plugin('asn'); + const asn = new fixtures.plugin('asn') it('parses AS 15169/23', function () { assert.deepEqual( asn.parse_monkey('74.125.44.0/23 | AS15169 | Google Inc. | 2000-03-30'), - { net: '74.125.44.0/23', asn: '15169', org: 'Google Inc.', - date: '2000-03-30', country: undefined, - } + { + net: '74.125.44.0/23', + asn: '15169', + org: 'Google Inc.', + date: '2000-03-30', + country: undefined, + }, ) }) it('parses AS 15169/16', function () { assert.deepEqual( - asn.parse_monkey('74.125.0.0/16 | AS15169 | Google Inc. | 2000-03-30 | US'), - { net: '74.125.0.0/16', asn: '15169', org: 'Google Inc.', - date: '2000-03-30', country: 'US', - } + asn.parse_monkey( + '74.125.0.0/16 | AS15169 | Google Inc. | 2000-03-30 | US', + ), + { + net: '74.125.0.0/16', + asn: '15169', + org: 'Google Inc.', + date: '2000-03-30', + country: 'US', + }, ) }) }) describe('parse_routeviews', function () { - - const asn = new fixtures.plugin('asn'); + const asn = new fixtures.plugin('asn') it('40431 string, asn-only', function () { - assert.deepEqual( - asn.parse_routeviews('40431'), - undefined - ) + assert.deepEqual(asn.parse_routeviews('40431'), undefined) }) it('40431 string', function () { - assert.deepEqual( - asn.parse_routeviews('40431 208.75.176.0 21'), - { - asn: '40431', net: '208.75.176.0/21', - } - ); - }); + assert.deepEqual(asn.parse_routeviews('40431 208.75.176.0 21'), { + asn: '40431', + net: '208.75.176.0/21', + }) + }) it('15169 CSV string', function () { - assert.deepEqual( - asn.parse_routeviews('15169,8.8.8.0,24'), { asn: '15169', net: '8.8.8.0/24' } - ) + assert.deepEqual(asn.parse_routeviews('15169,8.8.8.0,24'), { + asn: '15169', + net: '8.8.8.0/24', + }) }) it('40431 array', function () { - assert.deepEqual( - asn.parse_routeviews(['40431','208.75.176.0','21']), - { asn: '40431', net: '208.75.176.0/21' } - ) + assert.deepEqual(asn.parse_routeviews(['40431', '208.75.176.0', '21']), { + asn: '40431', + net: '208.75.176.0/21', + }) }) }) describe('parse_cymru', function () { - - const asn = new fixtures.plugin('asn'); + const asn = new fixtures.plugin('asn') it('40431', function () { assert.deepEqual( asn.parse_cymru('40431 | 208.75.176.0/21 | US | arin | 2007-03-02'), - { asn: '40431', net: '208.75.176.0/21', country: 'US', - assignor: 'arin', date: '2007-03-02', - } - ); - }); + { + asn: '40431', + net: '208.75.176.0/21', + country: 'US', + assignor: 'arin', + date: '2007-03-02', + }, + ) + }) it('10290', function () { - assert.deepEqual( - asn.parse_cymru('10290 | 12.129.48.0/24 | US | arin |'), - { asn: '10290', net: '12.129.48.0/24', country: 'US', - assignor: 'arin', date: '', - } - ) + assert.deepEqual(asn.parse_cymru('10290 | 12.129.48.0/24 | US | arin |'), { + asn: '10290', + net: '12.129.48.0/24', + country: 'US', + assignor: 'arin', + date: '', + }) }) }) describe('parse_rspamd', function () { - - const asn = new fixtures.plugin('asn'); + const asn = new fixtures.plugin('asn') it('40431', function () { - assert.deepEqual( - asn.parse_rspamd('15169|8.8.8.0/24|US|arin|'), - { - asn: '15169', net: '8.8.8.0/24', country: 'US', - assignor: 'arin', date: '', - } - ) + assert.deepEqual(asn.parse_rspamd('15169|8.8.8.0/24|US|arin|'), { + asn: '15169', + net: '8.8.8.0/24', + country: 'US', + assignor: 'arin', + date: '', + }) }) }) describe('get_dns_results', function () { - - const asn = new fixtures.plugin('asn'); - asn.cfg = { main: { }, protocols: { dns: true } }; - asn.connection = fixtures.connection.createConnection(); + const asn = new fixtures.plugin('asn') + asn.cfg = { main: {}, protocols: { dns: true } } + asn.connection = fixtures.connection.createConnection() it('origin.asn.cymru.com', function (done) { - this.timeout(4000) - asn.get_dns_results('origin.asn.cymru.com', '8.8.8.8').then(obj => { + this.timeout(5000) + asn.get_dns_results('origin.asn.cymru.com', '8.8.8.8').then((obj) => { if (obj) { - assert.equal('15169', obj.asn); - assert.equal('8.8.8.0/24', obj.net); - } - else { - assert.equal('something', obj); + assert.equal('15169', obj.asn) + assert.equal('8.8.8.0/24', obj.net) + } else { + assert.equal('something', obj) } - done(); + done() }) }) it('asn.routeviews.org', function (done) { - asn.get_dns_results('asn.routeviews.org', '8.8.8.8').then(obj => { + this.timeout(5000) + asn.get_dns_results('asn.routeviews.org', '8.8.8.8').then((obj) => { if (obj) { if (obj.asn && obj.asn === '15169') { - assert.equal('15169', obj.asn); + assert.equal('15169', obj.asn) } + } else { + assert.ok('Node DNS (c-ares) bug') } - else { - assert.ok("Node DNS (c-ares) bug"); - } - done(); - }); - }); + done() + }) + }) it('asn.rspamd.com', function (done) { - this.timeout(3000); + this.timeout(5000) asn.get_dns_results('asn.rspamd.com', '8.8.8.8').then((obj, zone) => { if (obj) { - assert.equal('15169', obj.asn); - assert.equal('8.8.8.0/24', obj.net); + assert.equal('15169', obj.asn) + assert.equal('8.8.8.0/24', obj.net) + } else { + assert.equal('something', obj) } - else { - assert.equal('something', obj); - } - done(); + done() }) }) it('origin.asn.spameatingmonkey.net', (done) => { - this.timeout(3000); - asn.get_dns_results('origin.asn.spameatingmonkey.net', '8.8.8.8').then((obj, zone) => { - if (obj) { - assert.equal('15169', obj.asn); - assert.equal('8.8.8.0/24', obj.net); - } - else { - assert.equal('something', obj); - } - done(); - }) + this.timeout(5000) + asn + .get_dns_results('origin.asn.spameatingmonkey.net', '8.8.8.8') + .then((obj, zone) => { + if (obj) { + assert.equal('15169', obj.asn) + assert.equal('8.8.8.0/24', obj.net) + } else { + assert.equal('something', obj) + } + done() + }) }) }) describe('lookup_via_dns', function () { it('returns results from active providers', function (done) { - const asn = new fixtures.plugin('asn'); - asn.cfg = { main: { }, protocols: { dns: true } }; - const connection = fixtures.connection.createConnection(); - connection.remote.ip='66.128.51.163'; + this.timeout(5000) + const asn = new fixtures.plugin('asn') + asn.cfg = { main: {}, protocols: { dns: true } } + const connection = fixtures.connection.createConnection() + connection.remote.ip = '66.128.51.163' asn.test_and_register_dns_providers().then((providers) => { asn.lookup_via_dns((r) => { assert.ok(r.length) done() - }, - connection) + }, connection) }) }) }) describe('maxmind geoip db', () => { - it('test_and_register_geoip', (done) => { - const asn = new fixtures.plugin('asn'); - asn.cfg = { main: { }, protocols: { geoip: true } }; + const asn = new fixtures.plugin('asn') + asn.cfg = { main: {}, protocols: { geoip: true } } asn.test_and_register_geoip().then((r) => { // console.log(r) assert.ok(asn.maxmind) @@ -194,44 +197,40 @@ describe('maxmind geoip db', () => { }) it('lookup_via_maxmind, IPv4', (done) => { - const asn = new fixtures.plugin('asn'); - asn.cfg = { main: { }, protocols: { geoip: true } }; - asn.connection = fixtures.connection.createConnection(); - asn.connection.remote.ip='8.8.8.8'; + const asn = new fixtures.plugin('asn') + asn.cfg = { main: {}, protocols: { geoip: true } } + asn.connection = fixtures.connection.createConnection() + asn.connection.remote.ip = '8.8.8.8' asn.test_and_register_geoip().then(() => { asn.lookup_via_maxmind(() => { if (asn.dbsLoaded) { - const res = asn.connection.results.get('asn'); - assert.equal(res.asn, 15169); - assert.equal(res.org, 'GOOGLE'); - } - else { - console.error('no DBs found'); + const res = asn.connection.results.get('asn') + assert.equal(res.asn, 15169) + assert.equal(res.org, 'GOOGLE') + } else { + console.error('no DBs found') } - done(); - }, - asn.connection); + done() + }, asn.connection) }) }) it('maxmind AS with org', (done) => { - const asn = new fixtures.plugin('asn'); - asn.cfg = { main: { }, protocols: { geoip: true } }; - asn.connection = fixtures.connection.createConnection(); - asn.connection.remote.ip='1.1.1.1'; + const asn = new fixtures.plugin('asn') + asn.cfg = { main: {}, protocols: { geoip: true } } + asn.connection = fixtures.connection.createConnection() + asn.connection.remote.ip = '1.1.1.1' asn.test_and_register_geoip().then(() => { try { asn.lookup_via_maxmind(() => { if (asn.dbsLoaded) { - const res = asn.connection.results.get('asn'); - assert.equal(res?.asn, 13335); - assert.equal(res?.org, 'CLOUDFLARENET'); + const res = asn.connection.results.get('asn') + assert.equal(res?.asn, 13335) + assert.equal(res?.org, 'CLOUDFLARENET') } done() - }, - asn.connection); - } - catch (e) { + }, asn.connection) + } catch (e) { console.error(e) done() }