1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-12-13 13:17:13 +01:00
metamask-extension/app/scripts/lib/ens-ipfs/setup.js

109 lines
3.4 KiB
JavaScript
Raw Normal View History

import base32Encode from 'base32-encode';
import base64 from 'base64-js';
import browser from 'webextension-polyfill';
import { SECOND } from '../../../../shared/constants/time';
import getFetchWithTimeout from '../../../../shared/modules/fetch-with-timeout';
import resolveEnsToIpfsContentId from './resolver';
const fetchWithTimeout = getFetchWithTimeout(SECOND * 30);
const supportedTopLevelDomains = ['eth'];
2020-11-03 00:41:28 +01:00
export default function setupEnsIpfsResolver({
provider,
getCurrentChainId,
2020-11-03 00:41:28 +01:00
getIpfsGateway,
}) {
// install listener
const urlPatterns = supportedTopLevelDomains.map((tld) => `*://*.${tld}/*`);
browser.webRequest.onErrorOccurred.addListener(webRequestDidFail, {
2020-11-03 00:41:28 +01:00
urls: urlPatterns,
types: ['main_frame'],
});
// return api object
return {
// uninstall listener
2020-11-03 00:41:28 +01:00
remove() {
browser.webRequest.onErrorOccurred.removeListener(webRequestDidFail);
},
};
2020-11-03 00:41:28 +01:00
async function webRequestDidFail(details) {
const { tabId, url } = details;
// ignore requests that are not associated with tabs
// only attempt ENS resolution on mainnet
if (tabId === -1 || getCurrentChainId() !== '0x1') {
return;
}
// parse ens name
const { hostname: name, pathname, search, hash: fragment } = new URL(url);
const domainParts = name.split('.');
const topLevelDomain = domainParts[domainParts.length - 1];
// if unsupported TLD, abort
if (!supportedTopLevelDomains.includes(topLevelDomain)) {
return;
}
// otherwise attempt resolve
attemptResolve({ tabId, name, pathname, search, fragment });
}
2020-11-03 00:41:28 +01:00
async function attemptResolve({ tabId, name, pathname, search, fragment }) {
const ipfsGateway = getIpfsGateway();
browser.tabs.update(tabId, { url: `loading.html` });
let url = `https://app.ens.domains/name/${name}`;
try {
const { type, hash } = await resolveEnsToIpfsContentId({
provider,
name,
});
if (type === 'ipfs-ns' || type === 'ipns-ns') {
2020-11-03 00:41:28 +01:00
const resolvedUrl = `https://${hash}.${type.slice(
0,
4,
)}.${ipfsGateway}${pathname}${search || ''}${fragment || ''}`;
2019-04-04 17:15:57 +02:00
try {
// check if ipfs gateway has result
const response = await fetchWithTimeout(resolvedUrl, {
method: 'HEAD',
});
if (response.status === 200) {
url = resolvedUrl;
}
2019-04-04 17:15:57 +02:00
} catch (err) {
console.warn(err);
}
2019-04-04 17:15:57 +02:00
} else if (type === 'swarm-ns') {
2020-11-03 00:41:28 +01:00
url = `https://swarm-gateways.net/bzz:/${hash}${pathname}${
search || ''
}${fragment || ''}`;
} else if (type === 'onion' || type === 'onion3') {
url = `http://${hash}.onion${pathname}${search || ''}${fragment || ''}`;
2019-10-31 19:37:06 +01:00
} else if (type === 'zeronet') {
2020-11-03 00:41:28 +01:00
url = `http://127.0.0.1:43110/${hash}${pathname}${search || ''}${
fragment || ''
}`;
} else if (type === 'skynet-ns') {
const padded = hash.padEnd(hash.length + 4 - (hash.length % 4), '=');
const decoded = base64.toByteArray(padded);
const options = { padding: false };
const base32EncodedSkylink = base32Encode(
decoded,
'RFC4648-HEX',
options,
).toLowerCase();
url = `https://${base32EncodedSkylink}.siasky.net${pathname}${
search || ''
}${fragment || ''}`;
}
} catch (err) {
console.warn(err);
2019-05-04 18:57:19 +02:00
} finally {
browser.tabs.update(tabId, { url });
}
}
}