mirror of
https://github.com/Instadapp/trustwallet-assets.git
synced 2024-07-29 22:37:31 +00:00
5efd533fdc
* Extra logs * External Updates * Move token validity check, cheksumFormat ID is available only later. * External Updates Co-authored-by: Catenocrypt <catenocrypt@users.noreply.github.com>
265 lines
8.8 KiB
TypeScript
265 lines
8.8 KiB
TypeScript
// Tokenlist.json handling
|
|
|
|
import { readJsonFile, writeJsonFile } from "../generic/json";
|
|
import { diff } from "jsondiffpatch";
|
|
import { tokenInfoFromTwApi, TokenTwInfo } from "../generic/asset";
|
|
import {
|
|
getChainAssetLogoPath,
|
|
getChainAllowlistPath,
|
|
getChainTokenlistBasePath,
|
|
getChainTokenlistPath,
|
|
} from "../generic/repo-structure";
|
|
import * as bluebird from "bluebird";
|
|
import { isPathExistsSync } from "../generic/filesystem";
|
|
|
|
class Version {
|
|
major: number
|
|
minor: number
|
|
patch: number
|
|
|
|
constructor(major: number, minor: number, patch: number) {
|
|
this.major = major
|
|
this.minor = minor
|
|
this.patch = patch
|
|
}
|
|
}
|
|
|
|
export class List {
|
|
name: string
|
|
logoURI: string
|
|
timestamp: string
|
|
tokens: TokenItem[]
|
|
pairs: Pair[]
|
|
version: Version
|
|
|
|
constructor(name: string, logoURI: string, timestamp: string, tokens: TokenItem[], version: Version) {
|
|
this.name = name
|
|
this.logoURI = logoURI
|
|
this.timestamp = timestamp;
|
|
this.tokens = tokens
|
|
this.version = version
|
|
}
|
|
}
|
|
|
|
export class TokenItem {
|
|
asset: string;
|
|
type: string;
|
|
address: string;
|
|
name: string;
|
|
symbol: string;
|
|
decimals: number;
|
|
logoURI: string;
|
|
pairs: Pair[];
|
|
|
|
constructor(asset: string, type: string, address: string, name: string, symbol: string, decimals: number, logoURI: string, pairs: Pair[]) {
|
|
this.asset = asset
|
|
this.type = type
|
|
this.address = address
|
|
this.name = name;
|
|
this.symbol = symbol
|
|
this.decimals = decimals
|
|
this.logoURI = logoURI
|
|
this.pairs = pairs
|
|
}
|
|
}
|
|
|
|
export class Pair {
|
|
base: string;
|
|
lotSize?: string;
|
|
tickSize?: string;
|
|
|
|
constructor(base: string, lotSize?: string, tickSize?: string) {
|
|
this.base = base
|
|
this.lotSize = lotSize
|
|
this.tickSize = tickSize
|
|
}
|
|
}
|
|
|
|
export function createTokensList(titleCoin: string, tokens: TokenItem[], time: string, versionMajor: number, versionMinor = 1, versionPatch = 0): List {
|
|
if (!time) {
|
|
time = (new Date()).toISOString();
|
|
}
|
|
const list = new List(
|
|
`Trust Wallet: ${titleCoin}`,
|
|
"https://trustwallet.com/assets/images/favicon.png",
|
|
time,
|
|
tokens,
|
|
new Version(versionMajor, versionMinor, versionPatch)
|
|
);
|
|
sort(list);
|
|
return list;
|
|
}
|
|
|
|
function totalPairs(list: List): number {
|
|
let c = 0;
|
|
list.tokens.forEach(t => c += (t.pairs || []).length);
|
|
return c;
|
|
}
|
|
|
|
export function writeToFile(filename: string, list: List): void {
|
|
writeJsonFile(filename, list);
|
|
console.log(`Tokenlist: list with ${list.tokens.length} tokens and ${totalPairs(list)} pairs written to ${filename}.`);
|
|
}
|
|
|
|
// Write out to file, updating version+timestamp if there was change
|
|
export function writeToFileWithUpdate(filename: string, list: List): void {
|
|
let listOld: List = undefined;
|
|
try {
|
|
listOld = readJsonFile(filename) as List;
|
|
} catch (err) {
|
|
listOld = undefined;
|
|
}
|
|
if (listOld !== undefined) {
|
|
list.version = listOld.version; // take over
|
|
list.timestamp = listOld.timestamp; // take over
|
|
const diffs = diffTokenlist(list, listOld);
|
|
if (diffs != undefined) {
|
|
//console.log("List has Changed", JSON.stringify(diffs));
|
|
list.version = new Version(list.version.major + 1, 0, 0);
|
|
list.timestamp = (new Date()).toISOString();
|
|
console.log(`Version and timestamp updated, ${list.version.major}.${list.version.minor}.${list.version.patch} timestamp ${list.timestamp}`);
|
|
}
|
|
}
|
|
writeToFile(filename, list);
|
|
}
|
|
|
|
async function addTokenIfNeeded(token: TokenItem, list: List): Promise<void> {
|
|
if (list.tokens.map(t => t.address.toLowerCase()).includes(token.address.toLowerCase())) {
|
|
return;
|
|
}
|
|
token = await updateTokenInfo(token);
|
|
list.tokens.push(token);
|
|
}
|
|
|
|
// Update/fix token info, with properties retrieved from TW API
|
|
async function updateTokenInfo(token: TokenItem): Promise<TokenItem> {
|
|
const tokenInfo: TokenTwInfo = await tokenInfoFromTwApi(token.asset);
|
|
if (tokenInfo) {
|
|
if (token.name && token.name != tokenInfo.name) {
|
|
console.log(`Token name adjusted: '${token.name}' -> '${tokenInfo.name}'`);
|
|
token.name = tokenInfo.name;
|
|
}
|
|
if (token.symbol && token.symbol != tokenInfo.symbol) {
|
|
console.log(`Token symbol adjusted: '${token.symbol}' -> '${tokenInfo.symbol}'`);
|
|
token.symbol = tokenInfo.symbol;
|
|
}
|
|
if (token.decimals && token.decimals != tokenInfo.decimals) {
|
|
console.log(`Token decimals adjusted: '${token.decimals}' -> '${tokenInfo.decimals}'`);
|
|
token.decimals = parseInt(tokenInfo.decimals.toString());
|
|
}
|
|
}
|
|
return token;
|
|
}
|
|
|
|
function addPairToToken(pairToken: TokenItem, token: TokenItem, list: List): void {
|
|
const tokenInList = list.tokens.find(t => t.address === token.address);
|
|
if (!tokenInList) {
|
|
return;
|
|
}
|
|
if (!tokenInList.pairs) {
|
|
tokenInList.pairs = [];
|
|
}
|
|
if (tokenInList.pairs.map(p => p.base).includes(pairToken.asset)) {
|
|
return;
|
|
}
|
|
tokenInList.pairs.push(new Pair(pairToken.asset));
|
|
}
|
|
|
|
function checkTokenExists(id: string, chainName: string, tokenAllowlist: string[]): boolean {
|
|
const logoPath = getChainAssetLogoPath(chainName, id);
|
|
if (!isPathExistsSync(logoPath)) {
|
|
//console.log("logo file missing", logoPath);
|
|
return false;
|
|
}
|
|
if (tokenAllowlist.find(t => (id.toLowerCase() === t.toLowerCase())) === undefined) {
|
|
//console.log(`Token not found in allowlist, ${id}`);
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
export async function addPairIfNeeded(token0: TokenItem, token1: TokenItem, list: List): Promise<void> {
|
|
await addTokenIfNeeded(token0, list);
|
|
await addTokenIfNeeded(token1, list);
|
|
addPairToToken(token1, token0, list);
|
|
// reverse direction not needed addPairToToken(token0, token1, list);
|
|
}
|
|
|
|
function sort(list: List) {
|
|
list.tokens.sort((t1, t2) => {
|
|
const t1pairs = (t1.pairs || []).length;
|
|
const t2pairs = (t2.pairs || []).length;
|
|
if (t1pairs != t2pairs) { return t2pairs - t1pairs; }
|
|
return t1.address.localeCompare(t2.address);
|
|
});
|
|
list.tokens.forEach(t => {
|
|
t.pairs.sort((p1, p2) => p1.base.localeCompare(p2.base));
|
|
});
|
|
}
|
|
|
|
function clearUnimportantFields(list: List) {
|
|
list.timestamp = "";
|
|
list.version = new Version(0, 0, 0);
|
|
}
|
|
|
|
export function diffTokenlist(listOrig1: List, listOrig2: List): unknown {
|
|
// deep copy, to avoid changes
|
|
const list1 = JSON.parse(JSON.stringify(listOrig1));
|
|
const list2 = JSON.parse(JSON.stringify(listOrig2));
|
|
clearUnimportantFields(list1);
|
|
clearUnimportantFields(list2);
|
|
sort(list1);
|
|
sort(list2);
|
|
// compare
|
|
const diffs = diff(list1, list2);
|
|
return diffs;
|
|
}
|
|
|
|
export async function rebuildTokenlist(chainName: string, pairs: [TokenItem, TokenItem][], listName: string): Promise<void> {
|
|
// sanity check, prevent deletion of many pairs
|
|
if (!pairs || pairs.length < 5) {
|
|
console.log(`Warning: Only ${pairs.length} pairs returned, ignoring`);
|
|
return;
|
|
}
|
|
|
|
// filter out missing tokens
|
|
// prepare phase, read allowlist
|
|
const allowlist: string[] = readJsonFile(getChainAllowlistPath(chainName)) as string[];
|
|
const pairs2: [TokenItem, TokenItem][] = [];
|
|
pairs.forEach(p => {
|
|
if (!checkTokenExists(p[0].address, chainName, allowlist)) {
|
|
console.log("pair with unsupported 1st coin:", p[0].symbol, "--", p[1].symbol);
|
|
return;
|
|
}
|
|
if (!checkTokenExists(p[1].address, chainName, allowlist)) {
|
|
console.log("pair with unsupported 2nd coin:", p[0].symbol, "--", p[1].symbol);
|
|
return;
|
|
}
|
|
pairs2.push(p);
|
|
});
|
|
const filteredCount: number = pairs.length - pairs2.length;
|
|
console.log(`${filteredCount} unsupported tokens filtered out, ${pairs2.length} pairs`);
|
|
|
|
const tokenlistFile = getChainTokenlistPath(chainName);
|
|
{
|
|
// show current size
|
|
const json = readJsonFile(tokenlistFile);
|
|
const list: List = json as List;
|
|
console.log(`Tokenlist original: ${list.tokens.length} tokens`);
|
|
}
|
|
const tokenlistBaseFile = getChainTokenlistBasePath(chainName);
|
|
const json = readJsonFile(tokenlistBaseFile);
|
|
const list: List = json as List;
|
|
console.log(`Tokenlist base: ${list.tokens.length} tokens`);
|
|
|
|
await bluebird.each(pairs2, async (p) => {
|
|
await addPairIfNeeded(p[0], p[1], list);
|
|
});
|
|
console.log(`Tokenlist updated: ${list.tokens.length} tokens`);
|
|
|
|
const newList = createTokensList(listName, list.tokens,
|
|
"2021-01-29T01:02:03.000+00:00", // use constant here to prevent changing time every time
|
|
0, 1, 0);
|
|
writeToFileWithUpdate(tokenlistFile, newList);
|
|
}
|