recker 1.0.19-next.f5bb375 → 1.0.20-next.595cc59

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (44) hide show
  1. package/dist/cli/index.d.ts +0 -1
  2. package/dist/cli/index.js +495 -2
  3. package/dist/cli/tui/shell.js +1 -1
  4. package/dist/core/client.d.ts +2 -0
  5. package/dist/core/client.d.ts.map +1 -1
  6. package/dist/core/client.js +4 -0
  7. package/dist/plugins/cache.js +1 -1
  8. package/dist/plugins/hls.d.ts +90 -17
  9. package/dist/plugins/hls.d.ts.map +1 -1
  10. package/dist/plugins/hls.js +343 -173
  11. package/dist/plugins/retry.js +2 -2
  12. package/dist/testing/index.d.ts +16 -0
  13. package/dist/testing/index.d.ts.map +1 -1
  14. package/dist/testing/index.js +8 -0
  15. package/dist/testing/mock-dns-server.d.ts +70 -0
  16. package/dist/testing/mock-dns-server.d.ts.map +1 -0
  17. package/dist/testing/mock-dns-server.js +269 -0
  18. package/dist/testing/mock-ftp-server.d.ts +90 -0
  19. package/dist/testing/mock-ftp-server.d.ts.map +1 -0
  20. package/dist/testing/mock-ftp-server.js +562 -0
  21. package/dist/testing/mock-hls-server.d.ts +81 -0
  22. package/dist/testing/mock-hls-server.d.ts.map +1 -0
  23. package/dist/testing/mock-hls-server.js +381 -0
  24. package/dist/testing/mock-http-server.d.ts +100 -0
  25. package/dist/testing/mock-http-server.d.ts.map +1 -0
  26. package/dist/testing/mock-http-server.js +298 -0
  27. package/dist/testing/mock-sse-server.d.ts +77 -0
  28. package/dist/testing/mock-sse-server.d.ts.map +1 -0
  29. package/dist/testing/mock-sse-server.js +291 -0
  30. package/dist/testing/mock-telnet-server.d.ts +60 -0
  31. package/dist/testing/mock-telnet-server.d.ts.map +1 -0
  32. package/dist/testing/mock-telnet-server.js +273 -0
  33. package/dist/testing/mock-websocket-server.d.ts +78 -0
  34. package/dist/testing/mock-websocket-server.d.ts.map +1 -0
  35. package/dist/testing/mock-websocket-server.js +316 -0
  36. package/dist/testing/mock-whois-server.d.ts +57 -0
  37. package/dist/testing/mock-whois-server.d.ts.map +1 -0
  38. package/dist/testing/mock-whois-server.js +234 -0
  39. package/dist/transport/undici.js +1 -1
  40. package/dist/utils/dns-toolkit.js +1 -1
  41. package/dist/utils/dns.js +2 -2
  42. package/dist/utils/optional-require.js +1 -1
  43. package/dist/webrtc/index.js +1 -1
  44. package/package.json +3 -1
@@ -1,3 +1,2 @@
1
- #!/usr/bin/env node
2
1
  export {};
3
2
  //# sourceMappingURL=index.d.ts.map
package/dist/cli/index.js CHANGED
@@ -1,4 +1,3 @@
1
- #!/usr/bin/env node
2
1
  import { program } from 'commander';
3
2
  import { promises as fs } from 'node:fs';
4
3
  import { join } from 'node:path';
@@ -725,7 +724,7 @@ ${colors.gray('Status:')} ${statusIcon}
725
724
  .command('generate-dmarc')
726
725
  .description('Generate a DMARC record interactively')
727
726
  .option('-p, --policy <policy>', 'Policy: none, quarantine, reject', 'none')
728
- .option('-sp, --subdomain-policy <policy>', 'Subdomain policy')
727
+ .option('--subdomain-policy <policy>', 'Subdomain policy')
729
728
  .option('--pct <percent>', 'Percentage of emails to apply policy', '100')
730
729
  .option('--rua <emails>', 'Aggregate report email(s), comma-separated')
731
730
  .option('--ruf <emails>', 'Forensic report email(s), comma-separated')
@@ -880,6 +879,500 @@ ${colors.bold(colors.yellow('Examples:'))}
880
879
  const { startLoadDashboard } = await import('./tui/load-dashboard.js');
881
880
  await startLoadDashboard({ url, users, duration, mode, http2, rampUp });
882
881
  });
