build-telegram-cidr.worker.ts 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. // @ts-check
  2. import { task } from './trace';
  3. import { SHARED_DESCRIPTION } from './constants/description';
  4. import { RulesetOutput } from './lib/rules/ruleset';
  5. import { getTelegramBackupIPFromBase64 } from './lib/get-telegram-backup-ip';
  6. import picocolors from 'picocolors';
  7. import { $$fetch } from './lib/fetch-retry';
  8. import dns from 'node:dns/promises';
  9. import { createReadlineInterfaceFromResponse } from './lib/fetch-text-by-line';
  10. import { fastIpVersion } from 'foxts/fast-ip-version';
  11. import { fastStringArrayJoin } from 'foxts/fast-string-array-join';
  12. export const buildTelegramCIDR = task(require.main === module, __filename)(async (span) => {
  13. const { timestamp, ipcidr, ipcidr6 } = await span.traceChildAsync('get telegram cidr', async () => {
  14. const resp = await $$fetch('https://core.telegram.org/resources/cidr.txt');
  15. const lastModified = resp.headers.get('last-modified');
  16. const date = lastModified ? new Date(lastModified) : new Date();
  17. const ipcidr: string[] = [
  18. // Unused secret Telegram backup CIDR, announced by AS62041
  19. '95.161.64.0/20'
  20. ];
  21. const ipcidr6: string[] = [];
  22. for await (const cidr of createReadlineInterfaceFromResponse(resp, true)) {
  23. const v = fastIpVersion(cidr);
  24. if (v === 4) {
  25. ipcidr.push(cidr);
  26. } else if (v === 6) {
  27. ipcidr6.push(cidr);
  28. }
  29. }
  30. const backupIPs = new Set<string>();
  31. // https://github.com/tdlib/td/blob/master/td/telegram/ConfigManager.cpp
  32. const resolvers = ['8.8.8.8', '1.0.0.1'].map((ip) => {
  33. const resolver = new dns.Resolver();
  34. resolver.setServers([ip]);
  35. return Object.assign(resolver, { server: ip });
  36. });
  37. // Backup IP Source 1 (DNS)
  38. await Promise.all(resolvers.flatMap((resolver) => [
  39. 'apv3.stel.com', // prod
  40. 'tapv3.stel.com' // test
  41. ].map(async (domain) => {
  42. try {
  43. // tapv3.stel.com was for testing server
  44. const resp = await resolver.resolveTxt(domain);
  45. const strings = resp.map(r => fastStringArrayJoin(r, '')); // flatten
  46. if (strings.length !== 2) {
  47. throw new TypeError(`Unexpected TXT record count: ${strings.length}`);
  48. }
  49. const str = strings[0].length > strings[1].length
  50. ? strings[0] + strings[1]
  51. : strings[1] + strings[0];
  52. const ips = getTelegramBackupIPFromBase64(str);
  53. ips.forEach(i => backupIPs.add(i.ip));
  54. console.log('[telegram backup ip]', picocolors.green('DNS TXT'), { domain, ips, server: resolver.server });
  55. } catch (e) {
  56. console.error('[telegram backup ip]', picocolors.red('DNS TXT error'), { domain }, e);
  57. }
  58. })));
  59. // Backup IP Source 2: Firebase Realtime Database (test server not supported)
  60. try {
  61. const text = await (await $$fetch('https://reserve-5a846.firebaseio.com/ipconfigv3.json')).json();
  62. if (typeof text === 'string' && text.length === 344) {
  63. const ips = getTelegramBackupIPFromBase64(text);
  64. ips.forEach(i => backupIPs.add(i.ip));
  65. console.log('[telegram backup ip]', picocolors.green('Firebase Realtime DB'), { ips });
  66. }
  67. } catch (e) {
  68. console.error('[telegram backup ip]', picocolors.red('Firebase Realtime DB error'), e);
  69. // ignore all errors
  70. }
  71. // Backup IP Source 3: Firebase Value Store (test server not supported)
  72. try {
  73. const json = await (await $$fetch('https://firestore.googleapis.com/v1/projects/reserve-5a846/databases/(default)/documents/ipconfig/v3', {
  74. headers: {
  75. Accept: '*/*',
  76. Origin: undefined // Without this line, Google API will return "Bad request: Origin doesn't match Host for XD3.". Probably have something to do with sqlite cache store
  77. }
  78. })).json();
  79. if (
  80. json && typeof json === 'object'
  81. && 'fields' in json && typeof json.fields === 'object' && json.fields
  82. && 'data' in json.fields && typeof json.fields.data === 'object' && json.fields.data
  83. && 'stringValue' in json.fields.data && typeof json.fields.data.stringValue === 'string' && json.fields.data.stringValue.length === 344
  84. ) {
  85. const ips = getTelegramBackupIPFromBase64(json.fields.data.stringValue);
  86. ips.forEach(i => backupIPs.add(i.ip));
  87. console.log('[telegram backup ip]', picocolors.green('Firebase Value Store'), { ips });
  88. } else {
  89. console.error('[telegram backup ip]', picocolors.red('Firebase Value Store data format invalid'), { json });
  90. }
  91. } catch (e) {
  92. console.error('[telegram backup ip]', picocolors.red('Firebase Value Store error'), e);
  93. }
  94. // Backup IP Source 4: Google App Engine
  95. await Promise.all([
  96. 'https://dns-telegram.appspot.com',
  97. 'https://dns-telegram.appspot.com/test'
  98. ].map(async (url) => {
  99. try {
  100. const text = await (await $$fetch(url)).text();
  101. if (text.length === 344) {
  102. const ips = getTelegramBackupIPFromBase64(text);
  103. ips.forEach(i => backupIPs.add(i.ip));
  104. console.log('[telegram backup ip]', picocolors.green('Google App Engine'), { url, ips });
  105. }
  106. } catch (e) {
  107. console.error('[telegram backup ip]', picocolors.red('Google App Engine error'), { url }, e);
  108. }
  109. }));
  110. // tcdnb.azureedge.net no longer works
  111. console.log('[telegram backup ip]', `Found ${backupIPs.size} backup IPs:`, backupIPs);
  112. ipcidr.push(...Array.from(backupIPs).map(i => i + '/32'));
  113. return { timestamp: date.getTime(), ipcidr, ipcidr6 };
  114. });
  115. if (ipcidr.length + ipcidr6.length === 0) {
  116. throw new Error('Failed to fetch data!');
  117. }
  118. const description = [
  119. ...SHARED_DESCRIPTION,
  120. 'Data from:',
  121. ' - https://core.telegram.org/resources/cidr.txt'
  122. ];
  123. return new RulesetOutput(span, 'telegram', 'ip')
  124. .withTitle('Sukka\'s Ruleset - Telegram IP CIDR')
  125. .withDescription(description)
  126. // .withDate(date) // With extra data source, we no longer use last-modified for file date
  127. .appendDataSource(
  128. 'https://core.telegram.org/resources/cidr.txt (last updated: ' + new Date(timestamp).toISOString() + ')'
  129. )
  130. .bulkAddCIDR4NoResolve(ipcidr)
  131. .bulkAddCIDR6NoResolve(ipcidr6)
  132. .write();
  133. });
  134. export const ___ = '';