@stacksjs/ts-cloud 0.1.9 → 0.1.14

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 (150) hide show
  1. package/README.md +39 -377
  2. package/dist/bin/cli.js +1047 -424
  3. package/dist/index.d.ts +36 -3
  4. package/dist/index.js +76430 -7096
  5. package/package.json +7 -8
  6. package/dist/aws/acm.d.ts +0 -129
  7. package/dist/aws/application-autoscaling.d.ts +0 -282
  8. package/dist/aws/bedrock.d.ts +0 -2292
  9. package/dist/aws/client.d.ts +0 -79
  10. package/dist/aws/cloudformation.d.ts +0 -105
  11. package/dist/aws/cloudfront.d.ts +0 -265
  12. package/dist/aws/cloudwatch-logs.d.ts +0 -48
  13. package/dist/aws/comprehend.d.ts +0 -505
  14. package/dist/aws/connect.d.ts +0 -377
  15. package/dist/aws/deploy-imap.d.ts +0 -14
  16. package/dist/aws/dynamodb.d.ts +0 -176
  17. package/dist/aws/ec2.d.ts +0 -272
  18. package/dist/aws/ecr.d.ts +0 -149
  19. package/dist/aws/ecs.d.ts +0 -162
  20. package/dist/aws/elasticache.d.ts +0 -71
  21. package/dist/aws/elbv2.d.ts +0 -248
  22. package/dist/aws/email.d.ts +0 -175
  23. package/dist/aws/eventbridge.d.ts +0 -142
  24. package/dist/aws/iam.d.ts +0 -638
  25. package/dist/aws/imap-server.d.ts +0 -119
  26. package/dist/aws/index.d.ts +0 -192
  27. package/dist/aws/kendra.d.ts +0 -782
  28. package/dist/aws/lambda.d.ts +0 -232
  29. package/dist/aws/opensearch.d.ts +0 -87
  30. package/dist/aws/personalize.d.ts +0 -516
  31. package/dist/aws/polly.d.ts +0 -214
  32. package/dist/aws/rds.d.ts +0 -240
  33. package/dist/aws/rekognition.d.ts +0 -543
  34. package/dist/aws/route53-domains.d.ts +0 -113
  35. package/dist/aws/route53.d.ts +0 -215
  36. package/dist/aws/s3.d.ts +0 -212
  37. package/dist/aws/scheduler.d.ts +0 -140
  38. package/dist/aws/secrets-manager.d.ts +0 -170
  39. package/dist/aws/ses.d.ts +0 -288
  40. package/dist/aws/setup-phone.d.ts +0 -0
  41. package/dist/aws/setup-sms.d.ts +0 -115
  42. package/dist/aws/sms.d.ts +0 -304
  43. package/dist/aws/smtp-server.d.ts +0 -61
  44. package/dist/aws/sns.d.ts +0 -117
  45. package/dist/aws/sqs.d.ts +0 -65
  46. package/dist/aws/ssm.d.ts +0 -179
  47. package/dist/aws/sts.d.ts +0 -15
  48. package/dist/aws/support.d.ts +0 -104
  49. package/dist/aws/test-imap.d.ts +0 -0
  50. package/dist/aws/textract.d.ts +0 -403
  51. package/dist/aws/transcribe.d.ts +0 -60
  52. package/dist/aws/translate.d.ts +0 -358
  53. package/dist/aws/voice.d.ts +0 -219
  54. package/dist/config.d.ts +0 -7
  55. package/dist/deploy/index.d.ts +0 -2
  56. package/dist/deploy/static-site-external-dns.d.ts +0 -51
  57. package/dist/deploy/static-site.d.ts +0 -71
  58. package/dist/dns/cloudflare.d.ts +0 -52
  59. package/dist/dns/godaddy.d.ts +0 -38
  60. package/dist/dns/index.d.ts +0 -45
  61. package/dist/dns/porkbun.d.ts +0 -18
  62. package/dist/dns/route53-adapter.d.ts +0 -38
  63. package/dist/dns/types.d.ts +0 -77
  64. package/dist/dns/validator.d.ts +0 -78
  65. package/dist/generators/index.d.ts +0 -1
  66. package/dist/generators/infrastructure.d.ts +0 -30
  67. package/dist/push/apns.d.ts +0 -60
  68. package/dist/push/fcm.d.ts +0 -117
  69. package/dist/push/index.d.ts +0 -14
  70. package/dist/security/pre-deploy-scanner.d.ts +0 -69
  71. package/dist/ssl/acme-client.d.ts +0 -67
  72. package/dist/ssl/index.d.ts +0 -2
  73. package/dist/ssl/letsencrypt.d.ts +0 -48
  74. package/dist/types.d.ts +0 -1
  75. package/dist/utils/cli.d.ts +0 -123
  76. package/dist/validation/index.d.ts +0 -1
  77. package/dist/validation/template.d.ts +0 -23
  78. package/src/aws/acm.ts +0 -768
  79. package/src/aws/application-autoscaling.ts +0 -845
  80. package/src/aws/bedrock.ts +0 -4074
  81. package/src/aws/client.ts +0 -891
  82. package/src/aws/cloudformation.ts +0 -896
  83. package/src/aws/cloudfront.ts +0 -1531
  84. package/src/aws/cloudwatch-logs.ts +0 -154
  85. package/src/aws/comprehend.ts +0 -839
  86. package/src/aws/connect.ts +0 -1056
  87. package/src/aws/deploy-imap.ts +0 -384
  88. package/src/aws/dynamodb.ts +0 -340
  89. package/src/aws/ec2.ts +0 -1385
  90. package/src/aws/ecr.ts +0 -621
  91. package/src/aws/ecs.ts +0 -615
  92. package/src/aws/elasticache.ts +0 -301
  93. package/src/aws/elbv2.ts +0 -942
  94. package/src/aws/email.ts +0 -928
  95. package/src/aws/eventbridge.ts +0 -248
  96. package/src/aws/iam.ts +0 -1689
  97. package/src/aws/imap-server.ts +0 -2100
  98. package/src/aws/index.ts +0 -213
  99. package/src/aws/kendra.ts +0 -1097
  100. package/src/aws/lambda.ts +0 -786
  101. package/src/aws/opensearch.ts +0 -158
  102. package/src/aws/personalize.ts +0 -977
  103. package/src/aws/polly.ts +0 -559
  104. package/src/aws/rds.ts +0 -888
  105. package/src/aws/rekognition.ts +0 -846
  106. package/src/aws/route53-domains.ts +0 -359
  107. package/src/aws/route53.ts +0 -1046
  108. package/src/aws/s3.ts +0 -2334
  109. package/src/aws/scheduler.ts +0 -571
  110. package/src/aws/secrets-manager.ts +0 -769
  111. package/src/aws/ses.ts +0 -1081
  112. package/src/aws/setup-phone.ts +0 -104
  113. package/src/aws/setup-sms.ts +0 -580
  114. package/src/aws/sms.ts +0 -1735
  115. package/src/aws/smtp-server.ts +0 -531
  116. package/src/aws/sns.ts +0 -758
  117. package/src/aws/sqs.ts +0 -382
  118. package/src/aws/ssm.ts +0 -807
  119. package/src/aws/sts.ts +0 -92
  120. package/src/aws/support.ts +0 -391
  121. package/src/aws/test-imap.ts +0 -86
  122. package/src/aws/textract.ts +0 -780
  123. package/src/aws/transcribe.ts +0 -108
  124. package/src/aws/translate.ts +0 -641
  125. package/src/aws/voice.ts +0 -1379
  126. package/src/config.ts +0 -35
  127. package/src/deploy/index.ts +0 -7
  128. package/src/deploy/static-site-external-dns.ts +0 -945
  129. package/src/deploy/static-site.ts +0 -1175
  130. package/src/dns/cloudflare.ts +0 -548
  131. package/src/dns/godaddy.ts +0 -412
  132. package/src/dns/index.ts +0 -205
  133. package/src/dns/porkbun.ts +0 -362
  134. package/src/dns/route53-adapter.ts +0 -414
  135. package/src/dns/types.ts +0 -119
  136. package/src/dns/validator.ts +0 -369
  137. package/src/generators/index.ts +0 -5
  138. package/src/generators/infrastructure.ts +0 -1660
  139. package/src/index.ts +0 -163
  140. package/src/push/apns.ts +0 -452
  141. package/src/push/fcm.ts +0 -506
  142. package/src/push/index.ts +0 -58
  143. package/src/security/pre-deploy-scanner.ts +0 -655
  144. package/src/ssl/acme-client.ts +0 -478
  145. package/src/ssl/index.ts +0 -7
  146. package/src/ssl/letsencrypt.ts +0 -747
  147. package/src/types.ts +0 -2
  148. package/src/utils/cli.ts +0 -398
  149. package/src/validation/index.ts +0 -5
  150. package/src/validation/template.ts +0 -405
