react-native-acoustic-connect-beta 18.0.34 → 18.0.36

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.
@@ -0,0 +1,487 @@
1
+ // Copyright (C) 2026 Acoustic, L.P. All rights reserved.
2
+ //
3
+ // NOTICE: This file contains material that is confidential and proprietary to
4
+ // Acoustic, L.P. and/or other developers. No license is granted under any
5
+ // intellectual or industrial property rights of Acoustic, L.P. except as may
6
+ // be provided in an agreement with Acoustic, L.P. Any unauthorized copying or
7
+ // distribution of content from this file is prohibited.
8
+
9
+ import {
10
+ withXcodeProject,
11
+ withDangerousMod,
12
+ type ConfigPlugin,
13
+ } from '@expo/config-plugins'
14
+ import type { ExpoConfig } from '@expo/config-types'
15
+ import * as fs from 'fs'
16
+ import * as path from 'path'
17
+ import {
18
+ resolveAppGroupIdentifier,
19
+ substituteSwiftTemplate,
20
+ getHostBuildSettingForConfig,
21
+ buildConnectPodTargetBlock,
22
+ withNSEEntitlements,
23
+ type ConnectPluginProps,
24
+ } from './withConnectNSE'
25
+
26
+ // XcodeProject is declared in the `xcode` package which ships no TypeScript
27
+ // types of its own. We derive the type from the withXcodeProject callback
28
+ // so the compiler can still check our usage without requiring @types/xcode.
29
+ type XcodeProject = Parameters<
30
+ Parameters<typeof withXcodeProject>[1]
31
+ >[0]['modResults']
32
+
33
+ // ─── Target / config constants ─────────────────────────────────────────────
34
+
35
+ const NSE_TARGET_NAME = 'ConnectNSE'
36
+ const NCE_TARGET_NAME = 'ConnectNCE'
37
+
38
+ // Configs belonging to either extension target must be skipped when mirroring
39
+ // host build settings. The NCE mod runs AFTER the NSE mod in the composition,
40
+ // so by the time this executes the pbxproj already carries ConnectNSE configs;
41
+ // without skipping both, the NCE could mirror the NSE target's settings rather
42
+ // than the host app's.
43
+ const EXTENSION_TARGET_NAMES = [NSE_TARGET_NAME, NCE_TARGET_NAME]
44
+
45
+ // ─── Plist helpers ────────────────────────────────────────────────────────────
46
+
47
+ /**
48
+ * Generates the Info.plist content for the ConnectNCE target.
49
+ *
50
+ * The NSExtensionAttributes mirror the canonical bare-workflow reference
51
+ * (Examples/bare-workflow/ios/ConnectNCE/Info.plist), which itself mirrors the
52
+ * iOS-SDK NCE 1:1:
53
+ * - UNNotificationExtensionCategory: the category identifiers the Connect
54
+ * APNs payload routes rich-media notifications through. Must match the
55
+ * category the SDK/Collector sets — keep in sync with the iOS-SDK NCE.
56
+ * - UNNotificationExtensionDefaultContentHidden=false: keep the default
57
+ * system body alongside the custom content view.
58
+ * - UNNotificationExtensionInitialContentSizeRatio=1.0: start full-height;
59
+ * the SDK content controller resizes to fit the rendered media.
60
+ *
61
+ * RCTNewArchEnabled is intentionally omitted: the NCE links the Connect SDK
62
+ * only — no React Native runtime is present in the extension process. (The
63
+ * bare-workflow reference carries it as an Xcode-template leftover; the
64
+ * plugin-generated target drops it, consistent with the NSE template.)
65
+ */
66
+ export function buildNCEInfoPlist(): string {
67
+ return `<?xml version="1.0" encoding="UTF-8"?>
68
+ <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
69
+ <plist version="1.0">
70
+ <dict>
71
+ \t<key>CFBundleDevelopmentRegion</key>
72
+ \t<string>$(DEVELOPMENT_LANGUAGE)</string>
73
+ \t<key>CFBundleDisplayName</key>
74
+ \t<string>ConnectNCE</string>
75
+ \t<key>CFBundleExecutable</key>
76
+ \t<string>$(EXECUTABLE_NAME)</string>
77
+ \t<key>CFBundleIdentifier</key>
78
+ \t<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>
79
+ \t<key>CFBundleInfoDictionaryVersion</key>
80
+ \t<string>6.0</string>
81
+ \t<key>CFBundleName</key>
82
+ \t<string>$(PRODUCT_NAME)</string>
83
+ \t<key>CFBundlePackageType</key>
84
+ \t<string>$(PRODUCT_BUNDLE_PACKAGE_TYPE)</string>
85
+ \t<key>CFBundleShortVersionString</key>
86
+ \t<string>$(MARKETING_VERSION)</string>
87
+ \t<key>CFBundleVersion</key>
88
+ \t<string>$(CURRENT_PROJECT_VERSION)</string>
89
+ \t<key>NSExtension</key>
90
+ \t<dict>
91
+ \t\t<key>NSExtensionAttributes</key>
92
+ \t\t<dict>
93
+ \t\t\t<key>UNNotificationExtensionCategory</key>
94
+ \t\t\t<array>
95
+ \t\t\t\t<string>ACOUSTIC_RICH_NOTIFICATION</string>
96
+ \t\t\t\t<string>ACTIONABLE_NOTIFICATION</string>
97
+ \t\t\t</array>
98
+ \t\t\t<key>UNNotificationExtensionDefaultContentHidden</key>
99
+ \t\t\t<false/>
100
+ \t\t\t<key>UNNotificationExtensionInitialContentSizeRatio</key>
101
+ \t\t\t<real>1.0</real>
102
+ \t\t</dict>
103
+ \t\t<key>NSExtensionPointIdentifier</key>
104
+ \t\t<string>com.apple.usernotifications.content-extension</string>
105
+ \t\t<key>NSExtensionPrincipalClass</key>
106
+ \t\t<string>$(PRODUCT_MODULE_NAME).NotificationViewController</string>
107
+ \t</dict>
108
+ </dict>
109
+ </plist>
110
+ `
111
+ }
112
+
113
+ /** Generates the entitlements plist content for the ConnectNCE target. */
114
+ export function buildNCEEntitlements(appGroupIdentifier: string): string {
115
+ return `<?xml version="1.0" encoding="UTF-8"?>
116
+ <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
117
+ <plist version="1.0">
118
+ <dict>
119
+ \t<key>com.apple.security.application-groups</key>
120
+ \t<array>
121
+ \t\t<string>${appGroupIdentifier}</string>
122
+ \t</array>
123
+ </dict>
124
+ </plist>
125
+ `
126
+ }
127
+
128
+ // ─── Podfile injection ────────────────────────────────────────────────────────
129
+
130
+ const PODFILE_MARKER =
131
+ '# @generated ConnectNCE target (react-native-acoustic-connect-beta)'
132
+
133
+ /**
134
+ * Returns the Ruby snippet to inject into the Expo-generated Podfile for the
135
+ * ConnectNCE target. Thin wrapper over the shared
136
+ * {@link buildConnectPodTargetBlock} (single source of truth for the
137
+ * AcousticConnect pod-resolution helper), parameterised with the NCE marker,
138
+ * helper name, and target name.
139
+ */
140
+ export function buildPodfileBlock(): string {
141
+ return buildConnectPodTargetBlock(
142
+ PODFILE_MARKER,
143
+ 'acoustic_connect_pod_nce',
144
+ NCE_TARGET_NAME
145
+ )
146
+ }
147
+
148
+ /**
149
+ * Injects the ConnectNCE Podfile block into `podfileContent` if not already
150
+ * present (idempotent, guarded by PODFILE_MARKER).
151
+ */
152
+ export function injectPodfileBlock(podfileContent: string): string {
153
+ if (podfileContent.includes(PODFILE_MARKER)) {
154
+ return podfileContent
155
+ }
156
+ return podfileContent + buildPodfileBlock()
157
+ }
158
+
159
+ // ─── Xcode project mod ───────────────────────────────────────────────────────
160
+
161
+ /**
162
+ * Adds the ConnectNCE Xcode target (app_extension) to the project.
163
+ *
164
+ * Sibling to withNSEXcodeProject; the only differences are the target name,
165
+ * the Swift source file (NotificationViewController.swift), the NCE Info.plist
166
+ * (content-extension NSExtension keys), and skipping BOTH extension targets'
167
+ * configs when mirroring host build settings.
168
+ *
169
+ * Idempotent: if a target named ConnectNCE already exists, skips target
170
+ * creation but still (re)writes the source files.
171
+ */
172
+ export function withNCEXcodeProject(
173
+ config: ExpoConfig,
174
+ appGroupIdentifier: string,
175
+ swiftContent: string
176
+ ): ExpoConfig {
177
+ return withXcodeProject(config, (c) => {
178
+ const xcodeProject: XcodeProject = c.modResults
179
+
180
+ // ios/ directory — modRequest.platformProjectRoot is the canonical
181
+ // native project root. (xcodeProject.filepath points INSIDE the
182
+ // .xcodeproj bundle, so deriving from it misplaces the files.)
183
+ const iosDir = c.modRequest.platformProjectRoot
184
+ const nceDir = path.join(iosDir, NCE_TARGET_NAME)
185
+
186
+ // Always write / overwrite the source files (idempotent)
187
+ fs.mkdirSync(nceDir, { recursive: true })
188
+
189
+ fs.writeFileSync(
190
+ path.join(nceDir, 'NotificationViewController.swift'),
191
+ swiftContent,
192
+ 'utf8'
193
+ )
194
+ fs.writeFileSync(
195
+ path.join(nceDir, 'Info.plist'),
196
+ buildNCEInfoPlist(),
197
+ 'utf8'
198
+ )
199
+ fs.writeFileSync(
200
+ path.join(nceDir, `${NCE_TARGET_NAME}.entitlements`),
201
+ buildNCEEntitlements(appGroupIdentifier),
202
+ 'utf8'
203
+ )
204
+
205
+ // Idempotency check — skip if target already registered in the pbxproj.
206
+ // Note: pbxTargetByName misses targets on a REPARSED project because the
207
+ // xcode lib stores written names with literal quotes ('"ConnectNCE"'),
208
+ // so compare quote-stripped names across the native-target section.
209
+ const nativeTargets = xcodeProject.pbxNativeTargetSection() as Record<
210
+ string,
211
+ { name?: string } | string
212
+ >
213
+ const targetExists = Object.values(nativeTargets).some(
214
+ (t) =>
215
+ typeof t === 'object' &&
216
+ typeof t.name === 'string' &&
217
+ t.name.replace(/"/g, '') === NCE_TARGET_NAME
218
+ )
219
+ if (targetExists) {
220
+ return c
221
+ }
222
+
223
+ // Derive the host bundle id — the NCE bundle id must be
224
+ // `<hostBundleId>.ConnectNCE`. Fail fast rather than emit a placeholder
225
+ // (a wrong bundle id breaks App Group pairing and App Store submission).
226
+ const hostBundleId = c.ios?.bundleIdentifier as string | undefined
227
+ if (!hostBundleId) {
228
+ throw new Error(
229
+ `[react-native-acoustic-connect-beta] ios.bundleIdentifier is not set.\n\n` +
230
+ `The ConnectNCE extension bundle id is derived as ` +
231
+ `"<ios.bundleIdentifier>.ConnectNCE", so the host bundle id must be ` +
232
+ `defined in app.json before prebuild:\n` +
233
+ ` { "expo": { "ios": { "bundleIdentifier": "com.example.app" } } }`
234
+ )
235
+ }
236
+
237
+ // Add the native target (app_extension)
238
+ const nceTarget = xcodeProject.addTarget(
239
+ NCE_TARGET_NAME,
240
+ 'app_extension',
241
+ NCE_TARGET_NAME,
242
+ `${hostBundleId}.ConnectNCE`
243
+ )
244
+
245
+ if (!nceTarget) {
246
+ throw new Error(
247
+ `[react-native-acoustic-connect-beta] Failed to add ConnectNCE Xcode target.`
248
+ )
249
+ }
250
+
251
+ // Add the build phase for Swift sources
252
+ xcodeProject.addBuildPhase(
253
+ ['NotificationViewController.swift'],
254
+ 'PBXSourcesBuildPhase',
255
+ 'Sources',
256
+ nceTarget.uuid
257
+ )
258
+
259
+ // Add Resources build phase (empty, but required by Xcode)
260
+ xcodeProject.addBuildPhase(
261
+ [],
262
+ 'PBXResourcesBuildPhase',
263
+ 'Resources',
264
+ nceTarget.uuid
265
+ )
266
+
267
+ // Add Frameworks build phase (empty — CocoaPods populates it)
268
+ xcodeProject.addBuildPhase(
269
+ [],
270
+ 'PBXFrameworksBuildPhase',
271
+ 'Frameworks',
272
+ nceTarget.uuid
273
+ )
274
+
275
+ // Make the host app target depend on ConnectNCE. CocoaPods resolves an
276
+ // extension's host target through PBXTargetDependency
277
+ // (xcodeproj's host_targets_for_embedded_target), NOT through the embed
278
+ // copy phase — without this edge `pod install` fails with "Unable to find
279
+ // host target(s) for ConnectNCE".
280
+ // The xcode lib SILENTLY no-ops addTargetDependency when these sections
281
+ // are absent from the parsed project — create them first. (The NSE mod
282
+ // already creates them when it runs before this; guarding here keeps the
283
+ // NCE mod correct if it ever runs first.)
284
+ const objects = xcodeProject.hash.project.objects
285
+ objects['PBXTargetDependency'] = objects['PBXTargetDependency'] ?? {}
286
+ objects['PBXContainerItemProxy'] = objects['PBXContainerItemProxy'] ?? {}
287
+ const hostTargetUuid = xcodeProject.getFirstTarget().uuid
288
+ xcodeProject.addTargetDependency(hostTargetUuid, [nceTarget.uuid])
289
+
290
+ // Create a PBXGroup for ConnectNCE source files
291
+ const nceGroup = xcodeProject.addPbxGroup(
292
+ [
293
+ 'NotificationViewController.swift',
294
+ 'Info.plist',
295
+ `${NCE_TARGET_NAME}.entitlements`,
296
+ ],
297
+ NCE_TARGET_NAME,
298
+ NCE_TARGET_NAME
299
+ )
300
+
301
+ // Add the group to the main project group
302
+ const mainGroup = xcodeProject.getFirstProject().firstProject.mainGroup
303
+ xcodeProject.addToPbxGroup(nceGroup.uuid, mainGroup)
304
+
305
+ // Set per-configuration build settings — mirror each host configuration
306
+ // individually so Debug gets Debug values and Release gets Release values.
307
+ const buildConfigurations = xcodeProject.pbxXCConfigurationList()
308
+ const nceConfigListUuid = nceTarget.pbxNativeTarget.buildConfigurationList
309
+
310
+ if (nceConfigListUuid) {
311
+ const configList = buildConfigurations[nceConfigListUuid]
312
+ if (
313
+ configList &&
314
+ Array.isArray(
315
+ (configList as { buildConfigurations?: unknown[] })
316
+ .buildConfigurations
317
+ )
318
+ ) {
319
+ const configs = xcodeProject.pbxXCBuildConfigurationSection()
320
+ for (const entry of (
321
+ configList as { buildConfigurations: Array<{ value: string }> }
322
+ ).buildConfigurations) {
323
+ const buildConfig = configs[entry.value]
324
+ if (buildConfig && buildConfig.buildSettings) {
325
+ // Determine which host config name to mirror (Debug → Debug, etc.)
326
+ const configName: string =
327
+ (buildConfig as { name?: string }).name ?? 'Debug'
328
+
329
+ // Mirror per-configuration values; fall back to the other config's
330
+ // value only when the matching one is absent. Skip BOTH extension
331
+ // targets so the NCE never mirrors the NSE target's settings.
332
+ const deploymentTarget = getHostBuildSettingForConfig(
333
+ xcodeProject,
334
+ 'IPHONEOS_DEPLOYMENT_TARGET',
335
+ configName,
336
+ // Matches the AcousticConnect podspec floor (iOS >= 15.1) so the
337
+ // NCE never demands a higher minimum than the host app.
338
+ '15.1',
339
+ EXTENSION_TARGET_NAMES
340
+ )
341
+ const swiftVersion = getHostBuildSettingForConfig(
342
+ xcodeProject,
343
+ 'SWIFT_VERSION',
344
+ configName,
345
+ '5.0',
346
+ EXTENSION_TARGET_NAMES
347
+ )
348
+ const marketingVersion = getHostBuildSettingForConfig(
349
+ xcodeProject,
350
+ 'MARKETING_VERSION',
351
+ configName,
352
+ '1.0',
353
+ EXTENSION_TARGET_NAMES
354
+ )
355
+ const currentProjectVersion = getHostBuildSettingForConfig(
356
+ xcodeProject,
357
+ 'CURRENT_PROJECT_VERSION',
358
+ configName,
359
+ '1',
360
+ EXTENSION_TARGET_NAMES
361
+ )
362
+
363
+ Object.assign(buildConfig.buildSettings, {
364
+ // Prevent Xcode from synthesising a second Info.plist that would
365
+ // shadow the NSExtension dictionary in ours.
366
+ GENERATE_INFOPLIST_FILE: 'NO',
367
+ INFOPLIST_FILE: `${NCE_TARGET_NAME}/Info.plist`,
368
+ // Extension must target the device SDK and restrict to
369
+ // extension-safe APIs (mirrors bare-workflow reference target).
370
+ SDKROOT: 'iphoneos',
371
+ APPLICATION_EXTENSION_API_ONLY: 'YES',
372
+ IPHONEOS_DEPLOYMENT_TARGET: deploymentTarget,
373
+ SWIFT_VERSION: swiftVersion,
374
+ MARKETING_VERSION: marketingVersion,
375
+ CURRENT_PROJECT_VERSION: currentProjectVersion,
376
+ CODE_SIGN_ENTITLEMENTS: `${NCE_TARGET_NAME}/${NCE_TARGET_NAME}.entitlements`,
377
+ PRODUCT_NAME: NCE_TARGET_NAME,
378
+ PRODUCT_BUNDLE_IDENTIFIER: `${hostBundleId}.ConnectNCE`,
379
+ TARGETED_DEVICE_FAMILY: '"1,2"',
380
+ CODE_SIGN_STYLE: 'Automatic',
381
+ })
382
+ }
383
+ }
384
+ }
385
+ }
386
+
387
+ return c
388
+ })
389
+ }
390
+
391
+ // ─── Podfile mod ─────────────────────────────────────────────────────────────
392
+
393
+ /**
394
+ * Injects the ConnectNCE Podfile target block via withDangerousMod.
395
+ * Guarded by a marker comment — re-runs are no-ops.
396
+ *
397
+ * Throws a clear, actionable error when the Podfile does not exist at
398
+ * mod-execution time, instead of silently returning — a silent skip would
399
+ * ship an NCE whose `import Connect` cannot resolve.
400
+ */
401
+ export function withNCEPodfile(config: ExpoConfig): ExpoConfig {
402
+ return withDangerousMod(config, [
403
+ 'ios',
404
+ async (c) => {
405
+ const podfilePath = path.join(c.modRequest.platformProjectRoot, 'Podfile')
406
+ if (!fs.existsSync(podfilePath)) {
407
+ throw new Error(
408
+ `[react-native-acoustic-connect-beta] ios/Podfile not found at ${podfilePath}.\n\n` +
409
+ `This mod runs after prebuild generates the ios/ directory. ` +
410
+ `If you are running this plugin outside of \`expo prebuild\`, ` +
411
+ `ensure the Podfile exists before the dangerous mod phase executes.\n` +
412
+ `Check prebuild ordering: withDangerousMod('ios') runs after ` +
413
+ `withXcodeProject, so the ios/ directory should already be present.`
414
+ )
415
+ }
416
+ const current = fs.readFileSync(podfilePath, 'utf8')
417
+ const updated = injectPodfileBlock(current)
418
+ if (updated !== current) {
419
+ fs.writeFileSync(podfilePath, updated, 'utf8')
420
+ }
421
+ return c
422
+ },
423
+ ])
424
+ }
425
+
426
+ // ─── Composed mod ─────────────────────────────────────────────────────────────
427
+
428
+ /**
429
+ * Expo Config Plugin mod that provisions a Notification Content Extension
430
+ * (NCE) Xcode target named `ConnectNCE`.
431
+ *
432
+ * Applies three mutations — all idempotent:
433
+ * 1. Host app entitlements: merges App Group into
434
+ * `com.apple.security.application-groups` (reuses the NSE host-entitlements
435
+ * mod — the host shares one App Group across app + NSE + NCE; the merge is
436
+ * a no-op when the NSE mod already added it).
437
+ * 2. Xcode project: adds ConnectNCE target + files (skips if already present).
438
+ * 3. Podfile: injects `target 'ConnectNCE'` block (guarded by marker comment).
439
+ *
440
+ * `config._internal.projectRoot` is required (injected by Expo CLI during
441
+ * `expo prebuild`); an actionable error is thrown if absent rather than
442
+ * silently falling back to `process.cwd()`.
443
+ *
444
+ * NSE coupling (intentional, tracked for refactor): this mod deliberately
445
+ * reuses three pieces of withConnectNSE — `withNSEEntitlements` (the host App
446
+ * Group is shared across app + NSE + NCE), `buildConnectPodTargetBlock` (one
447
+ * source for the pod-resolution helper), and `getHostBuildSettingForConfig`
448
+ * (the per-config mirroring logic). The composition always runs withConnectNSE
449
+ * before withConnectNCE, so the shared host entitlement and the
450
+ * PBXTargetDependency/PBXContainerItemProxy sections already exist when this
451
+ * runs. Extracting a shared `withConnectExtension` base for NSE + NCE is
452
+ * tracked as a follow-up refactor (see PR description) rather than done here,
453
+ * to keep the already-landed NSE mod stable.
454
+ */
455
+ export const withConnectNCE: ConfigPlugin<ConnectPluginProps> = (
456
+ config,
457
+ props = {}
458
+ ) => {
459
+ const projectRoot = config._internal?.projectRoot
460
+ if (!projectRoot) {
461
+ throw new Error(
462
+ `[react-native-acoustic-connect-beta] config._internal.projectRoot is not set.\n\n` +
463
+ `This value is injected by Expo CLI during \`expo prebuild\`. ` +
464
+ `If you are calling this plugin outside of Expo CLI (e.g. in a test or ` +
465
+ `custom script), set config._internal = { projectRoot: '/abs/path/to/project' } ` +
466
+ `before invoking the plugin.`
467
+ )
468
+ }
469
+
470
+ const appGroupIdentifier = resolveAppGroupIdentifier(projectRoot, props)
471
+
472
+ const swiftTemplatePath = path.join(
473
+ __dirname,
474
+ '..',
475
+ 'swift',
476
+ 'NotificationViewController.swift'
477
+ )
478
+ const swiftContent = substituteSwiftTemplate(
479
+ swiftTemplatePath,
480
+ appGroupIdentifier
481
+ )
482
+
483
+ let result = withNSEEntitlements(config, appGroupIdentifier)
484
+ result = withNCEXcodeProject(result, appGroupIdentifier, swiftContent)
485
+ result = withNCEPodfile(result)
486
+ return result
487
+ }