mirror of
https://github.com/FlipsideCrypto/eth-phishing-detect.git
synced 2026-02-06 11:16:45 +00:00
832 lines
24 KiB
JavaScript
832 lines
24 KiB
JavaScript
(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()({1:[function(require,module,exports){
|
|
const why = require('./why')
|
|
|
|
window.addEventListener('load', function() {
|
|
document.querySelector('form').addEventListener('submit', (ev) => {
|
|
ev.preventDefault()
|
|
result.innerText = why(input.value)
|
|
})
|
|
})
|
|
|
|
|
|
},{"./why":2}],2:[function(require,module,exports){
|
|
const PhishingDetector = require('eth-phishing-detect/src/detector')
|
|
const fetch = require('isomorphic-fetch')
|
|
let phishing
|
|
|
|
updatePhishingList()
|
|
.then(() => {
|
|
detector = new PhishingDetector(phishing)
|
|
})
|
|
|
|
async function updatePhishingList () {
|
|
// make request
|
|
let response
|
|
try {
|
|
response = await fetch('https://api.infura.io/v2/blacklist')
|
|
} catch (err) {
|
|
console.error(err)
|
|
return
|
|
}
|
|
// parse response
|
|
let rawResponse
|
|
try {
|
|
const rawResponse = await response.text()
|
|
phishing = JSON.parse(rawResponse)
|
|
} catch (err) {
|
|
console.error(new Error(`BlacklistController - failed to parse blacklist:\n${rawResponse}`))
|
|
return
|
|
}
|
|
}
|
|
|
|
/*
|
|
* A function for checking why a given domain is blocked.
|
|
* @param { string } domain - The domain that is blocked.
|
|
* @returns { string } reason - A string describing the reason for the block.
|
|
*/
|
|
function why (domain) {
|
|
if (!detector) {
|
|
return 'Cannot answer, still loading list data...'
|
|
}
|
|
|
|
const reason = detector.check(domain)
|
|
|
|
if (!reason.result) {
|
|
return 'This domain is not blocked! No problem here.'
|
|
}
|
|
|
|
if (reason.type === 'fuzzy') {
|
|
return `This domain was blocked for its similarity to ${reason.match}, a historical phishing target.`
|
|
}
|
|
|
|
if (reason.type === 'blacklist') {
|
|
return `This domain was blocked because it has been explicitly identified as a malicious site.`
|
|
}
|
|
|
|
return `There was an issue identifying the reason for the block. The data is ${JSON.stringify(reason)}`
|
|
}
|
|
|
|
module.exports = why
|
|
|
|
|
|
},{"eth-phishing-detect/src/detector":6,"isomorphic-fetch":4}],3:[function(require,module,exports){
|
|
(function() {
|
|
'use strict';
|
|
|
|
var collator;
|
|
try {
|
|
collator = (typeof Intl !== "undefined" && typeof Intl.Collator !== "undefined") ? Intl.Collator("generic", { sensitivity: "base" }) : null;
|
|
} catch (err){
|
|
console.log("Collator could not be initialized and wouldn't be used");
|
|
}
|
|
// arrays to re-use
|
|
var prevRow = [],
|
|
str2Char = [];
|
|
|
|
/**
|
|
* Based on the algorithm at http://en.wikipedia.org/wiki/Levenshtein_distance.
|
|
*/
|
|
var Levenshtein = {
|
|
/**
|
|
* Calculate levenshtein distance of the two strings.
|
|
*
|
|
* @param str1 String the first string.
|
|
* @param str2 String the second string.
|
|
* @param [options] Additional options.
|
|
* @param [options.useCollator] Use `Intl.Collator` for locale-sensitive string comparison.
|
|
* @return Integer the levenshtein distance (0 and above).
|
|
*/
|
|
get: function(str1, str2, options) {
|
|
var useCollator = (options && collator && options.useCollator);
|
|
|
|
var str1Len = str1.length,
|
|
str2Len = str2.length;
|
|
|
|
// base cases
|
|
if (str1Len === 0) return str2Len;
|
|
if (str2Len === 0) return str1Len;
|
|
|
|
// two rows
|
|
var curCol, nextCol, i, j, tmp;
|
|
|
|
// initialise previous row
|
|
for (i=0; i<str2Len; ++i) {
|
|
prevRow[i] = i;
|
|
str2Char[i] = str2.charCodeAt(i);
|
|
}
|
|
prevRow[str2Len] = str2Len;
|
|
|
|
var strCmp;
|
|
if (useCollator) {
|
|
// calculate current row distance from previous row using collator
|
|
for (i = 0; i < str1Len; ++i) {
|
|
nextCol = i + 1;
|
|
|
|
for (j = 0; j < str2Len; ++j) {
|
|
curCol = nextCol;
|
|
|
|
// substution
|
|
strCmp = 0 === collator.compare(str1.charAt(i), String.fromCharCode(str2Char[j]));
|
|
|
|
nextCol = prevRow[j] + (strCmp ? 0 : 1);
|
|
|
|
// insertion
|
|
tmp = curCol + 1;
|
|
if (nextCol > tmp) {
|
|
nextCol = tmp;
|
|
}
|
|
// deletion
|
|
tmp = prevRow[j + 1] + 1;
|
|
if (nextCol > tmp) {
|
|
nextCol = tmp;
|
|
}
|
|
|
|
// copy current col value into previous (in preparation for next iteration)
|
|
prevRow[j] = curCol;
|
|
}
|
|
|
|
// copy last col value into previous (in preparation for next iteration)
|
|
prevRow[j] = nextCol;
|
|
}
|
|
}
|
|
else {
|
|
// calculate current row distance from previous row without collator
|
|
for (i = 0; i < str1Len; ++i) {
|
|
nextCol = i + 1;
|
|
|
|
for (j = 0; j < str2Len; ++j) {
|
|
curCol = nextCol;
|
|
|
|
// substution
|
|
strCmp = str1.charCodeAt(i) === str2Char[j];
|
|
|
|
nextCol = prevRow[j] + (strCmp ? 0 : 1);
|
|
|
|
// insertion
|
|
tmp = curCol + 1;
|
|
if (nextCol > tmp) {
|
|
nextCol = tmp;
|
|
}
|
|
// deletion
|
|
tmp = prevRow[j + 1] + 1;
|
|
if (nextCol > tmp) {
|
|
nextCol = tmp;
|
|
}
|
|
|
|
// copy current col value into previous (in preparation for next iteration)
|
|
prevRow[j] = curCol;
|
|
}
|
|
|
|
// copy last col value into previous (in preparation for next iteration)
|
|
prevRow[j] = nextCol;
|
|
}
|
|
}
|
|
return nextCol;
|
|
}
|
|
|
|
};
|
|
|
|
// amd
|
|
if (typeof define !== "undefined" && define !== null && define.amd) {
|
|
define(function() {
|
|
return Levenshtein;
|
|
});
|
|
}
|
|
// commonjs
|
|
else if (typeof module !== "undefined" && module !== null && typeof exports !== "undefined" && module.exports === exports) {
|
|
module.exports = Levenshtein;
|
|
}
|
|
// web worker
|
|
else if (typeof self !== "undefined" && typeof self.postMessage === 'function' && typeof self.importScripts === 'function') {
|
|
self.Levenshtein = Levenshtein;
|
|
}
|
|
// browser main thread
|
|
else if (typeof window !== "undefined" && window !== null) {
|
|
window.Levenshtein = Levenshtein;
|
|
}
|
|
}());
|
|
|
|
|
|
},{}],4:[function(require,module,exports){
|
|
// the whatwg-fetch polyfill installs the fetch() function
|
|
// on the global object (window or self)
|
|
//
|
|
// Return that as the export for use in Webpack, Browserify etc.
|
|
require('whatwg-fetch');
|
|
module.exports = self.fetch.bind(self);
|
|
|
|
},{"whatwg-fetch":5}],5:[function(require,module,exports){
|
|
(function (global, factory) {
|
|
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports) :
|
|
typeof define === 'function' && define.amd ? define(['exports'], factory) :
|
|
(factory((global.WHATWGFetch = {})));
|
|
}(this, (function (exports) { 'use strict';
|
|
|
|
var support = {
|
|
searchParams: 'URLSearchParams' in self,
|
|
iterable: 'Symbol' in self && 'iterator' in Symbol,
|
|
blob:
|
|
'FileReader' in self &&
|
|
'Blob' in self &&
|
|
(function() {
|
|
try {
|
|
new Blob();
|
|
return true
|
|
} catch (e) {
|
|
return false
|
|
}
|
|
})(),
|
|
formData: 'FormData' in self,
|
|
arrayBuffer: 'ArrayBuffer' in self
|
|
};
|
|
|
|
function isDataView(obj) {
|
|
return obj && DataView.prototype.isPrototypeOf(obj)
|
|
}
|
|
|
|
if (support.arrayBuffer) {
|
|
var viewClasses = [
|
|
'[object Int8Array]',
|
|
'[object Uint8Array]',
|
|
'[object Uint8ClampedArray]',
|
|
'[object Int16Array]',
|
|
'[object Uint16Array]',
|
|
'[object Int32Array]',
|
|
'[object Uint32Array]',
|
|
'[object Float32Array]',
|
|
'[object Float64Array]'
|
|
];
|
|
|
|
var isArrayBufferView =
|
|
ArrayBuffer.isView ||
|
|
function(obj) {
|
|
return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1
|
|
};
|
|
}
|
|
|
|
function normalizeName(name) {
|
|
if (typeof name !== 'string') {
|
|
name = String(name);
|
|
}
|
|
if (/[^a-z0-9\-#$%&'*+.^_`|~]/i.test(name)) {
|
|
throw new TypeError('Invalid character in header field name')
|
|
}
|
|
return name.toLowerCase()
|
|
}
|
|
|
|
function normalizeValue(value) {
|
|
if (typeof value !== 'string') {
|
|
value = String(value);
|
|
}
|
|
return value
|
|
}
|
|
|
|
// Build a destructive iterator for the value list
|
|
function iteratorFor(items) {
|
|
var iterator = {
|
|
next: function() {
|
|
var value = items.shift();
|
|
return {done: value === undefined, value: value}
|
|
}
|
|
};
|
|
|
|
if (support.iterable) {
|
|
iterator[Symbol.iterator] = function() {
|
|
return iterator
|
|
};
|
|
}
|
|
|
|
return iterator
|
|
}
|
|
|
|
function Headers(headers) {
|
|
this.map = {};
|
|
|
|
if (headers instanceof Headers) {
|
|
headers.forEach(function(value, name) {
|
|
this.append(name, value);
|
|
}, this);
|
|
} else if (Array.isArray(headers)) {
|
|
headers.forEach(function(header) {
|
|
this.append(header[0], header[1]);
|
|
}, this);
|
|
} else if (headers) {
|
|
Object.getOwnPropertyNames(headers).forEach(function(name) {
|
|
this.append(name, headers[name]);
|
|
}, this);
|
|
}
|
|
}
|
|
|
|
Headers.prototype.append = function(name, value) {
|
|
name = normalizeName(name);
|
|
value = normalizeValue(value);
|
|
var oldValue = this.map[name];
|
|
this.map[name] = oldValue ? oldValue + ', ' + value : value;
|
|
};
|
|
|
|
Headers.prototype['delete'] = function(name) {
|
|
delete this.map[normalizeName(name)];
|
|
};
|
|
|
|
Headers.prototype.get = function(name) {
|
|
name = normalizeName(name);
|
|
return this.has(name) ? this.map[name] : null
|
|
};
|
|
|
|
Headers.prototype.has = function(name) {
|
|
return this.map.hasOwnProperty(normalizeName(name))
|
|
};
|
|
|
|
Headers.prototype.set = function(name, value) {
|
|
this.map[normalizeName(name)] = normalizeValue(value);
|
|
};
|
|
|
|
Headers.prototype.forEach = function(callback, thisArg) {
|
|
for (var name in this.map) {
|
|
if (this.map.hasOwnProperty(name)) {
|
|
callback.call(thisArg, this.map[name], name, this);
|
|
}
|
|
}
|
|
};
|
|
|
|
Headers.prototype.keys = function() {
|
|
var items = [];
|
|
this.forEach(function(value, name) {
|
|
items.push(name);
|
|
});
|
|
return iteratorFor(items)
|
|
};
|
|
|
|
Headers.prototype.values = function() {
|
|
var items = [];
|
|
this.forEach(function(value) {
|
|
items.push(value);
|
|
});
|
|
return iteratorFor(items)
|
|
};
|
|
|
|
Headers.prototype.entries = function() {
|
|
var items = [];
|
|
this.forEach(function(value, name) {
|
|
items.push([name, value]);
|
|
});
|
|
return iteratorFor(items)
|
|
};
|
|
|
|
if (support.iterable) {
|
|
Headers.prototype[Symbol.iterator] = Headers.prototype.entries;
|
|
}
|
|
|
|
function consumed(body) {
|
|
if (body.bodyUsed) {
|
|
return Promise.reject(new TypeError('Already read'))
|
|
}
|
|
body.bodyUsed = true;
|
|
}
|
|
|
|
function fileReaderReady(reader) {
|
|
return new Promise(function(resolve, reject) {
|
|
reader.onload = function() {
|
|
resolve(reader.result);
|
|
};
|
|
reader.onerror = function() {
|
|
reject(reader.error);
|
|
};
|
|
})
|
|
}
|
|
|
|
function readBlobAsArrayBuffer(blob) {
|
|
var reader = new FileReader();
|
|
var promise = fileReaderReady(reader);
|
|
reader.readAsArrayBuffer(blob);
|
|
return promise
|
|
}
|
|
|
|
function readBlobAsText(blob) {
|
|
var reader = new FileReader();
|
|
var promise = fileReaderReady(reader);
|
|
reader.readAsText(blob);
|
|
return promise
|
|
}
|
|
|
|
function readArrayBufferAsText(buf) {
|
|
var view = new Uint8Array(buf);
|
|
var chars = new Array(view.length);
|
|
|
|
for (var i = 0; i < view.length; i++) {
|
|
chars[i] = String.fromCharCode(view[i]);
|
|
}
|
|
return chars.join('')
|
|
}
|
|
|
|
function bufferClone(buf) {
|
|
if (buf.slice) {
|
|
return buf.slice(0)
|
|
} else {
|
|
var view = new Uint8Array(buf.byteLength);
|
|
view.set(new Uint8Array(buf));
|
|
return view.buffer
|
|
}
|
|
}
|
|
|
|
function Body() {
|
|
this.bodyUsed = false;
|
|
|
|
this._initBody = function(body) {
|
|
this._bodyInit = body;
|
|
if (!body) {
|
|
this._bodyText = '';
|
|
} else if (typeof body === 'string') {
|
|
this._bodyText = body;
|
|
} else if (support.blob && Blob.prototype.isPrototypeOf(body)) {
|
|
this._bodyBlob = body;
|
|
} else if (support.formData && FormData.prototype.isPrototypeOf(body)) {
|
|
this._bodyFormData = body;
|
|
} else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
|
|
this._bodyText = body.toString();
|
|
} else if (support.arrayBuffer && support.blob && isDataView(body)) {
|
|
this._bodyArrayBuffer = bufferClone(body.buffer);
|
|
// IE 10-11 can't handle a DataView body.
|
|
this._bodyInit = new Blob([this._bodyArrayBuffer]);
|
|
} else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {
|
|
this._bodyArrayBuffer = bufferClone(body);
|
|
} else {
|
|
this._bodyText = body = Object.prototype.toString.call(body);
|
|
}
|
|
|
|
if (!this.headers.get('content-type')) {
|
|
if (typeof body === 'string') {
|
|
this.headers.set('content-type', 'text/plain;charset=UTF-8');
|
|
} else if (this._bodyBlob && this._bodyBlob.type) {
|
|
this.headers.set('content-type', this._bodyBlob.type);
|
|
} else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {
|
|
this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');
|
|
}
|
|
}
|
|
};
|
|
|
|
if (support.blob) {
|
|
this.blob = function() {
|
|
var rejected = consumed(this);
|
|
if (rejected) {
|
|
return rejected
|
|
}
|
|
|
|
if (this._bodyBlob) {
|
|
return Promise.resolve(this._bodyBlob)
|
|
} else if (this._bodyArrayBuffer) {
|
|
return Promise.resolve(new Blob([this._bodyArrayBuffer]))
|
|
} else if (this._bodyFormData) {
|
|
throw new Error('could not read FormData body as blob')
|
|
} else {
|
|
return Promise.resolve(new Blob([this._bodyText]))
|
|
}
|
|
};
|
|
|
|
this.arrayBuffer = function() {
|
|
if (this._bodyArrayBuffer) {
|
|
return consumed(this) || Promise.resolve(this._bodyArrayBuffer)
|
|
} else {
|
|
return this.blob().then(readBlobAsArrayBuffer)
|
|
}
|
|
};
|
|
}
|
|
|
|
this.text = function() {
|
|
var rejected = consumed(this);
|
|
if (rejected) {
|
|
return rejected
|
|
}
|
|
|
|
if (this._bodyBlob) {
|
|
return readBlobAsText(this._bodyBlob)
|
|
} else if (this._bodyArrayBuffer) {
|
|
return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))
|
|
} else if (this._bodyFormData) {
|
|
throw new Error('could not read FormData body as text')
|
|
} else {
|
|
return Promise.resolve(this._bodyText)
|
|
}
|
|
};
|
|
|
|
if (support.formData) {
|
|
this.formData = function() {
|
|
return this.text().then(decode)
|
|
};
|
|
}
|
|
|
|
this.json = function() {
|
|
return this.text().then(JSON.parse)
|
|
};
|
|
|
|
return this
|
|
}
|
|
|
|
// HTTP methods whose capitalization should be normalized
|
|
var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT'];
|
|
|
|
function normalizeMethod(method) {
|
|
var upcased = method.toUpperCase();
|
|
return methods.indexOf(upcased) > -1 ? upcased : method
|
|
}
|
|
|
|
function Request(input, options) {
|
|
options = options || {};
|
|
var body = options.body;
|
|
|
|
if (input instanceof Request) {
|
|
if (input.bodyUsed) {
|
|
throw new TypeError('Already read')
|
|
}
|
|
this.url = input.url;
|
|
this.credentials = input.credentials;
|
|
if (!options.headers) {
|
|
this.headers = new Headers(input.headers);
|
|
}
|
|
this.method = input.method;
|
|
this.mode = input.mode;
|
|
this.signal = input.signal;
|
|
if (!body && input._bodyInit != null) {
|
|
body = input._bodyInit;
|
|
input.bodyUsed = true;
|
|
}
|
|
} else {
|
|
this.url = String(input);
|
|
}
|
|
|
|
this.credentials = options.credentials || this.credentials || 'same-origin';
|
|
if (options.headers || !this.headers) {
|
|
this.headers = new Headers(options.headers);
|
|
}
|
|
this.method = normalizeMethod(options.method || this.method || 'GET');
|
|
this.mode = options.mode || this.mode || null;
|
|
this.signal = options.signal || this.signal;
|
|
this.referrer = null;
|
|
|
|
if ((this.method === 'GET' || this.method === 'HEAD') && body) {
|
|
throw new TypeError('Body not allowed for GET or HEAD requests')
|
|
}
|
|
this._initBody(body);
|
|
}
|
|
|
|
Request.prototype.clone = function() {
|
|
return new Request(this, {body: this._bodyInit})
|
|
};
|
|
|
|
function decode(body) {
|
|
var form = new FormData();
|
|
body
|
|
.trim()
|
|
.split('&')
|
|
.forEach(function(bytes) {
|
|
if (bytes) {
|
|
var split = bytes.split('=');
|
|
var name = split.shift().replace(/\+/g, ' ');
|
|
var value = split.join('=').replace(/\+/g, ' ');
|
|
form.append(decodeURIComponent(name), decodeURIComponent(value));
|
|
}
|
|
});
|
|
return form
|
|
}
|
|
|
|
function parseHeaders(rawHeaders) {
|
|
var headers = new Headers();
|
|
// Replace instances of \r\n and \n followed by at least one space or horizontal tab with a space
|
|
// https://tools.ietf.org/html/rfc7230#section-3.2
|
|
var preProcessedHeaders = rawHeaders.replace(/\r?\n[\t ]+/g, ' ');
|
|
preProcessedHeaders.split(/\r?\n/).forEach(function(line) {
|
|
var parts = line.split(':');
|
|
var key = parts.shift().trim();
|
|
if (key) {
|
|
var value = parts.join(':').trim();
|
|
headers.append(key, value);
|
|
}
|
|
});
|
|
return headers
|
|
}
|
|
|
|
Body.call(Request.prototype);
|
|
|
|
function Response(bodyInit, options) {
|
|
if (!options) {
|
|
options = {};
|
|
}
|
|
|
|
this.type = 'default';
|
|
this.status = options.status === undefined ? 200 : options.status;
|
|
this.ok = this.status >= 200 && this.status < 300;
|
|
this.statusText = 'statusText' in options ? options.statusText : 'OK';
|
|
this.headers = new Headers(options.headers);
|
|
this.url = options.url || '';
|
|
this._initBody(bodyInit);
|
|
}
|
|
|
|
Body.call(Response.prototype);
|
|
|
|
Response.prototype.clone = function() {
|
|
return new Response(this._bodyInit, {
|
|
status: this.status,
|
|
statusText: this.statusText,
|
|
headers: new Headers(this.headers),
|
|
url: this.url
|
|
})
|
|
};
|
|
|
|
Response.error = function() {
|
|
var response = new Response(null, {status: 0, statusText: ''});
|
|
response.type = 'error';
|
|
return response
|
|
};
|
|
|
|
var redirectStatuses = [301, 302, 303, 307, 308];
|
|
|
|
Response.redirect = function(url, status) {
|
|
if (redirectStatuses.indexOf(status) === -1) {
|
|
throw new RangeError('Invalid status code')
|
|
}
|
|
|
|
return new Response(null, {status: status, headers: {location: url}})
|
|
};
|
|
|
|
exports.DOMException = self.DOMException;
|
|
try {
|
|
new exports.DOMException();
|
|
} catch (err) {
|
|
exports.DOMException = function(message, name) {
|
|
this.message = message;
|
|
this.name = name;
|
|
var error = Error(message);
|
|
this.stack = error.stack;
|
|
};
|
|
exports.DOMException.prototype = Object.create(Error.prototype);
|
|
exports.DOMException.prototype.constructor = exports.DOMException;
|
|
}
|
|
|
|
function fetch(input, init) {
|
|
return new Promise(function(resolve, reject) {
|
|
var request = new Request(input, init);
|
|
|
|
if (request.signal && request.signal.aborted) {
|
|
return reject(new exports.DOMException('Aborted', 'AbortError'))
|
|
}
|
|
|
|
var xhr = new XMLHttpRequest();
|
|
|
|
function abortXhr() {
|
|
xhr.abort();
|
|
}
|
|
|
|
xhr.onload = function() {
|
|
var options = {
|
|
status: xhr.status,
|
|
statusText: xhr.statusText,
|
|
headers: parseHeaders(xhr.getAllResponseHeaders() || '')
|
|
};
|
|
options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL');
|
|
var body = 'response' in xhr ? xhr.response : xhr.responseText;
|
|
resolve(new Response(body, options));
|
|
};
|
|
|
|
xhr.onerror = function() {
|
|
reject(new TypeError('Network request failed'));
|
|
};
|
|
|
|
xhr.ontimeout = function() {
|
|
reject(new TypeError('Network request failed'));
|
|
};
|
|
|
|
xhr.onabort = function() {
|
|
reject(new exports.DOMException('Aborted', 'AbortError'));
|
|
};
|
|
|
|
xhr.open(request.method, request.url, true);
|
|
|
|
if (request.credentials === 'include') {
|
|
xhr.withCredentials = true;
|
|
} else if (request.credentials === 'omit') {
|
|
xhr.withCredentials = false;
|
|
}
|
|
|
|
if ('responseType' in xhr && support.blob) {
|
|
xhr.responseType = 'blob';
|
|
}
|
|
|
|
request.headers.forEach(function(value, name) {
|
|
xhr.setRequestHeader(name, value);
|
|
});
|
|
|
|
if (request.signal) {
|
|
request.signal.addEventListener('abort', abortXhr);
|
|
|
|
xhr.onreadystatechange = function() {
|
|
// DONE (success or failure)
|
|
if (xhr.readyState === 4) {
|
|
request.signal.removeEventListener('abort', abortXhr);
|
|
}
|
|
};
|
|
}
|
|
|
|
xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit);
|
|
})
|
|
}
|
|
|
|
fetch.polyfill = true;
|
|
|
|
if (!self.fetch) {
|
|
self.fetch = fetch;
|
|
self.Headers = Headers;
|
|
self.Request = Request;
|
|
self.Response = Response;
|
|
}
|
|
|
|
exports.Headers = Headers;
|
|
exports.Request = Request;
|
|
exports.Response = Response;
|
|
exports.fetch = fetch;
|
|
|
|
Object.defineProperty(exports, '__esModule', { value: true });
|
|
|
|
})));
|
|
|
|
},{}],6:[function(require,module,exports){
|
|
const levenshtein = require('fast-levenshtein')
|
|
const DEFAULT_TOLERANCE = 3
|
|
|
|
class PhishingDetector {
|
|
|
|
constructor (opts) {
|
|
this.whitelist = processDomainList(opts.whitelist || [])
|
|
this.blacklist = processDomainList(opts.blacklist || [])
|
|
this.fuzzylist = processDomainList(opts.fuzzylist || [])
|
|
this.tolerance = ('tolerance' in opts) ? opts.tolerance : DEFAULT_TOLERANCE
|
|
}
|
|
|
|
check (domain) {
|
|
const source = domainToParts(domain)
|
|
|
|
// if source matches whitelist domain (or subdomain thereof), PASS
|
|
const whitelistMatch = matchPartsAgainstList(source, this.whitelist)
|
|
if (whitelistMatch) return { type: 'whitelist', result: false }
|
|
|
|
// if source matches blacklist domain (or subdomain thereof), FAIL
|
|
const blacklistMatch = matchPartsAgainstList(source, this.blacklist)
|
|
if (blacklistMatch) return { type: 'blacklist', result: true }
|
|
|
|
if (this.tolerance > 0) {
|
|
// check if near-match of whitelist domain, FAIL
|
|
let fuzzyForm = domainPartsToFuzzyForm(source)
|
|
// strip www
|
|
fuzzyForm = fuzzyForm.replace('www.', '')
|
|
// check against fuzzylist
|
|
const levenshteinMatched = this.fuzzylist.find((targetParts) => {
|
|
const fuzzyTarget = domainPartsToFuzzyForm(targetParts)
|
|
const distance = levenshtein.get(fuzzyForm, fuzzyTarget)
|
|
return distance <= this.tolerance
|
|
})
|
|
if (levenshteinMatched) {
|
|
const match = domainPartsToDomain(levenshteinMatched)
|
|
return { type: 'fuzzy', result: true, match }
|
|
}
|
|
}
|
|
|
|
// matched nothing, PASS
|
|
return { type: 'all', result: false }
|
|
}
|
|
|
|
}
|
|
|
|
module.exports = PhishingDetector
|
|
|
|
// util
|
|
|
|
function processDomainList (list) {
|
|
return list.map(domainToParts)
|
|
}
|
|
|
|
function domainToParts (domain) {
|
|
return domain.split('.').reverse()
|
|
}
|
|
|
|
function domainPartsToDomain(domainParts) {
|
|
return domainParts.slice().reverse().join('.')
|
|
}
|
|
|
|
// for fuzzy search, drop TLD and re-stringify
|
|
function domainPartsToFuzzyForm(domainParts) {
|
|
return domainParts.slice(1).reverse().join('.')
|
|
}
|
|
|
|
// match the target parts, ignoring extra subdomains on source
|
|
// source: [io, metamask, xyz]
|
|
// target: [io, metamask]
|
|
// result: PASS
|
|
function matchPartsAgainstList(source, list) {
|
|
return list.some((target) => {
|
|
// target domain has more parts than source, fail
|
|
if (target.length > source.length) return false
|
|
// source matches target or (is deeper subdomain)
|
|
return target.every((part, index) => source[index] === part)
|
|
})
|
|
}
|
|
},{"fast-levenshtein":3}]},{},[1]);
|