@codyswann/lisa 2.111.0 → 2.112.0
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.
- package/package.json +1 -1
- package/plugins/lisa/.claude-plugin/plugin.json +1 -1
- package/plugins/lisa/.codex-plugin/plugin.json +1 -1
- package/plugins/lisa-cdk/.claude-plugin/plugin.json +1 -1
- package/plugins/lisa-cdk/.codex-plugin/plugin.json +1 -1
- package/plugins/lisa-expo/.claude-plugin/plugin.json +1 -1
- package/plugins/lisa-expo/.codex-plugin/plugin.json +1 -1
- package/plugins/lisa-expo/.mcp.json +3 -3
- package/plugins/lisa-expo/THIRD-PARTY-NOTICES.md +57 -0
- package/plugins/lisa-expo/skills/add-app-clip/SKILL.md +280 -0
- package/plugins/lisa-expo/skills/add-app-clip/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/add-app-clip/references/native-module.md +96 -0
- package/plugins/lisa-expo/skills/building-native-ui/SKILL.md +321 -0
- package/plugins/lisa-expo/skills/building-native-ui/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/animations.md +220 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/controls.md +272 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/form-sheet.md +253 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/gradients.md +106 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/icons.md +213 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/media.md +198 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/route-structure.md +229 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/search.md +248 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/storage.md +121 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/tabs.md +433 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/toolbar-and-headers.md +284 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/visual-effects.md +197 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/webgpu-three.md +605 -0
- package/plugins/lisa-expo/skills/building-native-ui/references/zoom-transitions.md +158 -0
- package/plugins/lisa-expo/skills/eas-update-insights/SKILL.md +228 -0
- package/plugins/lisa-expo/skills/eas-update-insights/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/eas-update-insights/references/channel-insights-schema.md +47 -0
- package/plugins/lisa-expo/skills/eas-update-insights/references/update-insights-schema.md +69 -0
- package/plugins/lisa-expo/skills/expo-api-routes/SKILL.md +369 -0
- package/plugins/lisa-expo/skills/expo-api-routes/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/expo-brownfield/SKILL.md +54 -0
- package/plugins/lisa-expo/skills/expo-brownfield/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/expo-brownfield/references/brownfield-integrated.md +526 -0
- package/plugins/lisa-expo/skills/expo-brownfield/references/brownfield-isolated.md +402 -0
- package/plugins/lisa-expo/skills/expo-brownfield/references/comparison.md +63 -0
- package/plugins/lisa-expo/skills/expo-brownfield/references/troubleshooting.md +88 -0
- package/plugins/lisa-expo/skills/expo-cicd-workflows/SKILL.md +92 -0
- package/plugins/lisa-expo/skills/expo-cicd-workflows/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/expo-cicd-workflows/scripts/fetch.js +113 -0
- package/plugins/lisa-expo/skills/expo-cicd-workflows/scripts/package.json +11 -0
- package/plugins/lisa-expo/skills/expo-cicd-workflows/scripts/validate.js +85 -0
- package/plugins/lisa-expo/skills/expo-deployment/SKILL.md +190 -0
- package/plugins/lisa-expo/skills/expo-deployment/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/expo-deployment/references/app-store-metadata.md +479 -0
- package/plugins/lisa-expo/skills/expo-deployment/references/ios-app-store.md +355 -0
- package/plugins/lisa-expo/skills/expo-deployment/references/play-store.md +246 -0
- package/plugins/lisa-expo/skills/expo-deployment/references/testflight.md +58 -0
- package/plugins/lisa-expo/skills/expo-deployment/references/workflows.md +200 -0
- package/plugins/lisa-expo/skills/expo-dev-client/SKILL.md +164 -0
- package/plugins/lisa-expo/skills/expo-dev-client/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/expo-module/SKILL.md +141 -0
- package/plugins/lisa-expo/skills/expo-module/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/expo-module/references/config-plugin.md +90 -0
- package/plugins/lisa-expo/skills/expo-module/references/create-expo-module.md +206 -0
- package/plugins/lisa-expo/skills/expo-module/references/lifecycle.md +127 -0
- package/plugins/lisa-expo/skills/expo-module/references/module-config.md +48 -0
- package/plugins/lisa-expo/skills/expo-module/references/native-module.md +286 -0
- package/plugins/lisa-expo/skills/expo-module/references/native-view.md +171 -0
- package/plugins/lisa-expo/skills/expo-tailwind-setup/SKILL.md +480 -0
- package/plugins/lisa-expo/skills/expo-tailwind-setup/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/expo-ui-jetpack-compose/SKILL.md +40 -0
- package/plugins/lisa-expo/skills/expo-ui-jetpack-compose/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/expo-ui-swift-ui/SKILL.md +39 -0
- package/plugins/lisa-expo/skills/expo-ui-swift-ui/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/native-data-fetching/SKILL.md +507 -0
- package/plugins/lisa-expo/skills/native-data-fetching/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/native-data-fetching/references/expo-router-loaders.md +344 -0
- package/plugins/lisa-expo/skills/upgrading-expo/SKILL.md +134 -0
- package/plugins/lisa-expo/skills/upgrading-expo/agents/openai.yaml +4 -0
- package/plugins/lisa-expo/skills/upgrading-expo/references/expo-av-to-audio.md +132 -0
- package/plugins/lisa-expo/skills/upgrading-expo/references/expo-av-to-video.md +160 -0
- package/plugins/lisa-expo/skills/upgrading-expo/references/native-tabs.md +124 -0
- package/plugins/lisa-expo/skills/upgrading-expo/references/new-architecture.md +79 -0
- package/plugins/lisa-expo/skills/upgrading-expo/references/react-19.md +79 -0
- package/plugins/lisa-expo/skills/upgrading-expo/references/react-compiler.md +59 -0
- package/plugins/lisa-expo/skills/upgrading-expo/references/react-navigation-to-expo-router.md +61 -0
- package/plugins/lisa-expo/skills/use-dom/SKILL.md +417 -0
- package/plugins/lisa-expo/skills/use-dom/agents/openai.yaml +4 -0
- package/plugins/lisa-harper-fabric/.claude-plugin/plugin.json +1 -1
- package/plugins/lisa-harper-fabric/.codex-plugin/plugin.json +1 -1
- package/plugins/lisa-nestjs/.claude-plugin/plugin.json +1 -1
- package/plugins/lisa-nestjs/.codex-plugin/plugin.json +1 -1
- package/plugins/lisa-openclaw/.claude-plugin/plugin.json +1 -1
- package/plugins/lisa-openclaw/.codex-plugin/plugin.json +1 -1
- package/plugins/lisa-rails/.claude-plugin/plugin.json +1 -1
- package/plugins/lisa-rails/.codex-plugin/plugin.json +1 -1
- package/plugins/lisa-typescript/.claude-plugin/plugin.json +1 -1
- package/plugins/lisa-typescript/.codex-plugin/plugin.json +1 -1
- package/plugins/lisa-wiki/.claude-plugin/plugin.json +1 -1
- package/plugins/lisa-wiki/.codex-plugin/plugin.json +1 -1
- package/plugins/src/expo/.mcp.json +3 -3
- package/plugins/src/expo/THIRD-PARTY-NOTICES.md +57 -0
- package/plugins/src/expo/skills/add-app-clip/SKILL.md +280 -0
- package/plugins/src/expo/skills/add-app-clip/references/native-module.md +96 -0
- package/plugins/src/expo/skills/building-native-ui/SKILL.md +321 -0
- package/plugins/src/expo/skills/building-native-ui/references/animations.md +220 -0
- package/plugins/src/expo/skills/building-native-ui/references/controls.md +272 -0
- package/plugins/src/expo/skills/building-native-ui/references/form-sheet.md +253 -0
- package/plugins/src/expo/skills/building-native-ui/references/gradients.md +106 -0
- package/plugins/src/expo/skills/building-native-ui/references/icons.md +213 -0
- package/plugins/src/expo/skills/building-native-ui/references/media.md +198 -0
- package/plugins/src/expo/skills/building-native-ui/references/route-structure.md +229 -0
- package/plugins/src/expo/skills/building-native-ui/references/search.md +248 -0
- package/plugins/src/expo/skills/building-native-ui/references/storage.md +121 -0
- package/plugins/src/expo/skills/building-native-ui/references/tabs.md +433 -0
- package/plugins/src/expo/skills/building-native-ui/references/toolbar-and-headers.md +284 -0
- package/plugins/src/expo/skills/building-native-ui/references/visual-effects.md +197 -0
- package/plugins/src/expo/skills/building-native-ui/references/webgpu-three.md +605 -0
- package/plugins/src/expo/skills/building-native-ui/references/zoom-transitions.md +158 -0
- package/plugins/src/expo/skills/eas-update-insights/SKILL.md +228 -0
- package/plugins/src/expo/skills/eas-update-insights/references/channel-insights-schema.md +47 -0
- package/plugins/src/expo/skills/eas-update-insights/references/update-insights-schema.md +69 -0
- package/plugins/src/expo/skills/expo-api-routes/SKILL.md +369 -0
- package/plugins/src/expo/skills/expo-brownfield/SKILL.md +54 -0
- package/plugins/src/expo/skills/expo-brownfield/references/brownfield-integrated.md +526 -0
- package/plugins/src/expo/skills/expo-brownfield/references/brownfield-isolated.md +402 -0
- package/plugins/src/expo/skills/expo-brownfield/references/comparison.md +63 -0
- package/plugins/src/expo/skills/expo-brownfield/references/troubleshooting.md +88 -0
- package/plugins/src/expo/skills/expo-cicd-workflows/SKILL.md +92 -0
- package/plugins/src/expo/skills/expo-cicd-workflows/scripts/fetch.js +113 -0
- package/plugins/src/expo/skills/expo-cicd-workflows/scripts/package.json +11 -0
- package/plugins/src/expo/skills/expo-cicd-workflows/scripts/validate.js +85 -0
- package/plugins/src/expo/skills/expo-deployment/SKILL.md +190 -0
- package/plugins/src/expo/skills/expo-deployment/references/app-store-metadata.md +479 -0
- package/plugins/src/expo/skills/expo-deployment/references/ios-app-store.md +355 -0
- package/plugins/src/expo/skills/expo-deployment/references/play-store.md +246 -0
- package/plugins/src/expo/skills/expo-deployment/references/testflight.md +58 -0
- package/plugins/src/expo/skills/expo-deployment/references/workflows.md +200 -0
- package/plugins/src/expo/skills/expo-dev-client/SKILL.md +164 -0
- package/plugins/src/expo/skills/expo-module/SKILL.md +141 -0
- package/plugins/src/expo/skills/expo-module/references/config-plugin.md +90 -0
- package/plugins/src/expo/skills/expo-module/references/create-expo-module.md +206 -0
- package/plugins/src/expo/skills/expo-module/references/lifecycle.md +127 -0
- package/plugins/src/expo/skills/expo-module/references/module-config.md +48 -0
- package/plugins/src/expo/skills/expo-module/references/native-module.md +286 -0
- package/plugins/src/expo/skills/expo-module/references/native-view.md +171 -0
- package/plugins/src/expo/skills/expo-tailwind-setup/SKILL.md +480 -0
- package/plugins/src/expo/skills/expo-ui-jetpack-compose/SKILL.md +40 -0
- package/plugins/src/expo/skills/expo-ui-swift-ui/SKILL.md +39 -0
- package/plugins/src/expo/skills/native-data-fetching/SKILL.md +507 -0
- package/plugins/src/expo/skills/native-data-fetching/references/expo-router-loaders.md +344 -0
- package/plugins/src/expo/skills/upgrading-expo/SKILL.md +134 -0
- package/plugins/src/expo/skills/upgrading-expo/references/expo-av-to-audio.md +132 -0
- package/plugins/src/expo/skills/upgrading-expo/references/expo-av-to-video.md +160 -0
- package/plugins/src/expo/skills/upgrading-expo/references/native-tabs.md +124 -0
- package/plugins/src/expo/skills/upgrading-expo/references/new-architecture.md +79 -0
- package/plugins/src/expo/skills/upgrading-expo/references/react-19.md +79 -0
- package/plugins/src/expo/skills/upgrading-expo/references/react-compiler.md +59 -0
- package/plugins/src/expo/skills/upgrading-expo/references/react-navigation-to-expo-router.md +61 -0
- package/plugins/src/expo/skills/use-dom/SKILL.md +417 -0
- package/scripts/generate-codex-plugin-artifacts.mjs +7 -2
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
# Apple Zoom Transitions
|
|
2
|
+
|
|
3
|
+
Fluid zoom transitions for navigating between screens. iOS 18+, Expo SDK 55+, Stack navigator only.
|
|
4
|
+
|
|
5
|
+
```tsx
|
|
6
|
+
import { Link } from "expo-router";
|
|
7
|
+
```
|
|
8
|
+
|
|
9
|
+
## Basic Zoom
|
|
10
|
+
|
|
11
|
+
Use `withAppleZoom` on `Link.Trigger` to zoom the entire trigger element into the destination screen:
|
|
12
|
+
|
|
13
|
+
```tsx
|
|
14
|
+
<Link href="/photo" asChild>
|
|
15
|
+
<Link.Trigger withAppleZoom>
|
|
16
|
+
<Pressable>
|
|
17
|
+
<Image
|
|
18
|
+
source={{ uri: "https://example.com/thumb.jpg" }}
|
|
19
|
+
style={{ width: 120, height: 120, borderRadius: 12 }}
|
|
20
|
+
/>
|
|
21
|
+
</Pressable>
|
|
22
|
+
</Link.Trigger>
|
|
23
|
+
</Link>
|
|
24
|
+
```
|
|
25
|
+
|
|
26
|
+
## Targeted Zoom with `Link.AppleZoom`
|
|
27
|
+
|
|
28
|
+
Wrap only the element that should animate. Siblings outside `Link.AppleZoom` are not part of the transition:
|
|
29
|
+
|
|
30
|
+
```tsx
|
|
31
|
+
<Link href="/photo" asChild>
|
|
32
|
+
<Link.Trigger>
|
|
33
|
+
<Pressable style={{ alignItems: "center" }}>
|
|
34
|
+
<Link.AppleZoom>
|
|
35
|
+
<Image
|
|
36
|
+
source={{ uri: "https://example.com/thumb.jpg" }}
|
|
37
|
+
style={{ width: 200, aspectRatio: 4 / 3 }}
|
|
38
|
+
/>
|
|
39
|
+
</Link.AppleZoom>
|
|
40
|
+
<Text>Caption text (not zoomed)</Text>
|
|
41
|
+
</Pressable>
|
|
42
|
+
</Link.Trigger>
|
|
43
|
+
</Link>
|
|
44
|
+
```
|
|
45
|
+
|
|
46
|
+
`Link.AppleZoom` accepts only a single child element.
|
|
47
|
+
|
|
48
|
+
## Destination Target
|
|
49
|
+
|
|
50
|
+
Use `Link.AppleZoomTarget` on the destination screen to align the zoom animation to a specific element:
|
|
51
|
+
|
|
52
|
+
```tsx
|
|
53
|
+
// Destination screen (e.g., app/photo.tsx)
|
|
54
|
+
import { Link } from "expo-router";
|
|
55
|
+
|
|
56
|
+
export default function PhotoScreen() {
|
|
57
|
+
return (
|
|
58
|
+
<View style={{ flex: 1 }}>
|
|
59
|
+
<Link.AppleZoomTarget>
|
|
60
|
+
<Image
|
|
61
|
+
source={{ uri: "https://example.com/full.jpg" }}
|
|
62
|
+
style={{ width: "100%", aspectRatio: 4 / 3 }}
|
|
63
|
+
/>
|
|
64
|
+
</Link.AppleZoomTarget>
|
|
65
|
+
<Text>Photo details below</Text>
|
|
66
|
+
</View>
|
|
67
|
+
);
|
|
68
|
+
}
|
|
69
|
+
```
|
|
70
|
+
|
|
71
|
+
Without a target, the zoom animates to fill the entire destination screen.
|
|
72
|
+
|
|
73
|
+
## Custom Alignment Rectangle
|
|
74
|
+
|
|
75
|
+
For manual control over where the zoom lands on the destination, use `alignmentRect` instead of `Link.AppleZoomTarget`:
|
|
76
|
+
|
|
77
|
+
```tsx
|
|
78
|
+
<Link.AppleZoom alignmentRect={{ x: 0, y: 0, width: 200, height: 300 }}>
|
|
79
|
+
<Image source={{ uri: "https://example.com/thumb.jpg" }} />
|
|
80
|
+
</Link.AppleZoom>
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
Coordinates are in the destination screen's coordinate space. Prefer `Link.AppleZoomTarget` when possible — use `alignmentRect` only when the target element isn't available as a React component.
|
|
84
|
+
|
|
85
|
+
## Controlling Dismissal
|
|
86
|
+
|
|
87
|
+
Zoom screens support interactive dismissal gestures by default (pinch, swipe down when scrolled to top, swipe from leading edge). Use `usePreventZoomTransitionDismissal` on the destination screen to control this.
|
|
88
|
+
|
|
89
|
+
### Disable all dismissal gestures
|
|
90
|
+
|
|
91
|
+
```tsx
|
|
92
|
+
import { usePreventZoomTransitionDismissal } from "expo-router";
|
|
93
|
+
|
|
94
|
+
export default function PhotoScreen() {
|
|
95
|
+
usePreventZoomTransitionDismissal();
|
|
96
|
+
return <Image source={{ uri: "https://example.com/full.jpg" }} />;
|
|
97
|
+
}
|
|
98
|
+
```
|
|
99
|
+
|
|
100
|
+
### Restrict dismissal to a specific area
|
|
101
|
+
|
|
102
|
+
Use `unstable_dismissalBoundsRect` to prevent conflicts with scrollable content:
|
|
103
|
+
|
|
104
|
+
```tsx
|
|
105
|
+
usePreventZoomTransitionDismissal({
|
|
106
|
+
unstable_dismissalBoundsRect: {
|
|
107
|
+
minX: 0,
|
|
108
|
+
minY: 0,
|
|
109
|
+
maxX: 300,
|
|
110
|
+
maxY: 300,
|
|
111
|
+
},
|
|
112
|
+
});
|
|
113
|
+
```
|
|
114
|
+
|
|
115
|
+
This is useful when the destination contains a zoomable scroll view — the system gives that scroll view precedence over the dismiss gesture.
|
|
116
|
+
|
|
117
|
+
## Combining with Link.Preview
|
|
118
|
+
|
|
119
|
+
Zoom transitions work alongside long-press previews:
|
|
120
|
+
|
|
121
|
+
```tsx
|
|
122
|
+
<Link href="/photo" asChild>
|
|
123
|
+
<Link.Trigger withAppleZoom>
|
|
124
|
+
<Pressable>
|
|
125
|
+
<Image
|
|
126
|
+
source={{ uri: "https://example.com/thumb.jpg" }}
|
|
127
|
+
style={{ width: 120, height: 120 }}
|
|
128
|
+
/>
|
|
129
|
+
</Pressable>
|
|
130
|
+
</Link.Trigger>
|
|
131
|
+
<Link.Preview />
|
|
132
|
+
</Link>
|
|
133
|
+
```
|
|
134
|
+
|
|
135
|
+
## Best Practices
|
|
136
|
+
|
|
137
|
+
**Good use cases:**
|
|
138
|
+
- Thumbnail → full image (gallery, profile photos)
|
|
139
|
+
- Card → detail screen with similar visual content
|
|
140
|
+
- Source and destination with similar aspect ratios
|
|
141
|
+
|
|
142
|
+
**Avoid:**
|
|
143
|
+
- Skinny full-width list rows as zoom sources — the transition looks unnatural
|
|
144
|
+
- Mismatched aspect ratios between source and destination without `alignmentRect`
|
|
145
|
+
- Using zoom with sheets or popovers — only works in Stack navigator
|
|
146
|
+
- Hiding the navigation bar — known issues with header visibility during transitions
|
|
147
|
+
|
|
148
|
+
**Tips:**
|
|
149
|
+
- Always provide a close or back button — dismissal gestures are not discoverable
|
|
150
|
+
- If the destination has a zoomable scroll view, use `unstable_dismissalBoundsRect` to avoid gesture conflicts
|
|
151
|
+
- Source view doesn't need to match the tap target — only the `Link.AppleZoom` wrapped element animates
|
|
152
|
+
- When source is unavailable (e.g., scrolled off screen), the transition zooms from the center of the screen
|
|
153
|
+
|
|
154
|
+
## References
|
|
155
|
+
|
|
156
|
+
- Expo Router Zoom Transitions: https://docs.expo.dev/router/advanced/zoom-transition/
|
|
157
|
+
- Link.AppleZoom API: https://docs.expo.dev/versions/v55.0.0/sdk/router/#linkapplezoom
|
|
158
|
+
- Apple UIKit Fluid Transitions: https://developer.apple.com/documentation/uikit/enhancing-your-app-with-fluid-transitions
|
|
@@ -0,0 +1,228 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: eas-update-insights
|
|
3
|
+
description: "Check the health of published EAS Updates: crash rates, install/launch counts, unique users, payload size, and the split between embedded and OTA users per channel. Use when the user asks how an update is performing, whether a rollout is healthy, how many users are on the embedded build vs OTA, or wants to gate CI on update health."
|
|
4
|
+
version: 1.0.0
|
|
5
|
+
license: MIT
|
|
6
|
+
allowed-tools: "Bash(eas *)"
|
|
7
|
+
---
|
|
8
|
+
|
|
9
|
+
# EAS Update Insights
|
|
10
|
+
|
|
11
|
+
Query the health of published EAS Updates directly from the CLI: launches, failed launches, crash rates, unique users, payload size, the embedded-vs-OTA user split per channel, and the most popular updates per runtime version. The data is the same data that powers the update and channel detail pages on expo.dev; these commands expose it in the terminal in human and JSON form.
|
|
12
|
+
|
|
13
|
+
## When to use this skill
|
|
14
|
+
|
|
15
|
+
Use this when the user wants to assess the health or adoption of a published EAS Update: crash rates, install counts, unique users, bundle size, or the split between embedded and OTA users on a channel.
|
|
16
|
+
|
|
17
|
+
Example prompts:
|
|
18
|
+
|
|
19
|
+
- "How is the latest update doing?"
|
|
20
|
+
- "Is the latest update healthy?"
|
|
21
|
+
- "Is the new release crashing more than the last one?"
|
|
22
|
+
- "How many users are on the latest update vs the embedded build?"
|
|
23
|
+
- "Which update is most popular on production right now?"
|
|
24
|
+
- "How big is our update bundle?"
|
|
25
|
+
|
|
26
|
+
Also fits: post-publish rollout monitoring and regression detection.
|
|
27
|
+
|
|
28
|
+
Don't use when the user needs per-user crash detail or device-level reporting; this skill only exposes aggregate EAS metrics.
|
|
29
|
+
|
|
30
|
+
## Prerequisites
|
|
31
|
+
|
|
32
|
+
- `eas-cli` installed (`npm install -g eas-cli`).
|
|
33
|
+
- Logged in: `eas login`.
|
|
34
|
+
- For `channel:insights`: run from an Expo project directory (the command resolves the project ID from `app.json`). `update:insights` only needs a login.
|
|
35
|
+
|
|
36
|
+
## Commands at a glance
|
|
37
|
+
|
|
38
|
+
| Command | Purpose |
|
|
39
|
+
|---|---|
|
|
40
|
+
| `eas update:list` | Discover recent update groups, their `group` IDs, and branch names |
|
|
41
|
+
| `eas update:insights <groupId>` | Per-platform launches, failed launches, crash rate, unique users, payload size, daily breakdown |
|
|
42
|
+
| `eas update:view <groupId> --insights` | Update group details + the same metrics appended |
|
|
43
|
+
| `eas channel:insights --channel <name> --runtime-version <version>` | Embedded/OTA user counts, most popular updates, cumulative metrics for a channel + runtime |
|
|
44
|
+
|
|
45
|
+
All of these support `--json --non-interactive` for programmatic parsing.
|
|
46
|
+
|
|
47
|
+
## Discovering IDs
|
|
48
|
+
|
|
49
|
+
Before querying insights for an update group, you need its `group` ID. Use `eas update:list` with either `--branch <name>` (updates on that branch) or `--all` (updates across all branches). Always pass `--json --non-interactive` when running non-interactively; without a branch/`--all` flag the command will otherwise prompt for a branch selection:
|
|
50
|
+
|
|
51
|
+
```bash
|
|
52
|
+
# Latest group id across all branches
|
|
53
|
+
eas update:list --all --json --non-interactive | jq -r '.currentPage[0].group'
|
|
54
|
+
|
|
55
|
+
# Latest group id on a specific branch
|
|
56
|
+
eas update:list --branch production --json --non-interactive | jq -r '.currentPage[0].group'
|
|
57
|
+
```
|
|
58
|
+
|
|
59
|
+
The JSON response has a `currentPage` array with one entry per update group (both platforms of the same publish are collapsed into one entry):
|
|
60
|
+
|
|
61
|
+
```json
|
|
62
|
+
{
|
|
63
|
+
"currentPage": [
|
|
64
|
+
{
|
|
65
|
+
"branch": "production",
|
|
66
|
+
"message": "\"Fix checkout crash\" (1 week ago by someone)",
|
|
67
|
+
"runtimeVersion": "1.0.6",
|
|
68
|
+
"group": "03d5dfcf-736c-475a-8730-af039c3f4d06",
|
|
69
|
+
"platforms": "android, ios",
|
|
70
|
+
"isRollBackToEmbedded": false
|
|
71
|
+
}
|
|
72
|
+
]
|
|
73
|
+
}
|
|
74
|
+
```
|
|
75
|
+
|
|
76
|
+
Entries also carry `codeSigningKey` and `rolloutPercentage`, but only when those features are in use for the group (undefined values are omitted from the JSON output).
|
|
77
|
+
|
|
78
|
+
When called with `--branch <name>`, the response also includes `name` (the branch name) and `id` (the branch ID) at the top level.
|
|
79
|
+
|
|
80
|
+
## `eas update:insights <groupId>`
|
|
81
|
+
|
|
82
|
+
Shows launches, failed launches, crash rate, unique users, launch asset count, and average payload size for a single update group, broken down **per platform** (iOS, Android), plus a daily breakdown of launches and failures.
|
|
83
|
+
|
|
84
|
+
### Basic use
|
|
85
|
+
|
|
86
|
+
```bash
|
|
87
|
+
eas update:insights 03d5dfcf-736c-475a-8730-af039c3f4d06
|
|
88
|
+
```
|
|
89
|
+
|
|
90
|
+
### Flags
|
|
91
|
+
|
|
92
|
+
| Flag | Description |
|
|
93
|
+
|---|---|
|
|
94
|
+
| `--days <N>` | Look back N days. Default: **7**. Mutually exclusive with `--start`/`--end`. |
|
|
95
|
+
| `--start <iso-date>` / `--end <iso-date>` | Explicit time range, e.g. `--start 2026-04-01 --end 2026-04-15`. |
|
|
96
|
+
| `--platform <ios\|android>` | Filter to a single platform. Omit to see all platforms in the group. |
|
|
97
|
+
| `--json` | Machine-readable output. Implies `--non-interactive`. |
|
|
98
|
+
| `--non-interactive` | Required when scripting. |
|
|
99
|
+
|
|
100
|
+
### JSON output shape
|
|
101
|
+
|
|
102
|
+
Top level: `groupId`, `timespan` (`start`, `end`, `daysBack`), and `platforms[]` with one entry per platform the group was published to. Each platform entry has `updateId`, `totals` (`uniqueUsers`, `installs`, `failedInstalls`, `crashRatePercent`), `payload` (`launchAssetCount`, `averageUpdatePayloadBytes`), and a `daily[]` time series of `{ date, installs, failedInstalls }`.
|
|
103
|
+
|
|
104
|
+
For the complete schema and field reference, see [references/update-insights-schema.md](./references/update-insights-schema.md).
|
|
105
|
+
|
|
106
|
+
Fields that matter for health assessment:
|
|
107
|
+
|
|
108
|
+
- `platforms[].totals.crashRatePercent`, computed as `failedInstalls / (installs + failedInstalls) * 100`. Zero when there are no installs.
|
|
109
|
+
- `platforms[].totals.installs` and `uniqueUsers` give the adoption signal.
|
|
110
|
+
- `platforms[].daily` is a time series, useful for spotting a sudden spike in failures.
|
|
111
|
+
|
|
112
|
+
### Errors
|
|
113
|
+
|
|
114
|
+
- `Could not find any updates with group ID: "<id>"` — group doesn't exist or you lack access.
|
|
115
|
+
- `Update group "<id>" has no ios update (available platforms: android)` — `--platform ios` was used but the group wasn't published for iOS.
|
|
116
|
+
- `EAS Update insights is not supported by this version of eas-cli. Please upgrade ...` — the server deprecated a field the CLI relies on. Run `npm install -g eas-cli@latest`.
|
|
117
|
+
|
|
118
|
+
## `eas update:view <groupId> --insights`
|
|
119
|
+
|
|
120
|
+
Extends the standard `update:view` output with the same per-platform insights, inline.
|
|
121
|
+
|
|
122
|
+
```bash
|
|
123
|
+
# Human-readable
|
|
124
|
+
eas update:view 03d5dfcf-... --insights
|
|
125
|
+
eas update:view 03d5dfcf-... --insights --days 30
|
|
126
|
+
|
|
127
|
+
# JSON: wrapped as { updates: [...], insights: {...} }
|
|
128
|
+
eas update:view 03d5dfcf-... --json --insights
|
|
129
|
+
```
|
|
130
|
+
|
|
131
|
+
Without `--insights`, `update:view` behaves exactly as before — no JSON shape change for existing consumers. The `--days` / `--start` / `--end` flags only apply when `--insights` is set; passing them alone errors.
|
|
132
|
+
|
|
133
|
+
## `eas channel:insights --channel <name> --runtime-version <version>`
|
|
134
|
+
|
|
135
|
+
Shows, per channel, how many users are on the embedded build vs over-the-air updates and which updates are pulling the most traffic. Must be run from an Expo project directory.
|
|
136
|
+
|
|
137
|
+
### Basic use
|
|
138
|
+
|
|
139
|
+
```bash
|
|
140
|
+
eas channel:insights --channel production --runtime-version 1.0.6
|
|
141
|
+
```
|
|
142
|
+
|
|
143
|
+
### Flags
|
|
144
|
+
|
|
145
|
+
| Flag | Description |
|
|
146
|
+
|---|---|
|
|
147
|
+
| `--channel <name>` | **Required.** The channel name (e.g. `production`, `staging`). |
|
|
148
|
+
| `--runtime-version <version>` | **Required.** Match exactly what was published. Check `runtimeVersion` values in `update:list`. |
|
|
149
|
+
| `--days <N>` | Look back N days. Default: **7**. |
|
|
150
|
+
| `--start` / `--end` | Explicit time range, like `update:insights`. |
|
|
151
|
+
| `--json` / `--non-interactive` | Machine-readable output. |
|
|
152
|
+
|
|
153
|
+
### JSON output shape
|
|
154
|
+
|
|
155
|
+
Top level: `channel`, `runtimeVersion`, `timespan`, `embeddedUpdateTotalUniqueUsers`, `otaTotalUniqueUsers`, `mostPopularUpdates[]` (each with `rank`, `groupId`, `message`, `platform`, `totalUniqueUsers`), `cumulativeMetricsAtLastTimestamp[]`, plus chart-shaped `uniqueUsersOverTime` and `cumulativeMetricsOverTime` objects with `labels` and `datasets`.
|
|
156
|
+
|
|
157
|
+
For the complete schema and field reference, see [references/channel-insights-schema.md](./references/channel-insights-schema.md).
|
|
158
|
+
|
|
159
|
+
Fields that matter:
|
|
160
|
+
|
|
161
|
+
- `embeddedUpdateTotalUniqueUsers` is the count of users running the embedded (binary-bundled) build.
|
|
162
|
+
- `mostPopularUpdates[]` is updates ranked by `totalUniqueUsers`. **Caveat**: this is the top-N the server returns; `otaTotalUniqueUsers` is a sum of that list and may undercount total OTA reach if more than top-N updates are active.
|
|
163
|
+
- `uniqueUsersOverTime` and `cumulativeMetricsOverTime` are daily data series for charting.
|
|
164
|
+
|
|
165
|
+
### Errors
|
|
166
|
+
|
|
167
|
+
- `Could not find channel with the name <name>` — typo or wrong account.
|
|
168
|
+
- "No update launches recorded" in the table / empty `mostPopularUpdates` in JSON — no OTA update has been launched for that channel + runtime yet. Usually means the channel is still serving the embedded build only.
|
|
169
|
+
|
|
170
|
+
## Common workflows
|
|
171
|
+
|
|
172
|
+
### Verify the update I just published is healthy
|
|
173
|
+
|
|
174
|
+
```bash
|
|
175
|
+
# 1. Grab the latest publish on production
|
|
176
|
+
GROUP_ID=$(eas update:list --branch production --json --non-interactive \
|
|
177
|
+
| jq -r '.currentPage[0].group')
|
|
178
|
+
|
|
179
|
+
# 2. Give it some adoption time (minutes to hours), then check crash rate
|
|
180
|
+
eas update:insights "$GROUP_ID" --json --non-interactive \
|
|
181
|
+
| jq '.platforms[] | {platform, installs: .totals.installs, crashRate: .totals.crashRatePercent}'
|
|
182
|
+
```
|
|
183
|
+
|
|
184
|
+
Compare the `crashRate` across platforms and against previous releases; sudden spikes or asymmetric behaviour (iOS spiking while Android is flat, or vice versa) is the signal to investigate.
|
|
185
|
+
|
|
186
|
+
### Compare adoption between two channels
|
|
187
|
+
|
|
188
|
+
```bash
|
|
189
|
+
for channel in production staging; do
|
|
190
|
+
echo "--- $channel ---"
|
|
191
|
+
eas channel:insights --channel "$channel" --runtime-version 1.0.6 --json --non-interactive \
|
|
192
|
+
| jq '{
|
|
193
|
+
channel,
|
|
194
|
+
embedded: .embeddedUpdateTotalUniqueUsers,
|
|
195
|
+
ota: .otaTotalUniqueUsers,
|
|
196
|
+
topUpdate: .mostPopularUpdates[0]
|
|
197
|
+
}'
|
|
198
|
+
done
|
|
199
|
+
```
|
|
200
|
+
|
|
201
|
+
### Detect a rollout regression in the last 24 hours
|
|
202
|
+
|
|
203
|
+
```bash
|
|
204
|
+
eas update:insights "$GROUP_ID" --days 1 --json --non-interactive \
|
|
205
|
+
| jq '.platforms[] | select(.totals.crashRatePercent > 1)'
|
|
206
|
+
```
|
|
207
|
+
|
|
208
|
+
### Summarize group metrics for release notes
|
|
209
|
+
|
|
210
|
+
```bash
|
|
211
|
+
eas update:view "$GROUP_ID" --insights --days 30
|
|
212
|
+
```
|
|
213
|
+
|
|
214
|
+
Human-readable group details plus 30 days of launches/failures per platform — suitable for pasting into a changelog or incident review.
|
|
215
|
+
|
|
216
|
+
## Output tips
|
|
217
|
+
|
|
218
|
+
- Pipe JSON through `jq`; payloads are structured for easy filtering.
|
|
219
|
+
- `--json` implies `--non-interactive`, but passing both is explicit and scripting-friendly.
|
|
220
|
+
- Dates in `daily[].date` are UTC ISO timestamps; the human-readable table renders them as `YYYY-MM-DD` (UTC).
|
|
221
|
+
- The CLI table labels say "Launches" / "Crashes" while JSON uses `installs` / `failedInstalls`. Same field, different display name.
|
|
222
|
+
|
|
223
|
+
## Limitations
|
|
224
|
+
|
|
225
|
+
- **Unique users across platforms** may double-count users who run the same publish on both iOS and Android. The same caveat applies to `otaTotalUniqueUsers` in channel insights, which is a sum over `mostPopularUpdates`.
|
|
226
|
+
- **Fresh publishes** may show zeros for a short period while the metrics pipeline catches up.
|
|
227
|
+
- **Installs are downloads, not launches**: the `installs` / "Launches" field counts users who downloaded the manifest and launch asset. A confirmed run only registers on the user's *next* update check (typically up to 24h later, depending on the app's update policy). So metrics lag the real-world state slightly.
|
|
228
|
+
- **Crashes are self-reported**: `failedInstalls` / "Crashes" counts updates that errored during install/launch and were reported on the next update check. Crashes that don't trigger an update request (e.g. process kill before recovery) won't appear.
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
display_name: "Eas Update Insights"
|
|
2
|
+
short_description: "Check the health of published EAS Updates: crash rates, install/launch counts, unique users, payload size, and the split between embedded…"
|
|
3
|
+
default_prompt:
|
|
4
|
+
- "Use $eas-update-insights: Check the health of published EAS Updates: crash rates, install/launch counts, unique users, payload size, and the split between embedded…."
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
# `eas channel:insights` JSON schema
|
|
2
|
+
|
|
3
|
+
Complete JSON output shape returned by `eas channel:insights --channel <name> --runtime-version <version> --json --non-interactive`.
|
|
4
|
+
|
|
5
|
+
```json
|
|
6
|
+
{
|
|
7
|
+
"channel": "production",
|
|
8
|
+
"runtimeVersion": "1.0.6",
|
|
9
|
+
"timespan": { "start": "...", "end": "...", "daysBack": 7 },
|
|
10
|
+
"embeddedUpdateTotalUniqueUsers": 2401,
|
|
11
|
+
"otaTotalUniqueUsers": 8312,
|
|
12
|
+
"mostPopularUpdates": [
|
|
13
|
+
{
|
|
14
|
+
"rank": 1,
|
|
15
|
+
"groupId": "abc123",
|
|
16
|
+
"message": "Fix checkout crash",
|
|
17
|
+
"platform": "ios",
|
|
18
|
+
"totalUniqueUsers": 4210
|
|
19
|
+
}
|
|
20
|
+
],
|
|
21
|
+
"cumulativeMetricsAtLastTimestamp": [
|
|
22
|
+
{ "id": "...", "label": "Embedded update", "data": 12345 },
|
|
23
|
+
{ "id": "...", "label": "Embedded update failed installs", "data": 0 }
|
|
24
|
+
],
|
|
25
|
+
"uniqueUsersOverTime": { "labels": ["..."], "datasets": [ { "id": "...", "label": "...", "data": [100, 200] } ] },
|
|
26
|
+
"cumulativeMetricsOverTime": { "labels": ["..."], "datasets": [ { "id": "...", "label": "...", "data": [10, 20] } ] }
|
|
27
|
+
}
|
|
28
|
+
```
|
|
29
|
+
|
|
30
|
+
## Field reference
|
|
31
|
+
|
|
32
|
+
| Path | Meaning |
|
|
33
|
+
|---|---|
|
|
34
|
+
| `channel` | The channel queried. |
|
|
35
|
+
| `runtimeVersion` | The runtime version filter used. Channel insights are always scoped to a single runtime. |
|
|
36
|
+
| `timespan.start` / `.end` / `.daysBack` | Window bounds (UTC ISO) and size in days. |
|
|
37
|
+
| `embeddedUpdateTotalUniqueUsers` | Distinct users running the embedded (binary-bundled) build in the window. |
|
|
38
|
+
| `otaTotalUniqueUsers` | Sum of `totalUniqueUsers` across `mostPopularUpdates`. May undercount if more than top-N updates are active (see caveat below). |
|
|
39
|
+
| `mostPopularUpdates[]` | Top-N updates ranked by `totalUniqueUsers`. Each entry has `rank`, `groupId`, `message`, `platform`, `totalUniqueUsers`. |
|
|
40
|
+
| `cumulativeMetricsAtLastTimestamp[]` | Snapshot totals at the end of the window, labelled (e.g., "Embedded update", "Embedded update failed installs"). |
|
|
41
|
+
| `uniqueUsersOverTime` | Chart-shaped object with `labels` (dates) and `datasets` for plotting unique users over time. |
|
|
42
|
+
| `cumulativeMetricsOverTime` | Chart-shaped object for plotting cumulative metrics over time. |
|
|
43
|
+
|
|
44
|
+
## Caveats
|
|
45
|
+
|
|
46
|
+
- `otaTotalUniqueUsers` is a sum of `mostPopularUpdates[].totalUniqueUsers`. If more OTA updates are active than the top-N the server returns, this figure undercounts true OTA reach.
|
|
47
|
+
- A user running the same publish on both iOS and Android may be counted on each platform. Don't treat `uniqueUsers` as cross-platform-deduped.
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
# `eas update:insights` JSON schema
|
|
2
|
+
|
|
3
|
+
Complete JSON output shape returned by `eas update:insights <groupId> --json --non-interactive`.
|
|
4
|
+
|
|
5
|
+
```json
|
|
6
|
+
{
|
|
7
|
+
"groupId": "03d5dfcf-736c-475a-8730-af039c3f4d06",
|
|
8
|
+
"timespan": {
|
|
9
|
+
"start": "2026-04-10T00:00:00.000Z",
|
|
10
|
+
"end": "2026-04-17T00:00:00.000Z",
|
|
11
|
+
"daysBack": 7
|
|
12
|
+
},
|
|
13
|
+
"platforms": [
|
|
14
|
+
{
|
|
15
|
+
"platform": "android",
|
|
16
|
+
"updateId": "019d72ca-...",
|
|
17
|
+
"totals": {
|
|
18
|
+
"uniqueUsers": 500,
|
|
19
|
+
"installs": 990,
|
|
20
|
+
"failedInstalls": 10,
|
|
21
|
+
"crashRatePercent": 1.0
|
|
22
|
+
},
|
|
23
|
+
"payload": {
|
|
24
|
+
"launchAssetCount": 4,
|
|
25
|
+
"averageUpdatePayloadBytes": 1115771
|
|
26
|
+
},
|
|
27
|
+
"daily": [
|
|
28
|
+
{ "date": "2026-04-10T00:00:00.000Z", "installs": 182, "failedInstalls": 2 },
|
|
29
|
+
{ "date": "2026-04-11T00:00:00.000Z", "installs": 195, "failedInstalls": 1 }
|
|
30
|
+
]
|
|
31
|
+
},
|
|
32
|
+
{
|
|
33
|
+
"platform": "ios",
|
|
34
|
+
"updateId": "019d72ca-...",
|
|
35
|
+
"totals": { "uniqueUsers": 100, "installs": 1, "failedInstalls": 0, "crashRatePercent": 0 },
|
|
36
|
+
"payload": { "launchAssetCount": 4, "averageUpdatePayloadBytes": 1115771 },
|
|
37
|
+
"daily": [ { "date": "2026-04-10T00:00:00.000Z", "installs": 1, "failedInstalls": 0 } ]
|
|
38
|
+
}
|
|
39
|
+
]
|
|
40
|
+
}
|
|
41
|
+
```
|
|
42
|
+
|
|
43
|
+
## Field reference
|
|
44
|
+
|
|
45
|
+
| Path | Meaning |
|
|
46
|
+
|---|---|
|
|
47
|
+
| `groupId` | The update group queried. |
|
|
48
|
+
| `timespan.start` / `.end` | UTC ISO timestamps bounding the window. |
|
|
49
|
+
| `timespan.daysBack` | Convenience field: size of the window in days. |
|
|
50
|
+
| `platforms[]` | One entry per platform the group was published to (`ios`, `android`). |
|
|
51
|
+
| `platforms[].updateId` | Platform-specific update ID (distinct from the group ID). |
|
|
52
|
+
| `platforms[].totals.uniqueUsers` | Distinct users who ran this update in the window. |
|
|
53
|
+
| `platforms[].totals.installs` | Launches / successful installs in the window. |
|
|
54
|
+
| `platforms[].totals.failedInstalls` | Crashes / failed installs in the window. |
|
|
55
|
+
| `platforms[].totals.crashRatePercent` | `failedInstalls / (installs + failedInstalls) * 100`. Zero when no installs. |
|
|
56
|
+
| `platforms[].payload.launchAssetCount` | Number of assets the manifest references. |
|
|
57
|
+
| `platforms[].payload.averageUpdatePayloadBytes` | Mean bundle size for the window. |
|
|
58
|
+
| `platforms[].daily[]` | Per-day time series of installs and failed installs. |
|
|
59
|
+
|
|
60
|
+
## `eas update:view <groupId> --insights --json`
|
|
61
|
+
|
|
62
|
+
The `update:view --insights --json` command wraps the same insights payload:
|
|
63
|
+
|
|
64
|
+
```json
|
|
65
|
+
{
|
|
66
|
+
"updates": [ /* standard update:view entries */ ],
|
|
67
|
+
"insights": { /* same shape as eas update:insights above */ }
|
|
68
|
+
}
|
|
69
|
+
```
|