882
+ const serve = program.command('serve').description('Start mock servers for testing protocols');
883
+ serve
884
+ .command('http')
885
+ .description('Start a mock HTTP server')
886
+ .option('-p, --port <number>', 'Port to listen on', '3000')
887
+ .option('-h, --host <string>', 'Host to bind to', '127.0.0.1')
888
+ .option('--echo', 'Echo request body back in response')
889
+ .option('--delay <ms>', 'Add delay to responses (milliseconds)', '0')
890
+ .option('--cors', 'Enable CORS', true)
891
+ .addHelpText('after', `
892
+ ${colors.bold(colors.yellow('Examples:'))}
893
+ ${colors.green('$ rek serve http')} ${colors.gray('Start on port 3000')}
894
+ ${colors.green('$ rek serve http -p 8080')} ${colors.gray('Start on port 8080')}
895
+ ${colors.green('$ rek serve http --echo')} ${colors.gray('Echo mode')}
896
+ ${colors.green('$ rek serve http --delay 500')} ${colors.gray('Add 500ms delay')}
897
+ `)
898
+ .action(async (options) => {
899
+ const { MockHttpServer } = await import('../testing/mock-http-server.js');
900
+ const server = await MockHttpServer.create({
901
+ port: parseInt(options.port),
902
+ host: options.host,
903
+ delay: parseInt(options.delay),
904
+ cors: options.cors,
905
+ });
906
+ if (options.echo) {
907
+ server.any('/*', (req) => ({
908
+ status: 200,
909
+ body: {
910
+ method: req.method,
911
+ path: req.path,
912
+ query: req.query,
913
+ headers: req.headers,
914
+ body: req.body,
915
+ },
916
+ }));
917
+ }
918
+ console.log(colors.green(`
919
+ ┌─────────────────────────────────────────────┐
920
+ │ ${colors.bold('Recker Mock HTTP Server')} │
921
+ ├─────────────────────────────────────────────┤
922
+ │ URL: ${colors.cyan(server.url.padEnd(37))}│
923
+ │ Mode: ${colors.yellow((options.echo ? 'Echo' : 'Default').padEnd(36))}│
924
+ │ Delay: ${colors.gray((options.delay + 'ms').padEnd(35))}│
925
+ ├─────────────────────────────────────────────┤
926
+ │ Press ${colors.bold('Ctrl+C')} to stop │
927
+ └─────────────────────────────────────────────┘
928
+ `));
929
+ server.on('request', (req) => {
930
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.cyan(req.method.padEnd(7)) + req.path);
931
+ });
932
+ process.on('SIGINT', async () => {
933
+ console.log(colors.yellow('\nShutting down...'));
934
+ await server.stop();
935
+ process.exit(0);
936
+ });
937
+ });
938
+ serve
939
+ .command('websocket')
940
+ .alias('ws')
941
+ .description('Start a mock WebSocket server')
942
+ .option('-p, --port <number>', 'Port to listen on', '8080')
943
+ .option('-h, --host <string>', 'Host to bind to', '127.0.0.1')
944
+ .option('--echo', 'Echo messages back (default: true)', true)
945
+ .option('--no-echo', 'Disable echo mode')
946
+ .option('--delay <ms>', 'Add delay to responses (milliseconds)', '0')
947
+ .addHelpText('after', `
948
+ ${colors.bold(colors.yellow('Examples:'))}
949
+ ${colors.green('$ rek serve websocket')} ${colors.gray('Start on port 8080')}
950
+ ${colors.green('$ rek serve ws -p 9000')} ${colors.gray('Start on port 9000')}
951
+ ${colors.green('$ rek serve ws --no-echo')} ${colors.gray('Disable echo')}
952
+ `)
953
+ .action(async (options) => {
954
+ const { MockWebSocketServer } = await import('../testing/mock-websocket-server.js');
955
+ const server = await MockWebSocketServer.create({
956
+ port: parseInt(options.port),
957
+ host: options.host,
958
+ echo: options.echo,
959
+ delay: parseInt(options.delay),
960
+ });
961
+ console.log(colors.green(`
962
+ ┌─────────────────────────────────────────────┐
963
+ │ ${colors.bold('Recker Mock WebSocket Server')} │
964
+ ├─────────────────────────────────────────────┤
965
+ │ URL: ${colors.cyan(server.url.padEnd(37))}│
966
+ │ Echo: ${colors.yellow((options.echo ? 'Enabled' : 'Disabled').padEnd(36))}│
967
+ │ Delay: ${colors.gray((options.delay + 'ms').padEnd(35))}│
968
+ ├─────────────────────────────────────────────┤
969
+ │ Press ${colors.bold('Ctrl+C')} to stop │
970
+ └─────────────────────────────────────────────┘
971
+ `));
972
+ server.on('connection', (client) => {
973
+ console.log(colors.green(`+ Connected: ${client.id}`));
974
+ });
975
+ server.on('message', (msg, client) => {
976
+ const data = msg.data.toString().slice(0, 50);
977
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.cyan(client.id) + ` ${data}${msg.data.toString().length > 50 ? '...' : ''}`);
978
+ });
979
+ server.on('disconnect', (client) => {
980
+ console.log(colors.red(`- Disconnected: ${client.id}`));
981
+ });
982
+ process.on('SIGINT', async () => {
983
+ console.log(colors.yellow('\nShutting down...'));
984
+ await server.stop();
985
+ process.exit(0);
986
+ });
987
+ });
988
+ serve
989
+ .command('sse')
990
+ .description('Start a mock SSE (Server-Sent Events) server')
991
+ .option('-p, --port <number>', 'Port to listen on', '8081')
992
+ .option('-h, --host <string>', 'Host to bind to', '127.0.0.1')
993
+ .option('--path <string>', 'SSE endpoint path', '/events')
994
+ .option('--heartbeat <ms>', 'Send heartbeat every N ms (0 = disabled)', '0')
995
+ .addHelpText('after', `
996
+ ${colors.bold(colors.yellow('Examples:'))}
997
+ ${colors.green('$ rek serve sse')} ${colors.gray('Start on port 8081')}
998
+ ${colors.green('$ rek serve sse -p 9000')} ${colors.gray('Start on port 9000')}
999
+ ${colors.green('$ rek serve sse --heartbeat 5000')} ${colors.gray('Send heartbeat every 5s')}
1000
+
1001
+ ${colors.bold(colors.yellow('Interactive Commands:'))}
1002
+ Type a message and press Enter to broadcast it to all clients.
1003
+ `)
1004
+ .action(async (options) => {
1005
+ const { MockSSEServer } = await import('../testing/mock-sse-server.js');
1006
+ const readline = await import('node:readline');
1007
+ const server = await MockSSEServer.create({
1008
+ port: parseInt(options.port),
1009
+ host: options.host,
1010
+ path: options.path,
1011
+ });
1012
+ if (parseInt(options.heartbeat) > 0) {
1013
+ server.startPeriodicEvents('heartbeat', parseInt(options.heartbeat));
1014
+ }
1015
+ console.log(colors.green(`
1016
+ ┌─────────────────────────────────────────────┐
1017
+ │ ${colors.bold('Recker Mock SSE Server')} │
1018
+ ├─────────────────────────────────────────────┤
1019
+ │ URL: ${colors.cyan(server.url.padEnd(37))}│
1020
+ │ Heartbeat: ${colors.yellow((options.heartbeat === '0' ? 'Disabled' : options.heartbeat + 'ms').padEnd(31))}│
1021
+ ├─────────────────────────────────────────────┤
1022
+ │ Type message + Enter to broadcast │
1023
+ │ Press ${colors.bold('Ctrl+C')} to stop │
1024
+ └─────────────────────────────────────────────┘
1025
+ `));
1026
+ server.on('connection', (client) => {
1027
+ console.log(colors.green(`+ Connected: ${client.id}`));
1028
+ });
1029
+ server.on('disconnect', (client) => {
1030
+ console.log(colors.red(`- Disconnected: ${client.id}`));
1031
+ });
1032
+ const rl = readline.createInterface({
1033
+ input: process.stdin,
1034
+ output: process.stdout,
1035
+ });
1036
+ rl.on('line', (line) => {
1037
+ if (line.trim()) {
1038
+ const sent = server.sendData(line.trim());
1039
+ console.log(colors.gray(`Broadcast to ${sent} client(s): ${line.trim()}`));
1040
+ }
1041
+ });
1042
+ process.on('SIGINT', async () => {
1043
+ console.log(colors.yellow('\nShutting down...'));
1044
+ rl.close();
1045
+ await server.stop();
1046
+ process.exit(0);
1047
+ });
1048
+ });
1049
+ serve
1050
+ .command('hls')
1051
+ .description('Start a mock HLS streaming server')
1052
+ .option('-p, --port <number>', 'Port to listen on', '8082')
1053
+ .option('-h, --host <string>', 'Host to bind to', '127.0.0.1')
1054
+ .option('--mode <type>', 'Stream mode: vod, live, event', 'vod')
1055
+ .option('--segments <number>', 'Number of segments (VOD mode)', '10')
1056
+ .option('--duration <seconds>', 'Segment duration in seconds', '6')
1057
+ .option('--qualities <list>', 'Comma-separated quality variants', '720p,480p,360p')
1058
+ .addHelpText('after', `
1059
+ ${colors.bold(colors.yellow('Examples:'))}
1060
+ ${colors.green('$ rek serve hls')} ${colors.gray('Start VOD server')}
1061
+ ${colors.green('$ rek serve hls --mode live')} ${colors.gray('Start live stream')}
1062
+ ${colors.green('$ rek serve hls --segments 20')} ${colors.gray('VOD with 20 segments')}
1063
+ ${colors.green('$ rek serve hls --qualities 1080p,720p,480p')}
1064
+
1065
+ ${colors.bold(colors.yellow('Endpoints:'))}
1066
+ ${colors.cyan('/master.m3u8')} Master playlist (multi-quality)
1067
+ ${colors.cyan('/playlist.m3u8')} Single quality playlist
1068
+ ${colors.cyan('/<quality>/playlist.m3u8')} Quality-specific playlist
1069
+ `)
1070
+ .action(async (options) => {
1071
+ const { MockHlsServer } = await import('../testing/mock-hls-server.js');
1072
+ const http = await import('node:http');
1073
+ const port = parseInt(options.port);
1074
+ const host = options.host;
1075
+ const qualities = options.qualities.split(',').map(q => q.trim());
1076
+ const resolutions = ['1920x1080', '1280x720', '854x480', '640x360', '426x240'];
1077
+ const bandwidths = [5000000, 2500000, 1400000, 800000, 500000];
1078
+ const variants = qualities.map((name, i) => ({
1079
+ name,
1080
+ bandwidth: bandwidths[i] || 500000,
1081
+ resolution: resolutions[i] || '640x360',
1082
+ }));
1083
+ const baseUrl = `http://${host}:${port}`;
1084
+ const hlsServer = await MockHlsServer.create({
1085
+ baseUrl,
1086
+ mode: options.mode,
1087
+ segmentCount: parseInt(options.segments),
1088
+ segmentDuration: parseInt(options.duration),
1089
+ multiQuality: variants.length > 1,
1090
+ variants: variants.length > 1 ? variants : undefined,
1091
+ });
1092
+ const httpServer = http.createServer(async (req, res) => {
1093
+ const url = `${baseUrl}${req.url}`;
1094
+ try {
1095
+ const response = await hlsServer.transport.dispatch({ url, method: req.method || 'GET' });
1096
+ res.statusCode = response.status;
1097
+ response.headers.forEach((value, key) => {
1098
+ res.setHeader(key, value);
1099
+ });
1100
+ const body = await response.arrayBuffer();
1101
+ res.end(Buffer.from(body));
1102
+ }
1103
+ catch {
1104
+ res.statusCode = 404;
1105
+ res.end('Not Found');
1106
+ }
1107
+ });
1108
+ httpServer.listen(port, host, () => {
1109
+ console.log(colors.green(`
1110
+ ┌─────────────────────────────────────────────┐
1111
+ │ ${colors.bold('Recker Mock HLS Server')} │
1112
+ ├─────────────────────────────────────────────┤
1113
+ │ Master: ${colors.cyan((hlsServer.manifestUrl).padEnd(34))}│
1114
+ │ Mode: ${colors.yellow(options.mode.padEnd(36))}│
1115
+ │ Segments: ${colors.gray(options.segments.padEnd(32))}│
1116
+ │ Duration: ${colors.gray((options.duration + 's').padEnd(32))}│
1117
+ │ Qualities: ${colors.cyan(qualities.join(', ').padEnd(31))}│
1118
+ ├─────────────────────────────────────────────┤
1119
+ │ Press ${colors.bold('Ctrl+C')} to stop │
1120
+ └─────────────────────────────────────────────┘
1121
+ `));
1122
+ });
1123
+ process.on('SIGINT', async () => {
1124
+ console.log(colors.yellow('\nShutting down...'));
1125
+ httpServer.close();
1126
+ await hlsServer.stop();
1127
+ process.exit(0);
1128
+ });
1129
+ });
1130
+ serve
1131
+ .command('udp')
1132
+ .description('Start a mock UDP server')
1133
+ .option('-p, --port <number>', 'Port to listen on', '9000')
1134
+ .option('-h, --host <string>', 'Host to bind to', '127.0.0.1')
1135
+ .option('--echo', 'Echo messages back (default: true)', true)
1136
+ .option('--no-echo', 'Disable echo mode')
1137
+ .addHelpText('after', `
1138
+ ${colors.bold(colors.yellow('Examples:'))}
1139
+ ${colors.green('$ rek serve udp')} ${colors.gray('Start on port 9000')}
1140
+ ${colors.green('$ rek serve udp -p 5353')} ${colors.gray('Start on port 5353')}
1141
+ ${colors.green('$ rek serve udp --no-echo')} ${colors.gray('Disable echo')}
1142
+ `)
1143
+ .action(async (options) => {
1144
+ const { MockUDPServer } = await import('../testing/mock-udp-server.js');
1145
+ const server = new MockUDPServer({
1146
+ port: parseInt(options.port),
1147
+ host: options.host,
1148
+ echo: options.echo,
1149
+ });
1150
+ await server.start();
1151
+ console.log(colors.green(`
1152
+ ┌─────────────────────────────────────────────┐
1153
+ │ ${colors.bold('Recker Mock UDP Server')} │
1154
+ ├─────────────────────────────────────────────┤
1155
+ │ Address: ${colors.cyan(`${options.host}:${options.port}`.padEnd(33))}│
1156
+ │ Echo: ${colors.yellow((options.echo ? 'Enabled' : 'Disabled').padEnd(36))}│
1157
+ ├─────────────────────────────────────────────┤
1158
+ │ Press ${colors.bold('Ctrl+C')} to stop │
1159
+ └─────────────────────────────────────────────┘
1160
+ `));
1161
+ server.on('message', (msg) => {
1162
+ const data = msg.data.toString().slice(0, 50);
1163
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.cyan(`${msg.rinfo.address}:${msg.rinfo.port}`) + ` ${data}${msg.data.toString().length > 50 ? '...' : ''}`);
1164
+ });
1165
+ process.on('SIGINT', async () => {
1166
+ console.log(colors.yellow('\nShutting down...'));
1167
+ await server.stop();
1168
+ process.exit(0);
1169
+ });
1170
+ });
1171
+ serve
1172
+ .command('dns')
1173
+ .description('Start a mock DNS server')
1174
+ .option('-p, --port <number>', 'Port to listen on', '5353')
1175
+ .option('-h, --host <string>', 'Host to bind to', '127.0.0.1')
1176
+ .option('--delay <ms>', 'Add delay to responses (milliseconds)', '0')
1177
+ .addHelpText('after', `
1178
+ ${colors.bold(colors.yellow('Examples:'))}
1179
+ ${colors.green('$ rek serve dns')} ${colors.gray('Start on port 5353')}
1180
+ ${colors.green('$ rek serve dns -p 53')} ${colors.gray('Start on standard port (requires root)')}
1181
+ ${colors.green('$ dig @127.0.0.1 -p 5353 example.com')} ${colors.gray('Test with dig')}
1182
+
1183
+ ${colors.bold(colors.yellow('Default Records:'))}
1184
+ ${colors.cyan('localhost')} A: 127.0.0.1, AAAA: ::1
1185
+ ${colors.cyan('example.com')} A, AAAA, NS, MX, TXT records
1186
+ ${colors.cyan('test.local')} A: 192.168.1.100
1187
+ `)
1188
+ .action(async (options) => {
1189
+ const { MockDnsServer } = await import('../testing/mock-dns-server.js');
1190
+ const server = await MockDnsServer.create({
1191
+ port: parseInt(options.port),
1192
+ host: options.host,
1193
+ delay: parseInt(options.delay),
1194
+ });
1195
+ console.log(colors.green(`
1196
+ ┌─────────────────────────────────────────────┐
1197
+ │ ${colors.bold('Recker Mock DNS Server')} │
1198
+ ├─────────────────────────────────────────────┤
1199
+ │ Address: ${colors.cyan(`${options.host}:${options.port}`.padEnd(33))}│
1200
+ │ Protocol: ${colors.yellow('UDP'.padEnd(32))}│
1201
+ ├─────────────────────────────────────────────┤
1202
+ │ Test: dig @${options.host} -p ${options.port} example.com │
1203
+ │ Press ${colors.bold('Ctrl+C')} to stop │
1204
+ └─────────────────────────────────────────────┘
1205
+ `));
1206
+ server.on('query', (query) => {
1207
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.cyan(query.type.padEnd(6)) + ` ${query.domain}`);
1208
+ });
1209
+ process.on('SIGINT', async () => {
1210
+ console.log(colors.yellow('\nShutting down...'));
1211
+ await server.stop();
1212
+ process.exit(0);
1213
+ });
1214
+ });
1215
+ serve
1216
+ .command('whois')
1217
+ .description('Start a mock WHOIS server')
1218
+ .option('-p, --port <number>', 'Port to listen on', '4343')
1219
+ .option('-h, --host <string>', 'Host to bind to', '127.0.0.1')
1220
+ .option('--delay <ms>', 'Add delay to responses (milliseconds)', '0')
1221
+ .addHelpText('after', `
1222
+ ${colors.bold(colors.yellow('Examples:'))}
1223
+ ${colors.green('$ rek serve whois')} ${colors.gray('Start on port 4343')}
1224
+ ${colors.green('$ whois -h 127.0.0.1 -p 4343 example.com')} ${colors.gray('Test with whois')}
1225
+
1226
+ ${colors.bold(colors.yellow('Default Domains:'))}
1227
+ ${colors.cyan('example.com')} IANA reserved domain
1228
+ ${colors.cyan('google.com')} MarkMonitor registrar
1229
+ ${colors.cyan('test.local')} Test domain
1230
+ `)
1231
+ .action(async (options) => {
1232
+ const { MockWhoisServer } = await import('../testing/mock-whois-server.js');
1233
+ const server = await MockWhoisServer.create({
1234
+ port: parseInt(options.port),
1235
+ host: options.host,
1236
+ delay: parseInt(options.delay),
1237
+ });
1238
+ console.log(colors.green(`
1239
+ ┌─────────────────────────────────────────────┐
1240
+ │ ${colors.bold('Recker Mock WHOIS Server')} │
1241
+ ├─────────────────────────────────────────────┤
1242
+ │ Address: ${colors.cyan(`${options.host}:${options.port}`.padEnd(33))}│
1243
+ │ Protocol: ${colors.yellow('TCP'.padEnd(32))}│
1244
+ ├─────────────────────────────────────────────┤
1245
+ │ Test: whois -h ${options.host} -p ${options.port} example.com │
1246
+ │ Press ${colors.bold('Ctrl+C')} to stop │
1247
+ └─────────────────────────────────────────────┘
1248
+ `));
1249
+ server.on('query', (query) => {
1250
+ console.log(colors.gray(`${new Date().toISOString()} `) + `Query: ${colors.cyan(query)}`);
1251
+ });
1252
+ process.on('SIGINT', async () => {
1253
+ console.log(colors.yellow('\nShutting down...'));
1254
+ await server.stop();
1255
+ process.exit(0);
1256
+ });
1257
+ });
1258
+ serve
1259
+ .command('telnet')
1260
+ .description('Start a mock Telnet server')
1261
+ .option('-p, --port <number>', 'Port to listen on', '2323')
1262
+ .option('-h, --host <string>', 'Host to bind to', '127.0.0.1')
1263
+ .option('--echo', 'Echo input back (default: true)', true)
1264
+ .option('--no-echo', 'Disable echo mode')
1265
+ .option('--delay <ms>', 'Add delay to responses (milliseconds)', '0')
1266
+ .addHelpText('after', `
1267
+ ${colors.bold(colors.yellow('Examples:'))}
1268
+ ${colors.green('$ rek serve telnet')} ${colors.gray('Start on port 2323')}
1269
+ ${colors.green('$ telnet localhost 2323')} ${colors.gray('Connect to server')}
1270
+
1271
+ ${colors.bold(colors.yellow('Built-in Commands:'))}
1272
+ ${colors.cyan('help')} Show available commands
1273
+ ${colors.cyan('echo <msg>')} Echo message back
1274
+ ${colors.cyan('date')} Show current date
1275
+ ${colors.cyan('time')} Show current time
1276
+ ${colors.cyan('ping')} Returns "pong"
1277
+ ${colors.cyan('quit')} Disconnect
1278
+ `)
1279
+ .action(async (options) => {
1280
+ const { MockTelnetServer } = await import('../testing/mock-telnet-server.js');
1281
+ const server = await MockTelnetServer.create({
1282
+ port: parseInt(options.port),
1283
+ host: options.host,
1284
+ echo: options.echo,
1285
+ delay: parseInt(options.delay),
1286
+ });
1287
+ console.log(colors.green(`
1288
+ ┌─────────────────────────────────────────────┐
1289
+ │ ${colors.bold('Recker Mock Telnet Server')} │
1290
+ ├─────────────────────────────────────────────┤
1291
+ │ Address: ${colors.cyan(`${options.host}:${options.port}`.padEnd(33))}│
1292
+ │ Echo: ${colors.yellow((options.echo ? 'Enabled' : 'Disabled').padEnd(36))}│
1293
+ ├─────────────────────────────────────────────┤
1294
+ │ Connect: telnet ${options.host} ${options.port} │
1295
+ │ Press ${colors.bold('Ctrl+C')} to stop │
1296
+ └─────────────────────────────────────────────┘
1297
+ `));
1298
+ server.on('connect', (session) => {
1299
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.green('+ Connected: ') + colors.cyan(session.id));
1300
+ });
1301
+ server.on('disconnect', (session) => {
1302
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.red('- Disconnected: ') + colors.cyan(session.id));
1303
+ });
1304
+ server.on('command', (cmd, session) => {
1305
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.cyan(session.id) + ` $ ${cmd}`);
1306
+ });
1307
+ process.on('SIGINT', async () => {
1308
+ console.log(colors.yellow('\nShutting down...'));
1309
+ await server.stop();
1310
+ process.exit(0);
1311
+ });
1312
+ });
1313
+ serve
1314
+ .command('ftp')
1315
+ .description('Start a mock FTP server')
1316
+ .option('-p, --port <number>', 'Port to listen on', '2121')
1317
+ .option('-h, --host <string>', 'Host to bind to', '127.0.0.1')
1318
+ .option('-u, --username <user>', 'Username for auth', 'user')
1319
+ .option('--password <pass>', 'Password for auth', 'pass')
1320
+ .option('--anonymous', 'Allow anonymous login (default: true)', true)
1321
+ .option('--no-anonymous', 'Disable anonymous login')
1322
+ .option('--delay <ms>', 'Add delay to responses (milliseconds)', '0')
1323
+ .addHelpText('after', `
1324
+ ${colors.bold(colors.yellow('Examples:'))}
1325
+ ${colors.green('$ rek serve ftp')} ${colors.gray('Start on port 2121')}
1326
+ ${colors.green('$ ftp localhost 2121')} ${colors.gray('Connect to server')}
1327
+ ${colors.green('$ rek serve ftp --no-anonymous')} ${colors.gray('Require authentication')}
1328
+
1329
+ ${colors.bold(colors.yellow('Default Files:'))}
1330
+ ${colors.cyan('/welcome.txt')} Welcome message
1331
+ ${colors.cyan('/readme.md')} README file
1332
+ ${colors.cyan('/data/sample.json')} Sample JSON data
1333
+ ${colors.cyan('/public/index.html')} HTML file
1334
+
1335
+ ${colors.bold(colors.yellow('Credentials:'))}
1336
+ Username: ${colors.cyan('user')} Password: ${colors.cyan('pass')}
1337
+ Or use anonymous login with user: ${colors.cyan('anonymous')}
1338
+ `)
1339
+ .action(async (options) => {
1340
+ const { MockFtpServer } = await import('../testing/mock-ftp-server.js');
1341
+ const server = await MockFtpServer.create({
1342
+ port: parseInt(options.port),
1343
+ host: options.host,
1344
+ username: options.username,
1345
+ password: options.password,
1346
+ anonymous: options.anonymous,
1347
+ delay: parseInt(options.delay),
1348
+ });
1349
+ console.log(colors.green(`
1350
+ ┌─────────────────────────────────────────────┐
1351
+ │ ${colors.bold('Recker Mock FTP Server')} │
1352
+ ├─────────────────────────────────────────────┤
1353
+ │ Address: ${colors.cyan(`${options.host}:${options.port}`.padEnd(33))}│
1354
+ │ Anonymous: ${colors.yellow((options.anonymous ? 'Allowed' : 'Disabled').padEnd(31))}│
1355
+ │ User: ${colors.cyan(options.username.padEnd(36))}│
1356
+ ├─────────────────────────────────────────────┤
1357
+ │ Connect: ftp ${options.host} ${options.port} │
1358
+ │ Press ${colors.bold('Ctrl+C')} to stop │
1359
+ └─────────────────────────────────────────────┘
1360
+ `));
1361
+ server.on('connect', (session) => {
1362
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.green('+ Connected: ') + colors.cyan(session.id));
1363
+ });
1364
+ server.on('disconnect', (session) => {
1365
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.red('- Disconnected: ') + colors.cyan(session.id));
1366
+ });
1367
+ server.on('command', (cmd, _args, session) => {
1368
+ console.log(colors.gray(`${new Date().toISOString()} `) + colors.cyan(session.id) + ` ${cmd}`);
1369
+ });
1370
+ process.on('SIGINT', async () => {
1371
+ console.log(colors.yellow('\nShutting down...'));
1372
+ await server.stop();
1373
+ process.exit(0);
1374
+ });
1375
+ });
883
1376
  program
