singbox.ts 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. import { BaseWriteStrategy } from './base';
  2. import { appendArrayInPlace } from 'foxts/append-array-in-place';
  3. import { noop } from 'foxts/noop';
  4. import { fastIpVersion, withIdentityContent } from '../misc';
  5. import stringify from 'json-stringify-pretty-compact';
  6. import { OUTPUT_SINGBOX_DIR } from '../../constants/dir';
  7. import { MARKER_DOMAIN } from '../../constants/description';
  8. interface SingboxHeadlessRule {
  9. domain: string[],
  10. domain_suffix: string[],
  11. domain_keyword?: string[],
  12. domain_regex?: string[],
  13. source_ip_cidr?: string[],
  14. ip_cidr?: string[],
  15. source_port?: number[],
  16. source_port_range?: string[],
  17. port?: number[],
  18. port_range?: string[],
  19. process_name?: string[],
  20. process_path?: string[],
  21. network?: string[]
  22. }
  23. export interface SingboxSourceFormat {
  24. version: 2 | number & {},
  25. rules: SingboxHeadlessRule[]
  26. }
  27. export class SingboxSource extends BaseWriteStrategy {
  28. public readonly name = 'singbox';
  29. readonly fileExtension = 'json';
  30. static readonly jsonToLines = (json: unknown): string[] => stringify(json).split('\n');
  31. private singbox: SingboxHeadlessRule = {
  32. domain: [MARKER_DOMAIN],
  33. domain_suffix: [MARKER_DOMAIN]
  34. };
  35. protected get result() {
  36. return SingboxSource.jsonToLines({
  37. version: 2,
  38. rules: [this.singbox]
  39. });
  40. }
  41. constructor(
  42. /** Since sing-box only have one format that does not reflect type, we need to specify it */
  43. public type: 'domainset' | 'non_ip' | 'ip' /* | (string & {}) */,
  44. public readonly outputDir = OUTPUT_SINGBOX_DIR
  45. ) {
  46. super(outputDir);
  47. }
  48. withPadding = withIdentityContent;
  49. writeDomain(domain: string): void {
  50. this.singbox.domain.push(domain);
  51. }
  52. writeDomainSuffix(domain: string): void {
  53. this.singbox.domain_suffix.push(domain);
  54. }
  55. writeDomainKeywords(keyword: Set<string>): void {
  56. appendArrayInPlace(
  57. this.singbox.domain_keyword ??= [],
  58. Array.from(keyword)
  59. );
  60. }
  61. writeDomainWildcards(wildcard: Set<string>): void {
  62. appendArrayInPlace(
  63. this.singbox.domain_regex ??= [],
  64. Array.from(wildcard, SingboxSource.domainWildCardToRegex)
  65. );
  66. }
  67. writeUserAgents = noop;
  68. writeProcessNames(processName: Set<string>): void {
  69. appendArrayInPlace(
  70. this.singbox.process_name ??= [],
  71. Array.from(processName)
  72. );
  73. }
  74. writeProcessPaths(processPath: Set<string>): void {
  75. appendArrayInPlace(
  76. this.singbox.process_path ??= [],
  77. Array.from(processPath)
  78. );
  79. }
  80. writeUrlRegexes = noop;
  81. writeIpCidrs(ipCidr: string[]): void {
  82. appendArrayInPlace(
  83. this.singbox.ip_cidr ??= [],
  84. ipCidr
  85. );
  86. }
  87. writeIpCidr6s(ipCidr6: string[]): void {
  88. appendArrayInPlace(
  89. this.singbox.ip_cidr ??= [],
  90. ipCidr6
  91. );
  92. }
  93. writeGeoip = noop;
  94. writeIpAsns = noop;
  95. writeSourceIpCidrs(sourceIpCidr: string[]): void {
  96. this.singbox.source_ip_cidr ??= [];
  97. for (let i = 0, len = sourceIpCidr.length; i < len; i++) {
  98. const value = sourceIpCidr[i];
  99. if (value.includes('/')) {
  100. this.singbox.source_ip_cidr.push(value);
  101. continue;
  102. }
  103. const v = fastIpVersion(value);
  104. if (v === 4) {
  105. this.singbox.source_ip_cidr.push(`${value}/32`);
  106. continue;
  107. }
  108. if (v === 6) {
  109. this.singbox.source_ip_cidr.push(`${value}/128`);
  110. continue;
  111. }
  112. }
  113. }
  114. writeSourcePorts(port: Set<string>): void {
  115. this.singbox.source_port ??= [];
  116. for (const i of port) {
  117. const tmp = Number(i);
  118. if (!Number.isNaN(tmp)) {
  119. this.singbox.source_port.push(tmp);
  120. }
  121. }
  122. }
  123. writeDestinationPorts(port: Set<string>): void {
  124. this.singbox.port ??= [];
  125. for (const i of port) {
  126. const tmp = Number(i);
  127. if (!Number.isNaN(tmp)) {
  128. this.singbox.port.push(tmp);
  129. }
  130. }
  131. }
  132. writeProtocols(protocol: Set<string>): void {
  133. this.singbox.network ??= [];
  134. // protocol has already be normalized and will only be uppercase
  135. if (protocol.has('UDP')) {
  136. this.singbox.network.push('udp');
  137. }
  138. if (protocol.has('TCP')) {
  139. this.singbox.network.push('tcp');
  140. }
  141. }
  142. writeOtherRules = noop;
  143. }