name: prod-bundles description: Use when the user asks to build production bundles, check bundle sizes, audit tree-shaking, or verify mobile/desktop code separation. Covers both the desktop webpack prod build and the iOS/Android Metro bundle.
Build production bundles for both platforms and analyze them for correct tree-shaking.
Build Commands
Desktop (webpack):
# From shared/
yarn desktop:build:prod
Output lands in shared/desktop/dist/. Prod bundles have no .dev or .profile suffix — filter with:
ls shared/desktop/dist/*.bundle.js | grep -v '\.dev\.' | grep -v '\.profile\.'
iOS (Metro):
# From shared/
yarn ios:jsbundle
Output: shared/ios/dist/main.jsbundle
Android (Metro):
# From shared/
yarn android:jsbundle
Output: shared/android/dist/main.jsbundle
Tree-Shaking Audit
Desktop — check mobile-only modules are absent:
DIST=shared/desktop/dist
PROD=$(ls "$DIST"/*.bundle.js | grep -v '\.dev\.' | grep -v '\.profile\.')
for mod in expo-audio expo-location expo-video react-native-kb @gorhom/bottom-sheet lottie-react-native react-native-safe-area-context; do
hits=$(echo "$PROD" | xargs grep -l "$mod" 2>/dev/null | wc -l | tr -d ' ')
echo "$mod: $hits files"
done
iOS bundle — check bare isMobile/isElectron are inlined as literals (Babel plugin):
# Should report 0 occurrences — bare globals replaced with true/false at transform time
python3 -c "
import re
bundle = open('shared/ios/dist/main.jsbundle').read()
for name in ['isMobile', 'isElectron', 'isAndroid', 'isIOS']:
real = [m for m in re.finditer(r'(?<![.\w{,])' + name + r'(?![:\w])', bundle)]
print(f'{name} not as property/key: {len(real)} occurrences')
"
Key Facts
- Webpack (desktop):
DefinePluginreplaces bare globals (isMobile,isElectron, etc.) with literals. Terser DCEs dead branches. Works cross-module. - Metro (iOS/Android): The
makePlatformPluginBabel plugin inbabel.config.jsinlines the same globals at transform time, enabling Metro'sconstant-folding-pluginto DCE dead branches. - Native-only module aliasing (desktop): packages in
shared/native-only-modules.jsare aliased toshared/null-module.jsby webpack. Changes to that file require clearing the webpack cache:rm -rf shared/node_modules/.cache/webpack. - Webpack cache invalidation:
shared/desktop/webpack.config.mtslistsbuildDependencies— if you add a new file that affects the build, add it there so cache auto-invalidates.