package/src/index.ts DELETED
@@ -1,163 +0,0 @@
1
- export * from './config'
2
- // Note: ./types re-exports @stacksjs/ts-cloud-types, which we export below
3
- // export * from './types'
4
- export * from './generators'
5
-
6
- // Validation exports - functions
7
- export {
8
- validateTemplate,
9
- validateTemplateSize,
10
- validateResourceLimits,
11
- } from './validation'
12
-
13
- // Validation exports - types with prefixed names for conflicts with @stacksjs/ts-cloud-core
14
- export type {
15
- ValidationError as TemplateValidationError,
16
- ValidationResult as TemplateValidationResult,
17
- } from './validation'
18
-
19
- // Export AWS module - classes and functions
20
- export {
21
- AWSClient,
22
- CloudFormationClient as AWSCloudFormationClient,
23
- CloudFrontClient as AWSCloudFrontClient,
24
- EC2Client,
25
- S3Client,
26
- Route53Client,
27
- Route53DomainsClient,
28
- ACMClient,
29
- ACMDnsValidator,
30
- ECRClient,
31
- ECSClient,
32
- STSClient,
33
- SSMClient,
34
- SecretsManagerClient,
35
- SESClient,
36
- EmailClient,
37
- SNSClient,
38
- SQSClient,
39
- LambdaClient,
40
- CloudWatchLogsClient,
41
- ConnectClient,
42
- ELBv2Client,
43
- RDSClient,
44
- DynamoDBClient,
45
- OpenSearchClient,
46
- TranscribeClient,
47
- BedrockRuntimeClient,
48
- ComprehendClient,
49
- RekognitionClient,
50
- TextractClient,
51
- PollyClient,
52
- TranslateClient,
53
- PersonalizeClient,
54
- KendraClient,
55
- EventBridgeClient,
56
- ElastiCacheClient,
57
- SchedulerClient,
58
- IAMClient,
59
- ApplicationAutoScalingClient,
60
- SmsClient,
61
- VoiceClient,
62
- SupportClient,
63
- } from './aws'
64
-
65
- // Export AWS module - types with prefixed names where needed
66
- export type {
67
- AWSRequestOptions,
68
- AWSClientConfig,
69
- AWSError,
70
- AWSCredentials as AWSClientCredentials,
71
- StackParameter,
72
- StackTag,
73
- CreateStackOptions,
74
- UpdateStackOptions,
75
- DescribeStacksOptions,
76
- StackEvent,
77
- Stack,
78
- InvalidationOptions,
79
- Distribution,
80
- S3SyncOptions,
81
- S3CopyOptions,
82
- S3ListOptions,
83
- S3Object,
84
- CertificateDetail,
85
- Certificate as ELBv2Certificate,
86
- RekognitionS3Object,
87
- RekognitionBoundingBox,
88
- TextractS3Object,
89
- TextractBoundingBox,
90
- KendraCreateDataSourceCommandInput,
91
- KendraCreateDataSourceCommandOutput,
92
- KendraListDataSourcesCommandInput,
93
- KendraListDataSourcesCommandOutput,
94
- } from './aws'
95
-
96
- export * from './ssl'
97
-
98
- // Export deployment modules
99
- export {
100
- deployStaticSite,
101
- deployStaticSiteFull,
102
- uploadStaticFiles,
103
- invalidateCache,
104
- deleteStaticSite,
105
- generateStaticSiteTemplate,
106
- // External DNS support
107
- deployStaticSiteWithExternalDns,
108
- deployStaticSiteWithExternalDnsFull,
109
- generateExternalDnsStaticSiteTemplate,
110
- } from './deploy'
111
- export type {
112
- StaticSiteConfig,
113
- DeployResult,
114
- UploadOptions,
115
- // External DNS types
116
- ExternalDnsStaticSiteConfig,
117
- ExternalDnsDeployResult,
118
- } from './deploy'
119
-
120
- // Export DNS providers
121
- export {
122
- createDnsProvider,
123
- detectDnsProvider,
124
- DnsProviderFactory,
125
- dnsProviders,
126
- PorkbunProvider,
127
- GoDaddyProvider,
128
- Route53Provider,
129
- UnifiedDnsValidator,
130
- createPorkbunValidator,
131
- createGoDaddyValidator,
132
- createRoute53Validator,
133
- } from './dns'
134
- export type {
135
- DnsProvider,
136
- DnsProviderConfig,
137
- DnsRecord,
138
- DnsRecordType,
139
- DnsRecordResult,
140
- CreateRecordResult,
141
- DeleteRecordResult,
142
- ListRecordsResult,
143
- } from './dns'
144
-
145
- // Re-export core functionality (these take precedence for common types)
146
- export * from '@stacksjs/ts-cloud-core'
147
-
148
- // Re-export @stacksjs/ts-cloud-types (includes VpcConfig, etc.)
149
- export * from '@stacksjs/ts-cloud-types'
150
-
151
- // Re-export @stacksjs/ts-cloud-aws-types with explicit handling for duplicates
152
- // Note: @stacksjs/ts-cloud-core also exports CloudFormation* types, so we skip re-exporting them here
153
- // to avoid duplicates. Users can import directly from @stacksjs/ts-cloud-aws-types if needed.
154
- export type {
155
- // S3 types
156
- S3Bucket,
157
- S3BucketPolicy,
158
- // CloudFront types
159
- CloudFrontDistribution,
160
- CloudFrontOriginAccessControl,
161
- CloudFrontCacheBehavior,
162
- CloudFrontOrigin,
163
- } from '@stacksjs/ts-cloud-aws-types'
package/src/push/apns.ts DELETED
@@ -1,452 +0,0 @@
1
- /**
2
- * Apple Push Notification Service (APNs) Client
3
- * Uses HTTP/2 with JWT token authentication
4
- *
5
- * Prerequisites:
6
- * - Apple Developer account
7
- * - APNs Key (p8 file) from Apple Developer Portal
8
- * - Key ID and Team ID
9
- *
10
- * @example
11
- * ```ts
12
- * const apns = new APNsClient({
13
- * keyId: 'ABC123DEFG',
14
- * teamId: 'DEF456GHIJ',
15
- * privateKey: fs.readFileSync('AuthKey_ABC123DEFG.p8', 'utf8'),
16
- * bundleId: 'com.example.app',
17
- * production: false // true for production, false for sandbox
18
- * })
19
- *
20
- * await apns.send({
21
- * deviceToken: '...',
22
- * title: 'Hello',
23
- * body: 'World',
24
- * })
25
- * ```
26
- */
27
-
28
- import { createSign } from 'node:crypto'
29
- import * as http2 from 'node:http2'
30
-
31
- export interface APNsConfig {
32
- /** APNs Key ID from Apple Developer Portal */
33
- keyId: string
34
- /** Team ID from Apple Developer Portal */
35
- teamId: string
36
- /** Private key content (p8 file content) */
37
- privateKey: string
38
- /** iOS app bundle ID (e.g., com.example.app) */
39
- bundleId: string
40
- /** Use production APNs server (default: false) */
41
- production?: boolean
42
- }
43
-
44
- export interface APNsNotification {
45
- /** Device token to send to */
46
- deviceToken: string
47
- /** Alert title */
48
- title?: string
49
- /** Alert subtitle */
50
- subtitle?: string
51
- /** Alert body */
52
- body?: string
53
- /** Badge number to display on app icon */
54
- badge?: number
55
- /** Sound to play (use 'default' for default sound) */
56
- sound?: string | { name: string; critical?: number; volume?: number }
57
- /** Category identifier for actionable notifications */
58
- category?: string
59
- /** Thread identifier for grouping notifications */
60
- threadId?: string
61
- /** Custom data payload */
62
- data?: Record<string, any>
63
- /** Content available flag for background updates */
64
- contentAvailable?: boolean
65
- /** Mutable content flag for notification service extension */
66
- mutableContent?: boolean
67
- /** Push type (default: 'alert') */
68
- pushType?: 'alert' | 'background' | 'voip' | 'complication' | 'fileprovider' | 'mdm' | 'liveactivity'
69
- /** Notification priority (10 = immediate, 5 = can be delayed) */
70
- priority?: 5 | 10
71
- /** Expiration timestamp (Unix time in seconds) */
72
- expiration?: number
73
- /** Collapse identifier for coalescing notifications */
74
- collapseId?: string
75
- /** Target content id for live activities */
76
- targetContentId?: string
77
- }
78
-
79
- export interface APNsSendResult {
80
- success: boolean
81
- deviceToken: string
82
- apnsId?: string
83
- statusCode?: number
84
- error?: string
85
- reason?: string
86
- timestamp?: number
87
- }
88
-
89
- export interface APNsBatchResult {
90
- sent: number
91
- failed: number
92
- results: APNsSendResult[]
93
- }
94
-
95
- const APNS_PRODUCTION_HOST = 'api.push.apple.com'
96
- const APNS_SANDBOX_HOST = 'api.sandbox.push.apple.com'
97
- const TOKEN_EXPIRY_MS = 45 * 60 * 1000 // 45 minutes (tokens valid for 1 hour)
98
-
99
- /**
100
- * Apple Push Notification Service client
101
- */
102
- export class APNsClient {
103
- private config: APNsConfig
104
- private token: string | null = null
105
- private tokenGeneratedAt: number = 0
106
- private client: http2.ClientHttp2Session | null = null
107
- private host: string
108
-
109
- constructor(config: APNsConfig) {
110
- this.config = config
111
- this.host = config.production ? APNS_PRODUCTION_HOST : APNS_SANDBOX_HOST
112
- }
113
-
114
- /**
115
- * Generate a new JWT token for APNs authentication
116
- */
117
- private generateToken(): string {
118
- const now = Math.floor(Date.now() / 1000)
119
-
120
- // Check if current token is still valid
121
- if (this.token && (Date.now() - this.tokenGeneratedAt) < TOKEN_EXPIRY_MS) {
122
- return this.token
123
- }
124
-
125
- // JWT Header
126
- const header = {
127
- alg: 'ES256',
128
- kid: this.config.keyId,
129
- }
130
-
131
- // JWT Payload
132
- const payload = {
133
- iss: this.config.teamId,
134
- iat: now,
135
- }
136
-
137
- // Encode header and payload
138
- const encodedHeader = Buffer.from(JSON.stringify(header)).toString('base64url')
139
- const encodedPayload = Buffer.from(JSON.stringify(payload)).toString('base64url')
140
- const signatureInput = `${encodedHeader}.${encodedPayload}`
141
-
142
- // Sign with ES256 (ECDSA P-256)
143
- const sign = createSign('SHA256')
144
- sign.update(signatureInput)
145
- const signature = sign.sign(this.config.privateKey)
146
-
147
- // Convert DER signature to raw format (64 bytes)
148
- const rawSignature = this.derToRaw(signature)
149
- const encodedSignature = rawSignature.toString('base64url')
150
-
151
- this.token = `${signatureInput}.${encodedSignature}`
152
- this.tokenGeneratedAt = Date.now()
153
-
154
- return this.token
155
- }
156
-
157
- /**
158
- * Convert DER encoded ECDSA signature to raw format
159
- */
160
- private derToRaw(derSignature: Buffer): Buffer {
161
- // DER format: 0x30 [total-length] 0x02 [r-length] [r] 0x02 [s-length] [s]
162
- let offset = 2 // Skip 0x30 and total length
163
-
164
- // Read R
165
- if (derSignature[offset] !== 0x02) {
166
- throw new Error('Invalid DER signature: expected 0x02 for R')
167
- }
168
- offset++
169
- const rLength = derSignature[offset]
170
- offset++
171
- let r = derSignature.subarray(offset, offset + rLength)
172
- offset += rLength
173
-
174
- // Read S
175
- if (derSignature[offset] !== 0x02) {
176
- throw new Error('Invalid DER signature: expected 0x02 for S')
177
- }
178
- offset++
179
- const sLength = derSignature[offset]
180
- offset++
181
- let s = derSignature.subarray(offset, offset + sLength)
182
-
183
- // Remove leading zeros and pad to 32 bytes
184
- if (r[0] === 0x00 && r.length === 33) {
185
- r = r.subarray(1)
186
- }
187
- if (s[0] === 0x00 && s.length === 33) {
188
- s = s.subarray(1)
189
- }
190
-
191
- // Pad to 32 bytes if needed
192
- const result = Buffer.alloc(64)
193
- r.copy(result, 32 - r.length)
194
- s.copy(result, 64 - s.length)
195
-
196
- return result
197
- }
198
-
199
- /**
200
- * Get or create HTTP/2 client connection
201
- */
202
- private async getClient(): Promise<http2.ClientHttp2Session> {
203
- if (this.client && !this.client.destroyed) {
204
- return this.client
205
- }
206
-
207
- return new Promise((resolve, reject) => {
208
- this.client = http2.connect(`https://${this.host}`)
209
-
210
- this.client.on('error', (err) => {
211
- reject(err)
212
- })
213
-
214
- this.client.on('connect', () => {
215
- resolve(this.client!)
216
- })
217
-
218
- // Set up automatic reconnection on close
219
- this.client.on('close', () => {
220
- this.client = null
221
- })
222
- })
223
- }
224
-
225
- /**
226
- * Build APNs payload from notification options
227
- */
228
- private buildPayload(notification: APNsNotification): object {
229
- const aps: Record<string, any> = {}
230
-
231
- // Alert
232
- if (notification.title || notification.body || notification.subtitle) {
233
- aps.alert = {}
234
- if (notification.title) aps.alert.title = notification.title
235
- if (notification.subtitle) aps.alert.subtitle = notification.subtitle
236
- if (notification.body) aps.alert.body = notification.body
237
- }
238
-
239
- // Badge
240
- if (notification.badge !== undefined) {
241
- aps.badge = notification.badge
242
- }
243
-
244
- // Sound
245
- if (notification.sound !== undefined) {
246
- aps.sound = notification.sound
247
- }
248
-
249
- // Category
250
- if (notification.category) {
251
- aps.category = notification.category
252
- }
253
-
254
- // Thread ID
255
- if (notification.threadId) {
256
- aps['thread-id'] = notification.threadId
257
- }
258
-
259
- // Content available (for background updates)
260
- if (notification.contentAvailable) {
261
- aps['content-available'] = 1
262
- }
263
-
264
- // Mutable content (for notification service extension)
265
- if (notification.mutableContent) {
266
- aps['mutable-content'] = 1
267
- }
268
-
269
- // Target content id (for live activities)
270
- if (notification.targetContentId) {
271
- aps['target-content-id'] = notification.targetContentId
272
- }
273
-
274
- const payload: Record<string, any> = { aps }
275
-
276
- // Add custom data
277
- if (notification.data) {
278
- Object.assign(payload, notification.data)
279
- }
280
-
281
- return payload
282
- }
283
-
284
- /**
285
- * Send a push notification to a single device
286
- */
287
- async send(notification: APNsNotification): Promise<APNsSendResult> {
288
- try {
289
- const client = await this.getClient()
290
- const token = this.generateToken()
291
- const payload = JSON.stringify(this.buildPayload(notification))
292
-
293
- const headers: http2.OutgoingHttpHeaders = {
294
- ':method': 'POST',
295
- ':path': `/3/device/${notification.deviceToken}`,
296
- 'authorization': `bearer ${token}`,
297
- 'apns-topic': this.config.bundleId,
298
- 'apns-push-type': notification.pushType || 'alert',
299
- 'apns-priority': String(notification.priority || 10),
300
- }
301
-
302
- if (notification.expiration !== undefined) {
303
- headers['apns-expiration'] = String(notification.expiration)
304
- }
305
-
306
- if (notification.collapseId) {
307
- headers['apns-collapse-id'] = notification.collapseId
308
- }
309
-
310
- return new Promise((resolve) => {
311
- const req = client.request(headers)
312
-
313
- let responseData = ''
314
- let statusCode: number = 0
315
- let apnsId: string | undefined
316
-
317
- req.on('response', (headers) => {
318
- statusCode = headers[':status'] as number
319
- apnsId = headers['apns-id'] as string | undefined
320
- })
321
-
322
- req.on('data', (chunk) => {
323
- responseData += chunk.toString()
324
- })
325
-
326
- req.on('end', () => {
327
- if (statusCode === 200) {
328
- resolve({
329
- success: true,
330
- deviceToken: notification.deviceToken,
331
- apnsId,
332
- statusCode,
333
- })
334
- } else {
335
- let error = 'Unknown error'
336
- let reason: string | undefined
337
- let timestamp: number | undefined
338
-
339
- if (responseData) {
340
- try {
341
- const parsed = JSON.parse(responseData)
342
- reason = parsed.reason
343
- timestamp = parsed.timestamp
344
- error = reason || error
345
- } catch {
346
- error = responseData
347
- }
348
- }
349
-
350
- resolve({
351
- success: false,
352
- deviceToken: notification.deviceToken,
353
- apnsId,
354
- statusCode,
355
- error,
356
- reason,
357
- timestamp,
358
- })
359
- }
360
- })
361
-
362
- req.on('error', (err) => {
363
- resolve({
364
- success: false,
365
- deviceToken: notification.deviceToken,
366
- error: err.message,
367
- })
368
- })
369
-
370
- req.write(payload)
371
- req.end()
372
- })
373
- } catch (error: any) {
374
- return {
375
- success: false,
376
- deviceToken: notification.deviceToken,
377
- error: error.message,
378
- }
379
- }
380
- }
381
-
382
- /**
383
- * Send push notifications to multiple devices
384
- */
385
- async sendBatch(
386
- notifications: APNsNotification[],
387
- options?: { concurrency?: number }
388
- ): Promise<APNsBatchResult> {
389
- const concurrency = options?.concurrency || 10
390
- const results: APNsSendResult[] = []
391
-
392
- // Process in batches
393
- for (let i = 0; i < notifications.length; i += concurrency) {
394
- const batch = notifications.slice(i, i + concurrency)
395
- const batchResults = await Promise.all(batch.map(n => this.send(n)))
396
- results.push(...batchResults)
397
- }
398
-
399
- return {
400
- sent: results.filter(r => r.success).length,
401
- failed: results.filter(r => !r.success).length,
402
- results,
403
- }
404
- }
405
-
406
- /**
407
- * Send a simple notification (convenience method)
408
- */
409
- async sendSimple(
410
- deviceToken: string,
411
- title: string,
412
- body: string,
413
- data?: Record<string, any>
414
- ): Promise<APNsSendResult> {
415
- return this.send({
416
- deviceToken,
417
- title,
418
- body,
419
- data,
420
- })
421
- }
422
-
423
- /**
424
- * Send a silent/background notification
425
- */
426
- async sendSilent(
427
- deviceToken: string,
428
- data?: Record<string, any>
429
- ): Promise<APNsSendResult> {
430
- return this.send({
431
- deviceToken,
432
- contentAvailable: true,
433
- pushType: 'background',
434
- priority: 5,
435
- data,
436
- })
437
- }
438
-
439
- /**
440
- * Close the HTTP/2 connection
441
- */
442
- close(): void {
443
- if (this.client) {
444
- this.client.close()
445
- this.client = null
446
- }
447
- this.token = null
448
- this.tokenGeneratedAt = 0
449
- }
450
- }
451
-
452
- export default APNsClient