884
1377
  .command('mcp')
885
1378
  .description('Start MCP server for AI agents to access Recker documentation')
@@ -1716,7 +1716,7 @@ ${colors.bold('Network:')}
1716
1716
  console.log(colors.gray(` Names: ${mapData.names.length}`));
1717
1717
  console.log(colors.bold('\nOriginal sources:'));
1718
1718
  mapData.sources.forEach((source, i) => {
1719
- const hasContent = mapData.sourcesContent && mapData.sourcesContent[i];
1719
+ const hasContent = mapData.sourcesContent?.[i];
1720
1720
  const sizeInfo = hasContent
1721
1721
  ? colors.green(`[${(mapData.sourcesContent[i].length / 1024).toFixed(1)}kb]`)
1722
1722
  : colors.yellow('[no content]');
@@ -7,6 +7,7 @@ import { DedupOptions } from '../plugins/dedup.js';
7
7
  import { ReckerWebSocket, type WebSocketOptions } from '../websocket/client.js';
8
8
  import { type WhoisOptions, type WhoisResult } from '../utils/whois.js';
9
9
  import { type ScrapePromise } from '../plugins/scrape.js';
10
+ import { HlsPromise, type HlsOptions } from '../plugins/hls.js';
10
11
  interface ClientCacheConfig extends Omit<CacheOptions, 'storage'> {
11
12
  storage?: CacheStorage;
12
13
  driver?: 'memory' | 'file';
@@ -111,6 +112,7 @@ export declare class Client {
111
112
  ws(path: string, options?: WebSocketOptions): ReckerWebSocket;
112
113
  whois(query: string, options?: WhoisOptions): Promise<WhoisResult>;
113
114
  isDomainAvailable(domain: string, options?: WhoisOptions): Promise<boolean>;
115
+ hls(manifestUrl: string, options?: HlsOptions): HlsPromise;
114
116
  }
115
117
  export declare function createClient(options?: ExtendedClientOptions): Client;
116
118
  export {};
@@ -1 +1 @@
1
- {"version":3,"file":"client.d.ts","sourceRoot":"","sources":["../../src/core/client.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"client.d.ts","sourceRoot":"","sources":["../../src/core/client.ts"],"names":[],"mappings":"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"}
@@ -22,6 +22,7 @@ import { ReckerWebSocket } from '../websocket/client.js';
22
22
  import { whois as performWhois, isDomainAvailable } from '../utils/whois.js';
23
23
  import { MemoryCookieJar } from '../cookies/memory-cookie-jar.js';
24
24
  import { scrape as scrapeHelper } from '../plugins/scrape.js';
25
+ import { HlsPromise } from '../plugins/hls.js';
25
26
  export class Client {
26
27
  baseUrl;
27
28
  middlewares;
@@ -657,6 +658,9 @@ export class Client {
657
658
  async isDomainAvailable(domain, options) {
658
659
  return isDomainAvailable(domain, options);
659
660
  }
661
+ hls(manifestUrl, options = {}) {
662
+ return new HlsPromise(this, manifestUrl, options);
663
+ }
660
664
  }
661
665
  export function createClient(options = {}) {
662
666
  return new Client(options);
@@ -293,7 +293,7 @@ export function cache(options = {}) {
293
293
  key = generateKey(req, baseEntry.vary);
294
294
  cachedEntry = await storage.get(key);
295
295
  }
296
- else if (baseEntry && baseEntry.body) {
296
+ else if (baseEntry?.body) {
297
297
  cachedEntry = baseEntry;
298
298
  }
299
299
  else {