quiver-cli 0.1.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/README.md +188 -0
- package/bin/quiver-cli.mjs +2 -0
- package/dist/cli.js +3074 -0
- package/package.json +55 -0
- package/template/.agents/AGENTS.md +25 -0
- package/template/.agents/commands/cp.md +116 -0
- package/template/.agents/commands/next-setup.md +1064 -0
- package/template/.agents/commands/tf-readme.md +38 -0
- package/template/.agents/config.json +60 -0
- package/template/.agents/skills/agent-browser/SKILL.md +55 -0
- package/template/.agents/skills/apps/skybridge/SKILL.md +46 -0
- package/template/.agents/skills/apps/skybridge/references/architecture.md +175 -0
- package/template/.agents/skills/apps/skybridge/references/copy-template.md +24 -0
- package/template/.agents/skills/apps/skybridge/references/csp.md +33 -0
- package/template/.agents/skills/apps/skybridge/references/deploy.md +33 -0
- package/template/.agents/skills/apps/skybridge/references/discover.md +84 -0
- package/template/.agents/skills/apps/skybridge/references/download-file.md +77 -0
- package/template/.agents/skills/apps/skybridge/references/fetch-and-render-data.md +151 -0
- package/template/.agents/skills/apps/skybridge/references/oauth.md +115 -0
- package/template/.agents/skills/apps/skybridge/references/open-external-links.md +71 -0
- package/template/.agents/skills/apps/skybridge/references/prompt-llm.md +20 -0
- package/template/.agents/skills/apps/skybridge/references/publish.md +19 -0
- package/template/.agents/skills/apps/skybridge/references/run-locally.md +51 -0
- package/template/.agents/skills/apps/skybridge/references/state-and-context.md +151 -0
- package/template/.agents/skills/apps/skybridge/references/ui-guidelines.md +205 -0
- package/template/.agents/skills/code/cleanup/SKILL.md +26 -0
- package/template/.agents/skills/code/vercel-react-best-practices/AGENTS.md +3810 -0
- package/template/.agents/skills/code/vercel-react-best-practices/README.md +123 -0
- package/template/.agents/skills/code/vercel-react-best-practices/SKILL.md +149 -0
- package/template/.agents/skills/code/vercel-react-best-practices/metadata.json +15 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/_sections.md +46 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/_template.md +28 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/advanced-effect-event-deps.md +56 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/advanced-event-handler-refs.md +55 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/advanced-init-once.md +42 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/advanced-use-latest.md +39 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/async-api-routes.md +38 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/async-cheap-condition-before-await.md +37 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/async-defer-await.md +82 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/async-dependencies.md +51 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/async-parallel.md +28 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/async-suspense-boundaries.md +99 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-analyzable-paths.md +63 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-barrel-imports.md +60 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-conditional.md +31 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-defer-third-party.md +49 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-dynamic-imports.md +35 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/bundle-preload.md +50 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/client-event-listeners.md +74 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/client-localstorage-schema.md +71 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/client-passive-event-listeners.md +48 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/client-swr-dedup.md +56 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-batch-dom-css.md +107 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-cache-function-results.md +80 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-cache-property-access.md +28 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-cache-storage.md +70 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-combine-iterations.md +32 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-early-exit.md +50 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-flatmap-filter.md +60 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-hoist-regexp.md +45 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-index-maps.md +37 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-length-check-first.md +49 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-min-max-loop.md +82 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-request-idle-callback.md +105 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-set-map-lookups.md +24 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/js-tosorted-immutable.md +57 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-activity.md +26 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-animate-svg-wrapper.md +47 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-conditional-render.md +40 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-content-visibility.md +38 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-hoist-jsx.md +46 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-hydration-no-flicker.md +82 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-hydration-suppress-warning.md +30 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-resource-hints.md +85 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-script-defer-async.md +68 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-svg-precision.md +28 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rendering-usetransition-loading.md +75 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-defer-reads.md +39 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-dependencies.md +45 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-derived-state-no-effect.md +40 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-derived-state.md +29 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-functional-setstate.md +74 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-lazy-state-init.md +58 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-memo-with-default-value.md +38 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-memo.md +44 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-move-effect-to-event.md +45 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-no-inline-components.md +82 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-simple-expression-in-memo.md +35 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-split-combined-hooks.md +64 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-transitions.md +40 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-use-deferred-value.md +59 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/rerender-use-ref-transient-values.md +73 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-after-nonblocking.md +73 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-auth-actions.md +96 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-cache-lru.md +41 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-cache-react.md +76 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-dedup-props.md +65 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-hoist-static-io.md +149 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-no-shared-module-state.md +50 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-parallel-fetching.md +83 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-parallel-nested-fetching.md +34 -0
- package/template/.agents/skills/code/vercel-react-best-practices/rules/server-serialization.md +38 -0
- package/template/.agents/skills/data/prisma-cli/SKILL.md +247 -0
- package/template/.agents/skills/data/prisma-cli/references/db-execute.md +78 -0
- package/template/.agents/skills/data/prisma-cli/references/db-pull.md +185 -0
- package/template/.agents/skills/data/prisma-cli/references/db-push.md +148 -0
- package/template/.agents/skills/data/prisma-cli/references/db-seed.md +188 -0
- package/template/.agents/skills/data/prisma-cli/references/debug.md +46 -0
- package/template/.agents/skills/data/prisma-cli/references/dev.md +157 -0
- package/template/.agents/skills/data/prisma-cli/references/format.md +48 -0
- package/template/.agents/skills/data/prisma-cli/references/generate.md +173 -0
- package/template/.agents/skills/data/prisma-cli/references/init.md +136 -0
- package/template/.agents/skills/data/prisma-cli/references/mcp.md +38 -0
- package/template/.agents/skills/data/prisma-cli/references/migrate-deploy.md +127 -0
- package/template/.agents/skills/data/prisma-cli/references/migrate-dev.md +145 -0
- package/template/.agents/skills/data/prisma-cli/references/migrate-diff.md +89 -0
- package/template/.agents/skills/data/prisma-cli/references/migrate-reset.md +78 -0
- package/template/.agents/skills/data/prisma-cli/references/migrate-resolve.md +57 -0
- package/template/.agents/skills/data/prisma-cli/references/migrate-status.md +65 -0
- package/template/.agents/skills/data/prisma-cli/references/studio.md +137 -0
- package/template/.agents/skills/data/prisma-cli/references/validate.md +53 -0
- package/template/.agents/skills/data/prisma-client-api/SKILL.md +216 -0
- package/template/.agents/skills/data/prisma-client-api/references/client-methods.md +223 -0
- package/template/.agents/skills/data/prisma-client-api/references/constructor.md +208 -0
- package/template/.agents/skills/data/prisma-client-api/references/filters.md +256 -0
- package/template/.agents/skills/data/prisma-client-api/references/model-queries.md +281 -0
- package/template/.agents/skills/data/prisma-client-api/references/query-options.md +276 -0
- package/template/.agents/skills/data/prisma-client-api/references/raw-queries.md +194 -0
- package/template/.agents/skills/data/prisma-client-api/references/relations.md +308 -0
- package/template/.agents/skills/data/prisma-client-api/references/transactions.md +184 -0
- package/template/.agents/skills/design/impeccable/SKILL.md +176 -0
- package/template/.agents/skills/design/impeccable/reference/adapt.md +311 -0
- package/template/.agents/skills/design/impeccable/reference/animate.md +201 -0
- package/template/.agents/skills/design/impeccable/reference/audit.md +133 -0
- package/template/.agents/skills/design/impeccable/reference/bolder.md +113 -0
- package/template/.agents/skills/design/impeccable/reference/brand.md +108 -0
- package/template/.agents/skills/design/impeccable/reference/clarify.md +288 -0
- package/template/.agents/skills/design/impeccable/reference/codex.md +105 -0
- package/template/.agents/skills/design/impeccable/reference/colorize.md +257 -0
- package/template/.agents/skills/design/impeccable/reference/craft.md +123 -0
- package/template/.agents/skills/design/impeccable/reference/critique.md +767 -0
- package/template/.agents/skills/design/impeccable/reference/delight.md +302 -0
- package/template/.agents/skills/design/impeccable/reference/distill.md +111 -0
- package/template/.agents/skills/design/impeccable/reference/document.md +429 -0
- package/template/.agents/skills/design/impeccable/reference/extract.md +69 -0
- package/template/.agents/skills/design/impeccable/reference/harden.md +347 -0
- package/template/.agents/skills/design/impeccable/reference/init.md +172 -0
- package/template/.agents/skills/design/impeccable/reference/interaction-design.md +189 -0
- package/template/.agents/skills/design/impeccable/reference/layout.md +161 -0
- package/template/.agents/skills/design/impeccable/reference/live.md +718 -0
- package/template/.agents/skills/design/impeccable/reference/onboard.md +234 -0
- package/template/.agents/skills/design/impeccable/reference/optimize.md +258 -0
- package/template/.agents/skills/design/impeccable/reference/overdrive.md +130 -0
- package/template/.agents/skills/design/impeccable/reference/polish.md +241 -0
- package/template/.agents/skills/design/impeccable/reference/product.md +60 -0
- package/template/.agents/skills/design/impeccable/reference/quieter.md +99 -0
- package/template/.agents/skills/design/impeccable/reference/shape.md +165 -0
- package/template/.agents/skills/design/impeccable/reference/typeset.md +279 -0
- package/template/.agents/skills/design/impeccable/scripts/cleanup-deprecated.mjs +284 -0
- package/template/.agents/skills/design/impeccable/scripts/command-metadata.json +94 -0
- package/template/.agents/skills/design/impeccable/scripts/context-signals.mjs +225 -0
- package/template/.agents/skills/design/impeccable/scripts/context.mjs +270 -0
- package/template/.agents/skills/design/impeccable/scripts/critique-storage.mjs +242 -0
- package/template/.agents/skills/design/impeccable/scripts/design-parser.mjs +835 -0
- package/template/.agents/skills/design/impeccable/scripts/detect-csp.mjs +198 -0
- package/template/.agents/skills/design/impeccable/scripts/detect.mjs +21 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/browser/injected/index.mjs +1733 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/cli/main.mjs +244 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/detect-antipatterns-browser.js +4551 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/detect-antipatterns.mjs +43 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/engines/browser/detect-url.mjs +252 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/engines/regex/detect-text.mjs +535 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/engines/static-html/css-cascade.mjs +986 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/engines/static-html/detect-html.mjs +208 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/engines/visual/screenshot-contrast.mjs +189 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/findings.mjs +12 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/node/file-system.mjs +198 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/profile/profiler.mjs +166 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/registry/antipatterns.mjs +419 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/rules/checks.mjs +2316 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/shared/color.mjs +124 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/shared/constants.mjs +101 -0
- package/template/.agents/skills/design/impeccable/scripts/detector/shared/page.mjs +7 -0
- package/template/.agents/skills/design/impeccable/scripts/impeccable-paths.mjs +126 -0
- package/template/.agents/skills/design/impeccable/scripts/is-generated.mjs +69 -0
- package/template/.agents/skills/design/impeccable/scripts/live-accept.mjs +812 -0
- package/template/.agents/skills/design/impeccable/scripts/live-browser-session.js +123 -0
- package/template/.agents/skills/design/impeccable/scripts/live-browser.js +10316 -0
- package/template/.agents/skills/design/impeccable/scripts/live-commit-manual-edits.mjs +1241 -0
- package/template/.agents/skills/design/impeccable/scripts/live-complete.mjs +75 -0
- package/template/.agents/skills/design/impeccable/scripts/live-completion.mjs +19 -0
- package/template/.agents/skills/design/impeccable/scripts/live-copy-edit-agent.mjs +683 -0
- package/template/.agents/skills/design/impeccable/scripts/live-discard-manual-edits.mjs +51 -0
- package/template/.agents/skills/design/impeccable/scripts/live-event-validation.mjs +136 -0
- package/template/.agents/skills/design/impeccable/scripts/live-inject.mjs +557 -0
- package/template/.agents/skills/design/impeccable/scripts/live-insert-ui.mjs +458 -0
- package/template/.agents/skills/design/impeccable/scripts/live-insert.mjs +272 -0
- package/template/.agents/skills/design/impeccable/scripts/live-manual-edit-evidence.mjs +363 -0
- package/template/.agents/skills/design/impeccable/scripts/live-manual-edits-buffer.mjs +152 -0
- package/template/.agents/skills/design/impeccable/scripts/live-poll.mjs +379 -0
- package/template/.agents/skills/design/impeccable/scripts/live-resume.mjs +94 -0
- package/template/.agents/skills/design/impeccable/scripts/live-server.mjs +2322 -0
- package/template/.agents/skills/design/impeccable/scripts/live-session-store.mjs +289 -0
- package/template/.agents/skills/design/impeccable/scripts/live-status.mjs +61 -0
- package/template/.agents/skills/design/impeccable/scripts/live-svelte-component.mjs +826 -0
- package/template/.agents/skills/design/impeccable/scripts/live-sveltekit-adapter.mjs +274 -0
- package/template/.agents/skills/design/impeccable/scripts/live-ui-core.mjs +179 -0
- package/template/.agents/skills/design/impeccable/scripts/live-wrap.mjs +894 -0
- package/template/.agents/skills/design/impeccable/scripts/live.mjs +246 -0
- package/template/.agents/skills/design/impeccable/scripts/modern-screenshot.umd.js +14 -0
- package/template/.agents/skills/design/impeccable/scripts/palette.mjs +633 -0
- package/template/.agents/skills/design/impeccable/scripts/pin.mjs +214 -0
- package/template/.agents/skills/design/shadcn/SKILL.md +242 -0
- package/template/.agents/skills/design/shadcn/agents/openai.yml +5 -0
- package/template/.agents/skills/design/shadcn/assets/shadcn-small.png +0 -0
- package/template/.agents/skills/design/shadcn/assets/shadcn.png +0 -0
- package/template/.agents/skills/design/shadcn/cli.md +257 -0
- package/template/.agents/skills/design/shadcn/customization.md +202 -0
- package/template/.agents/skills/design/shadcn/evals/evals.json +47 -0
- package/template/.agents/skills/design/shadcn/mcp.md +94 -0
- package/template/.agents/skills/design/shadcn/rules/base-vs-radix.md +306 -0
- package/template/.agents/skills/design/shadcn/rules/composition.md +195 -0
- package/template/.agents/skills/design/shadcn/rules/forms.md +192 -0
- package/template/.agents/skills/design/shadcn/rules/icons.md +101 -0
- package/template/.agents/skills/design/shadcn/rules/styling.md +162 -0
- package/template/.agents/skills/find-skills/SKILL.md +142 -0
- package/template/.agents/skills/integrations/langfuse/SKILL.md +142 -0
- package/template/.agents/skills/integrations/langfuse/references/cli.md +52 -0
- package/template/.agents/skills/integrations/langfuse/references/error-analysis.md +100 -0
- package/template/.agents/skills/integrations/langfuse/references/instrumentation.md +134 -0
- package/template/.agents/skills/integrations/langfuse/references/judge-calibration.md +288 -0
- package/template/.agents/skills/integrations/langfuse/references/prompt-migration.md +234 -0
- package/template/.agents/skills/integrations/langfuse/references/sdk-upgrade.md +175 -0
- package/template/.agents/skills/integrations/langfuse/references/skill-feedback.md +52 -0
- package/template/.agents/skills/integrations/langfuse/references/user-feedback.md +88 -0
- package/template/.agents/skills/integrations/posthog/SKILL.md +102 -0
- package/template/.agents/skills/integrations/posthog/references/error-tracking-alerts.md +63 -0
- package/template/.agents/skills/integrations/posthog/references/error-tracking-assigning-issues.md +77 -0
- package/template/.agents/skills/integrations/posthog/references/error-tracking-fingerprints.md +57 -0
- package/template/.agents/skills/integrations/posthog/references/error-tracking-monitoring.md +140 -0
- package/template/.agents/skills/integrations/posthog/references/error-tracking-nextjs.md +490 -0
- package/template/.agents/skills/integrations/posthog/references/error-tracking-source-maps.md +45 -0
- package/template/.agents/skills/integrations/posthog/references/feature-flags-best-practices.md +139 -0
- package/template/.agents/skills/integrations/posthog/references/feature-flags-react.md +302 -0
- package/template/.agents/skills/integrations/posthog/references/identify-users.md +202 -0
- package/template/.agents/skills/integrations/posthog/references/integration-example.md +706 -0
- package/template/.agents/skills/integrations/posthog/references/integration-nextjs.md +385 -0
- package/template/.agents/skills/integrations/posthog/references/integration-step-1-begin.md +43 -0
- package/template/.agents/skills/integrations/posthog/references/integration-step-2-edit.md +37 -0
- package/template/.agents/skills/integrations/posthog/references/integration-step-3-revise.md +22 -0
- package/template/.agents/skills/integrations/posthog/references/integration-step-4-conclude.md +38 -0
- package/template/.agents/skills/integrations/posthog/references/llm-analytics-anthropic.md +200 -0
- package/template/.agents/skills/integrations/posthog/references/llm-analytics-basics.md +62 -0
- package/template/.agents/skills/integrations/posthog/references/llm-analytics-costs.md +197 -0
- package/template/.agents/skills/integrations/posthog/references/llm-analytics-manual-capture.md +397 -0
- package/template/.agents/skills/integrations/posthog/references/llm-analytics-traces.md +98 -0
- package/template/.agents/skills/integrations/posthog/references/llm-analytics-vercel-ai.md +120 -0
- package/template/.agents/skills/repo/repo-ci/SKILL.md +265 -0
- package/template/.agents/skills/repo/repo-init-next-js/SKILL.md +129 -0
- package/template/.agents/skills/repo/repo-init-next-js/references/file-contents.md +800 -0
- package/template/.agents/skills/repo/repo-init-next-js/scripts/setup.sh +47 -0
- package/template/.agents/skills/repo/repo-init-node/SKILL.md +196 -0
- package/template/.agents/skills/skill-creator/LICENSE.txt +202 -0
- package/template/.agents/skills/skill-creator/SKILL.md +485 -0
- package/template/.agents/skills/skill-creator/agents/analyzer.md +274 -0
- package/template/.agents/skills/skill-creator/agents/comparator.md +202 -0
- package/template/.agents/skills/skill-creator/agents/grader.md +223 -0
- package/template/.agents/skills/skill-creator/assets/eval_review.html +146 -0
- package/template/.agents/skills/skill-creator/eval-viewer/generate_review.py +471 -0
- package/template/.agents/skills/skill-creator/eval-viewer/viewer.html +1325 -0
- package/template/.agents/skills/skill-creator/references/schemas.md +430 -0
- package/template/.agents/skills/skill-creator/scripts/__init__.py +0 -0
- package/template/.agents/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
- package/template/.agents/skills/skill-creator/scripts/generate_report.py +326 -0
- package/template/.agents/skills/skill-creator/scripts/improve_description.py +247 -0
- package/template/.agents/skills/skill-creator/scripts/package_skill.py +136 -0
- package/template/.agents/skills/skill-creator/scripts/quick_validate.py +103 -0
- package/template/.agents/skills/skill-creator/scripts/run_eval.py +310 -0
- package/template/.agents/skills/skill-creator/scripts/run_loop.py +328 -0
- package/template/.agents/skills/skill-creator/scripts/utils.py +47 -0
- package/template/.agents/upstreams.json +80 -0
|
@@ -0,0 +1,302 @@
|
|
|
1
|
+
# React feature flags installation - Docs
|
|
2
|
+
|
|
3
|
+
1. 1
|
|
4
|
+
|
|
5
|
+
## Install the package
|
|
6
|
+
|
|
7
|
+
Required
|
|
8
|
+
|
|
9
|
+
Install [`posthog-js`](https://github.com/posthog/posthog-js) and `@posthog/react` using your package manager:
|
|
10
|
+
|
|
11
|
+
PostHog AI
|
|
12
|
+
|
|
13
|
+
### npm
|
|
14
|
+
|
|
15
|
+
```bash
|
|
16
|
+
npm install posthog-js @posthog/react
|
|
17
|
+
```
|
|
18
|
+
|
|
19
|
+
### yarn
|
|
20
|
+
|
|
21
|
+
```bash
|
|
22
|
+
yarn add posthog-js @posthog/react
|
|
23
|
+
```
|
|
24
|
+
|
|
25
|
+
### pnpm
|
|
26
|
+
|
|
27
|
+
```bash
|
|
28
|
+
pnpm add posthog-js @posthog/react
|
|
29
|
+
```
|
|
30
|
+
|
|
31
|
+
2. 2
|
|
32
|
+
|
|
33
|
+
## Add environment variables
|
|
34
|
+
|
|
35
|
+
Required
|
|
36
|
+
|
|
37
|
+
Add your PostHog project token and host to your environment variables. For Vite-based React apps, use the `VITE_PUBLIC_` prefix:
|
|
38
|
+
|
|
39
|
+
.env
|
|
40
|
+
|
|
41
|
+
PostHog AI
|
|
42
|
+
|
|
43
|
+
```bash
|
|
44
|
+
VITE_PUBLIC_POSTHOG_PROJECT_TOKEN=<ph_project_token>
|
|
45
|
+
VITE_PUBLIC_POSTHOG_HOST=https://us.i.posthog.com
|
|
46
|
+
```
|
|
47
|
+
|
|
48
|
+
3. 3
|
|
49
|
+
|
|
50
|
+
## Initialize PostHog
|
|
51
|
+
|
|
52
|
+
Required
|
|
53
|
+
|
|
54
|
+
Wrap your app with the `PostHogProvider` component at the root of your application (such as `main.tsx` if you're using Vite):
|
|
55
|
+
|
|
56
|
+
main.tsx
|
|
57
|
+
|
|
58
|
+
PostHog AI
|
|
59
|
+
|
|
60
|
+
```jsx
|
|
61
|
+
import { StrictMode } from 'react'
|
|
62
|
+
import { createRoot } from 'react-dom/client'
|
|
63
|
+
import './index.css'
|
|
64
|
+
import App from './App.jsx'
|
|
65
|
+
import { PostHogProvider } from '@posthog/react'
|
|
66
|
+
const options = {
|
|
67
|
+
api_host: import.meta.env.VITE_PUBLIC_POSTHOG_HOST,
|
|
68
|
+
defaults: '2026-01-30',
|
|
69
|
+
} as const
|
|
70
|
+
createRoot(document.getElementById('root')).render(
|
|
71
|
+
<StrictMode>
|
|
72
|
+
<PostHogProvider apiKey={import.meta.env.VITE_PUBLIC_POSTHOG_PROJECT_TOKEN} options={options}>
|
|
73
|
+
<App />
|
|
74
|
+
</PostHogProvider>
|
|
75
|
+
</StrictMode>
|
|
76
|
+
)
|
|
77
|
+
```
|
|
78
|
+
|
|
79
|
+
**defaults option**
|
|
80
|
+
|
|
81
|
+
The `defaults` option automatically configures PostHog with recommended settings for new projects. See [SDK defaults](/docs/libraries/js.md#sdk-defaults) for details.
|
|
82
|
+
|
|
83
|
+
4. 4
|
|
84
|
+
|
|
85
|
+
## Accessing PostHog in your code
|
|
86
|
+
|
|
87
|
+
Recommended
|
|
88
|
+
|
|
89
|
+
Use the `usePostHog` hook to access the PostHog instance in any component wrapped by `PostHogProvider`:
|
|
90
|
+
|
|
91
|
+
MyComponent.tsx
|
|
92
|
+
|
|
93
|
+
PostHog AI
|
|
94
|
+
|
|
95
|
+
```jsx
|
|
96
|
+
import { usePostHog } from '@posthog/react'
|
|
97
|
+
function MyComponent() {
|
|
98
|
+
const posthog = usePostHog()
|
|
99
|
+
function handleClick() {
|
|
100
|
+
posthog.capture('button_clicked', { button_name: 'signup' })
|
|
101
|
+
}
|
|
102
|
+
return <button onClick={handleClick}>Sign up</button>
|
|
103
|
+
}
|
|
104
|
+
```
|
|
105
|
+
|
|
106
|
+
You can also import `posthog` directly for non-React code or utility functions:
|
|
107
|
+
|
|
108
|
+
utils/analytics.ts
|
|
109
|
+
|
|
110
|
+
PostHog AI
|
|
111
|
+
|
|
112
|
+
```jsx
|
|
113
|
+
import posthog from 'posthog-js'
|
|
114
|
+
export function trackPurchase(amount: number) {
|
|
115
|
+
posthog.capture('purchase_completed', { amount })
|
|
116
|
+
}
|
|
117
|
+
```
|
|
118
|
+
|
|
119
|
+
5. 5
|
|
120
|
+
|
|
121
|
+
## Send events
|
|
122
|
+
|
|
123
|
+
Recommended
|
|
124
|
+
|
|
125
|
+
Click around and view a couple pages to generate some events. PostHog automatically captures pageviews, clicks, and other interactions for you.
|
|
126
|
+
|
|
127
|
+
If you'd like, you can also manually capture custom events:
|
|
128
|
+
|
|
129
|
+
JavaScript
|
|
130
|
+
|
|
131
|
+
PostHog AI
|
|
132
|
+
|
|
133
|
+
```javascript
|
|
134
|
+
posthog.capture('my_custom_event', { property: 'value' })
|
|
135
|
+
```
|
|
136
|
+
|
|
137
|
+
6. 6
|
|
138
|
+
|
|
139
|
+
## Use feature flags
|
|
140
|
+
|
|
141
|
+
Required
|
|
142
|
+
|
|
143
|
+
## Using hooks
|
|
144
|
+
|
|
145
|
+
PostHog provides several hooks to make it easy to use feature flags in your React app. Use `useFeatureFlagEnabled` for boolean flags:
|
|
146
|
+
|
|
147
|
+
```jsx
|
|
148
|
+
import { useFeatureFlagEnabled } from '@posthog/react'
|
|
149
|
+
function App() {
|
|
150
|
+
const showWelcomeMessage = useFeatureFlagEnabled('flag-key')
|
|
151
|
+
const payload = useFeatureFlagPayload('flag-key')
|
|
152
|
+
return (
|
|
153
|
+
<div className="App">
|
|
154
|
+
{showWelcomeMessage ? (
|
|
155
|
+
<div>
|
|
156
|
+
<h1>Welcome!</h1>
|
|
157
|
+
<p>Thanks for trying out our feature flags.</p>
|
|
158
|
+
</div>
|
|
159
|
+
) : (
|
|
160
|
+
<div>
|
|
161
|
+
<h2>No welcome message</h2>
|
|
162
|
+
<p>Because the feature flag evaluated to false.</p>
|
|
163
|
+
</div>
|
|
164
|
+
)}
|
|
165
|
+
</div>
|
|
166
|
+
)
|
|
167
|
+
}
|
|
168
|
+
```
|
|
169
|
+
|
|
170
|
+
### Multivariate flags
|
|
171
|
+
|
|
172
|
+
For multivariate flags, use `useFeatureFlagVariantKey`:
|
|
173
|
+
|
|
174
|
+
```jsx
|
|
175
|
+
import { useFeatureFlagVariantKey } from '@posthog/react'
|
|
176
|
+
function App() {
|
|
177
|
+
const variantKey = useFeatureFlagVariantKey('show-welcome-message')
|
|
178
|
+
let welcomeMessage = ''
|
|
179
|
+
if (variantKey === 'variant-a') {
|
|
180
|
+
welcomeMessage = 'Welcome to the Alpha!'
|
|
181
|
+
} else if (variantKey === 'variant-b') {
|
|
182
|
+
welcomeMessage = 'Welcome to the Beta!'
|
|
183
|
+
}
|
|
184
|
+
return (
|
|
185
|
+
<div className="App">
|
|
186
|
+
{welcomeMessage ? (
|
|
187
|
+
<div>
|
|
188
|
+
<h1>{welcomeMessage}</h1>
|
|
189
|
+
<p>Thanks for trying out our feature flags.</p>
|
|
190
|
+
</div>
|
|
191
|
+
) : (
|
|
192
|
+
<div>
|
|
193
|
+
<h2>No welcome message</h2>
|
|
194
|
+
<p>Because the feature flag evaluated to false.</p>
|
|
195
|
+
</div>
|
|
196
|
+
)}
|
|
197
|
+
</div>
|
|
198
|
+
)
|
|
199
|
+
}
|
|
200
|
+
```
|
|
201
|
+
|
|
202
|
+
### Flag payloads
|
|
203
|
+
|
|
204
|
+
The `useFeatureFlagPayload` hook does *not* send a `$feature_flag_called` event, which is required for experiments. Always use it with `useFeatureFlagEnabled` or `useFeatureFlagVariantKey`:
|
|
205
|
+
|
|
206
|
+
```jsx
|
|
207
|
+
import { useFeatureFlagPayload, useFeatureFlagEnabled } from '@posthog/react'
|
|
208
|
+
function App() {
|
|
209
|
+
const variant = useFeatureFlagEnabled('show-welcome-message')
|
|
210
|
+
const payload = useFeatureFlagPayload('show-welcome-message')
|
|
211
|
+
return (
|
|
212
|
+
<>
|
|
213
|
+
{variant ? (
|
|
214
|
+
<div className="welcome-message">
|
|
215
|
+
<h2>{payload?.welcomeTitle}</h2>
|
|
216
|
+
<p>{payload?.welcomeMessage}</p>
|
|
217
|
+
</div>
|
|
218
|
+
) : (
|
|
219
|
+
<div>
|
|
220
|
+
<h2>No custom welcome message</h2>
|
|
221
|
+
<p>Because the feature flag evaluated to false.</p>
|
|
222
|
+
</div>
|
|
223
|
+
)}
|
|
224
|
+
</>
|
|
225
|
+
)
|
|
226
|
+
}
|
|
227
|
+
```
|
|
228
|
+
|
|
229
|
+
## Using PostHogFeature component
|
|
230
|
+
|
|
231
|
+
The `PostHogFeature` component simplifies code by handling feature flag related logic:
|
|
232
|
+
|
|
233
|
+
App.tsx
|
|
234
|
+
|
|
235
|
+
PostHog AI
|
|
236
|
+
|
|
237
|
+
```jsx
|
|
238
|
+
import { PostHogFeature } from '@posthog/react'
|
|
239
|
+
function App() {
|
|
240
|
+
return (
|
|
241
|
+
<PostHogFeature flag='show-welcome-message' match={true}>
|
|
242
|
+
<div>
|
|
243
|
+
<h1>Hello</h1>
|
|
244
|
+
<p>Thanks for trying out our feature flags.</p>
|
|
245
|
+
</div>
|
|
246
|
+
</PostHogFeature>
|
|
247
|
+
)
|
|
248
|
+
}
|
|
249
|
+
```
|
|
250
|
+
|
|
251
|
+
The `match` prop can be either `true`, or the variant key, to match on a specific variant. If you also want to show a default message, you can pass these in the `fallback` prop.
|
|
252
|
+
|
|
253
|
+
If your flag has a payload, you can pass a function to children whose first argument is the payload:
|
|
254
|
+
|
|
255
|
+
App.tsx
|
|
256
|
+
|
|
257
|
+
PostHog AI
|
|
258
|
+
|
|
259
|
+
```jsx
|
|
260
|
+
<PostHogFeature flag='show-welcome-message' match={true}>
|
|
261
|
+
{(payload) => {
|
|
262
|
+
return (
|
|
263
|
+
<div>
|
|
264
|
+
<h1>{payload.welcomeMessage}</h1>
|
|
265
|
+
<p>Thanks for trying out our feature flags.</p>
|
|
266
|
+
</div>
|
|
267
|
+
)
|
|
268
|
+
}}
|
|
269
|
+
</PostHogFeature>
|
|
270
|
+
```
|
|
271
|
+
|
|
272
|
+
7. 7
|
|
273
|
+
|
|
274
|
+
## Running experiments
|
|
275
|
+
|
|
276
|
+
Optional
|
|
277
|
+
|
|
278
|
+
Experiments run on top of our feature flags. Once you've implemented the flag in your code, you run an experiment by creating a new experiment in the PostHog dashboard.
|
|
279
|
+
|
|
280
|
+
8. 8
|
|
281
|
+
|
|
282
|
+
## Next steps
|
|
283
|
+
|
|
284
|
+
Recommended
|
|
285
|
+
|
|
286
|
+
Now that you're evaluating flags, continue with the resources below to learn what else Feature Flags enables within the PostHog platform.
|
|
287
|
+
|
|
288
|
+
| Resource | Description |
|
|
289
|
+
| --- | --- |
|
|
290
|
+
| [Creating a feature flag](/docs/feature-flags/creating-feature-flags.md) | How to create a feature flag in PostHog |
|
|
291
|
+
| [Adding feature flag code](/docs/feature-flags/adding-feature-flag-code.md) | How to check flags in your code for all platforms |
|
|
292
|
+
| [Framework-specific guides](/docs/feature-flags/tutorials.md#framework-guides) | Setup guides for React Native, Next.js, Flutter, and other frameworks |
|
|
293
|
+
| [How to do a phased rollout](/tutorials/phased-rollout.md) | Gradually roll out features to minimize risk |
|
|
294
|
+
| [More tutorials](/docs/feature-flags/tutorials.md) | Other real-world examples and use cases |
|
|
295
|
+
|
|
296
|
+
### Community questions
|
|
297
|
+
|
|
298
|
+
Ask a question
|
|
299
|
+
|
|
300
|
+
### Was this page useful?
|
|
301
|
+
|
|
302
|
+
HelpfulCould be better
|
|
@@ -0,0 +1,202 @@
|
|
|
1
|
+
# Identify users - Docs
|
|
2
|
+
|
|
3
|
+
Linking events to specific users enables you to build a full picture of how they're using your product across different sessions, devices, and platforms.
|
|
4
|
+
|
|
5
|
+
This is straightforward to do when [capturing backend events](/docs/product-analytics/capture-events?tab=Node.js.md), as you associate events to a specific user using a `distinct_id`, which is a required argument.
|
|
6
|
+
|
|
7
|
+
However, in the frontend of a [web](/docs/libraries/js/features.md#capturing-events) or [mobile app](/docs/libraries/ios.md#capturing-events), a `distinct_id` is not a required argument — PostHog's SDKs will generate an anonymous `distinct_id` for you automatically and you can capture events anonymously, provided you use the appropriate [configuration](/docs/libraries/js/features.md#capturing-anonymous-events).
|
|
8
|
+
|
|
9
|
+
To link events to specific users, call `identify`:
|
|
10
|
+
|
|
11
|
+
PostHog AI
|
|
12
|
+
|
|
13
|
+
### Web
|
|
14
|
+
|
|
15
|
+
```javascript
|
|
16
|
+
posthog.identify(
|
|
17
|
+
'distinct_id', // Replace 'distinct_id' with your user's unique identifier
|
|
18
|
+
{ email: 'max@hedgehogmail.com', name: 'Max Hedgehog' } // optional: set additional person properties
|
|
19
|
+
);
|
|
20
|
+
```
|
|
21
|
+
|
|
22
|
+
### Android
|
|
23
|
+
|
|
24
|
+
```kotlin
|
|
25
|
+
PostHog.identify(
|
|
26
|
+
distinctId = distinctID, // Replace 'distinctID' with your user's unique identifier
|
|
27
|
+
// optional: set additional person properties
|
|
28
|
+
userProperties = mapOf(
|
|
29
|
+
"name" to "Max Hedgehog",
|
|
30
|
+
"email" to "max@hedgehogmail.com"
|
|
31
|
+
)
|
|
32
|
+
)
|
|
33
|
+
```
|
|
34
|
+
|
|
35
|
+
### iOS
|
|
36
|
+
|
|
37
|
+
```swift
|
|
38
|
+
PostHogSDK.shared.identify("distinct_id", // Replace "distinct_id" with your user's unique identifier
|
|
39
|
+
userProperties: ["name": "Max Hedgehog", "email": "max@hedgehogmail.com"]) // optional: set additional person properties
|
|
40
|
+
```
|
|
41
|
+
|
|
42
|
+
### React Native
|
|
43
|
+
|
|
44
|
+
```jsx
|
|
45
|
+
posthog.identify('distinct_id', { // Replace "distinct_id" with your user's unique identifier
|
|
46
|
+
email: 'max@hedgehogmail.com', // optional: set additional person properties
|
|
47
|
+
name: 'Max Hedgehog'
|
|
48
|
+
})
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
### Dart
|
|
52
|
+
|
|
53
|
+
```dart
|
|
54
|
+
await Posthog().identify(
|
|
55
|
+
userId: 'distinct_id', // Replace "distinct_id" with your user's unique identifier
|
|
56
|
+
userProperties: {
|
|
57
|
+
email: "max@hedgehogmail.com", // optional: set additional person properties
|
|
58
|
+
name: "Max Hedgehog"
|
|
59
|
+
});
|
|
60
|
+
```
|
|
61
|
+
|
|
62
|
+
Events captured after calling `identify` are identified events and this creates a person profile if one doesn't exist already.
|
|
63
|
+
|
|
64
|
+
Due to the cost of processing them, anonymous events can be up to 4x cheaper than identified events, so it's recommended you only capture identified events when needed.
|
|
65
|
+
|
|
66
|
+
## How identify works
|
|
67
|
+
|
|
68
|
+
When a user starts browsing your website or app, PostHog automatically assigns them an **anonymous ID**, which is stored locally.
|
|
69
|
+
|
|
70
|
+
Provided you've [configured persistence](/docs/libraries/js/persistence.md) to use cookies or `localStorage`, this enables us to track anonymous users – even across different sessions.
|
|
71
|
+
|
|
72
|
+
By calling `identify` with a `distinct_id` of your choice (usually the user's ID in your database, or their email), you link the anonymous ID and distinct ID together.
|
|
73
|
+
|
|
74
|
+
Thus, all past and future events made with that anonymous ID are now associated with the distinct ID.
|
|
75
|
+
|
|
76
|
+
This enables you to do things like associate events with a user from before they log in for the first time, or associate their events across different devices or platforms.
|
|
77
|
+
|
|
78
|
+
Using identify in the backend
|
|
79
|
+
|
|
80
|
+
Although you can call `identify` using our backend SDKs, it is used most in frontends. This is because there is no concept of anonymous sessions in the backend SDKs, so calling `identify` only updates person profiles.
|
|
81
|
+
|
|
82
|
+
## Best practices when using `identify`
|
|
83
|
+
|
|
84
|
+
### 1\. Call `identify` as soon as you're able to
|
|
85
|
+
|
|
86
|
+
In your frontend, you should call `identify` as soon as you're able to.
|
|
87
|
+
|
|
88
|
+
Typically, this is every time your **app loads** for the first time, and directly after your **users log in**.
|
|
89
|
+
|
|
90
|
+
This ensures that events sent during your users' sessions are correctly associated with them.
|
|
91
|
+
|
|
92
|
+
You only need to call `identify` once per session, and you should avoid calling it multiple times unnecessarily.
|
|
93
|
+
|
|
94
|
+
If you call `identify` multiple times with the same data without reloading the page in between, PostHog will ignore the subsequent calls.
|
|
95
|
+
|
|
96
|
+
### 2\. Use unique strings for distinct IDs
|
|
97
|
+
|
|
98
|
+
If two users have the same distinct ID, their data is merged and they are considered one user in PostHog. Two common ways this can happen are:
|
|
99
|
+
|
|
100
|
+
- Your logic for generating IDs does not generate sufficiently strong IDs and you can end up with a clash where 2 users have the same ID.
|
|
101
|
+
- There's a bug, typo, or mistake in your code leading to most or all users being identified with generic IDs like `null`, `true`, or `distinctId`.
|
|
102
|
+
|
|
103
|
+
PostHog also has built-in protections to stop the most common distinct ID mistakes.
|
|
104
|
+
|
|
105
|
+
### 3\. Reset after logout
|
|
106
|
+
|
|
107
|
+
If a user logs out on your frontend, you should call `reset()` to unlink any future events made on that device with that user.
|
|
108
|
+
|
|
109
|
+
This is important if your users are sharing a computer, as otherwise all of those users are grouped together into a single user due to shared cookies between sessions.
|
|
110
|
+
|
|
111
|
+
**We strongly recommend you call `reset` on logout even if you don't expect users to share a computer.**
|
|
112
|
+
|
|
113
|
+
You can do that like so:
|
|
114
|
+
|
|
115
|
+
PostHog AI
|
|
116
|
+
|
|
117
|
+
### Web
|
|
118
|
+
|
|
119
|
+
```javascript
|
|
120
|
+
posthog.reset()
|
|
121
|
+
```
|
|
122
|
+
|
|
123
|
+
### iOS
|
|
124
|
+
|
|
125
|
+
```swift
|
|
126
|
+
PostHogSDK.shared.reset()
|
|
127
|
+
```
|
|
128
|
+
|
|
129
|
+
### Android
|
|
130
|
+
|
|
131
|
+
```kotlin
|
|
132
|
+
PostHog.reset()
|
|
133
|
+
```
|
|
134
|
+
|
|
135
|
+
### React Native
|
|
136
|
+
|
|
137
|
+
```jsx
|
|
138
|
+
posthog.reset()
|
|
139
|
+
```
|
|
140
|
+
|
|
141
|
+
### Dart
|
|
142
|
+
|
|
143
|
+
```dart
|
|
144
|
+
Posthog().reset()
|
|
145
|
+
```
|
|
146
|
+
|
|
147
|
+
If you *also* want to reset the `device_id` so that the device will be considered a new device in future events, you can pass `true` as an argument:
|
|
148
|
+
|
|
149
|
+
Web
|
|
150
|
+
|
|
151
|
+
PostHog AI
|
|
152
|
+
|
|
153
|
+
```javascript
|
|
154
|
+
posthog.reset(true)
|
|
155
|
+
```
|
|
156
|
+
|
|
157
|
+
### 4\. Person profiles and properties
|
|
158
|
+
|
|
159
|
+
You'll notice that one of the parameters in the `identify` method is a `properties` object.
|
|
160
|
+
|
|
161
|
+
This enables you to set [person properties](/docs/product-analytics/person-properties.md).
|
|
162
|
+
|
|
163
|
+
Whenever possible, we recommend passing in all person properties you have available each time you call identify, as this ensures their person profile on PostHog is up to date.
|
|
164
|
+
|
|
165
|
+
Person properties can also be set being adding a `$set` property to a event `capture` call.
|
|
166
|
+
|
|
167
|
+
See our [person properties docs](/docs/product-analytics/person-properties.md) for more details on how to work with them and best practices.
|
|
168
|
+
|
|
169
|
+
### 5\. Use deep links between platforms
|
|
170
|
+
|
|
171
|
+
We recommend you call `identify` [as soon as you're able](#1-call-identify-as-soon-as-youre-able), typically when a user signs up or logs in.
|
|
172
|
+
|
|
173
|
+
This doesn't work if one or both platforms are unauthenticated. Some examples of such cases are:
|
|
174
|
+
|
|
175
|
+
- Onboarding and signup flows before authentication.
|
|
176
|
+
- Unauthenticated web pages redirecting to authenticated mobile apps.
|
|
177
|
+
- Authenticated web apps prompting an app download.
|
|
178
|
+
|
|
179
|
+
In these cases, you can use a [deep link](https://developer.android.com/training/app-links/deep-linking) on Android and [universal links](https://developer.apple.com/documentation/xcode/supporting-universal-links-in-your-app) on iOS to identify users.
|
|
180
|
+
|
|
181
|
+
1. Use `posthog.get_distinct_id()` to get the current distinct ID. Even if you cannot call identify because the user is unauthenticated, this will return an anonymous distinct ID generated by PostHog.
|
|
182
|
+
2. Add the distinct ID to the deep link as query parameters, along with other properties like UTM parameters.
|
|
183
|
+
3. When the user is redirected to the app, parse the deep link and handle the following cases:
|
|
184
|
+
|
|
185
|
+
- The user is already authenticated on the mobile app. In this case, call [`posthog.alias()`](/docs/libraries/js/features.md#alias) with the distinct ID from the web. This associates the two distinct IDs as a single person.
|
|
186
|
+
- The user is unauthenticated. In this case, call [`posthog.identify()`](/docs/libraries/js/features.md#identifying-users) with the distinct ID from the web. Events will be associated with this distinct ID.
|
|
187
|
+
|
|
188
|
+
As long as you associate the distinct IDs with `posthog.identify()` or `posthog.alias()`, you can track events generated across platforms.
|
|
189
|
+
|
|
190
|
+
## Further reading
|
|
191
|
+
|
|
192
|
+
- [Identifying users docs](/docs/product-analytics/identify.md)
|
|
193
|
+
- [How person processing works](/docs/how-posthog-works/ingestion-pipeline.md#2-person-processing)
|
|
194
|
+
- [An introductory guide to identifying users in PostHog](/tutorials/identifying-users-guide.md)
|
|
195
|
+
|
|
196
|
+
### Community questions
|
|
197
|
+
|
|
198
|
+
Ask a question
|
|
199
|
+
|
|
200
|
+
### Was this page useful?
|
|
201
|
+
|
|
202
|
+
HelpfulCould be better
|