singbox.ts 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  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. writeDomainWildcard(wildcard: string): void {
  62. this.singbox.domain_regex ??= [];
  63. this.singbox.domain_regex.push(SingboxSource.domainWildCardToRegex(wildcard));
  64. }
  65. writeUserAgents = noop;
  66. writeProcessNames(processName: Set<string>): void {
  67. appendArrayInPlace(
  68. this.singbox.process_name ??= [],
  69. Array.from(processName)
  70. );
  71. }
  72. writeProcessPaths(processPath: Set<string>): void {
  73. appendArrayInPlace(
  74. this.singbox.process_path ??= [],
  75. Array.from(processPath)
  76. );
  77. }
  78. writeUrlRegexes = noop;
  79. writeIpCidrs(ipCidr: string[]): void {
  80. appendArrayInPlace(
  81. this.singbox.ip_cidr ??= [],
  82. ipCidr
  83. );
  84. }
  85. writeIpCidr6s(ipCidr6: string[]): void {
  86. appendArrayInPlace(
  87. this.singbox.ip_cidr ??= [],
  88. ipCidr6
  89. );
  90. }
  91. writeGeoip = noop;
  92. writeIpAsns = noop;
  93. writeSourceIpCidrs(sourceIpCidr: string[]): void {
  94. this.singbox.source_ip_cidr ??= [];
  95. for (let i = 0, len = sourceIpCidr.length; i < len; i++) {
  96. const value = sourceIpCidr[i];
  97. if (value.includes('/')) {
  98. this.singbox.source_ip_cidr.push(value);
  99. continue;
  100. }
  101. const v = fastIpVersion(value);
  102. if (v === 4) {
  103. this.singbox.source_ip_cidr.push(`${value}/32`);
  104. continue;
  105. }
  106. if (v === 6) {
  107. this.singbox.source_ip_cidr.push(`${value}/128`);
  108. continue;
  109. }
  110. }
  111. }
  112. writeSourcePorts(port: Set<string>): void {
  113. this.singbox.source_port ??= [];
  114. for (const i of port) {
  115. const tmp = Number(i);
  116. if (!Number.isNaN(tmp)) {
  117. this.singbox.source_port.push(tmp);
  118. }
  119. }
  120. }
  121. writeDestinationPorts(port: Set<string>): void {
  122. this.singbox.port ??= [];
  123. for (const i of port) {
  124. const tmp = Number(i);
  125. if (!Number.isNaN(tmp)) {
  126. this.singbox.port.push(tmp);
  127. }
  128. }
  129. }
  130. writeProtocols(protocol: Set<string>): void {
  131. this.singbox.network ??= [];
  132. // protocol has already be normalized and will only be uppercase
  133. if (protocol.has('UDP')) {
  134. this.singbox.network.push('udp');
  135. }
  136. if (protocol.has('TCP')) {
  137. this.singbox.network.push('tcp');
  138. }
  139. }
  140. writeOtherRules = noop;
  141. }