mirror of
https://github.com/formbricks/formbricks.git
synced 2026-04-24 03:21:20 -05:00
Compare commits
82 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 0878428c79 | |||
| b655e649ac | |||
| ddb95b7cbb | |||
| eced597e8a | |||
| 93c72df4d9 | |||
| 49560ccba8 | |||
| 3f98283d4d | |||
| 7b64422a3f | |||
| da72101320 | |||
| 5f02ad49c1 | |||
| 6644bba6ea | |||
| 0b7734f725 | |||
| 1536bf6907 | |||
| e81190214f | |||
| 48c8906a89 | |||
| 717b30115b | |||
| 1f3962d2d5 | |||
| 619f6e408f | |||
| 4a8719abaa | |||
| 7b59eb3b26 | |||
| 8ac280268d | |||
| 34e8f4931d | |||
| ac46850a24 | |||
| 6328be220a | |||
| 882ad99ed7 | |||
| ce47b4c2d8 | |||
| ce8f9de8ec | |||
| ed3c2d2b58 | |||
| 9ae226329b | |||
| 12c3899b85 | |||
| ccb1353eb5 | |||
| 22eb0b79ee | |||
| 5eb7a496da | |||
| 7ea55e199f | |||
| 83eb472acd | |||
| d9fe6ee4f4 | |||
| 51b58be079 | |||
| 397643330a | |||
| e5fa4328e1 | |||
| 4b777f1907 | |||
| c3547ccb36 | |||
| a0f334b300 | |||
| a9f635b768 | |||
| d385b4a0d6 | |||
| 5e825413d2 | |||
| 8c3e816ccd | |||
| 6ddc91ee85 | |||
| 14023ca8a9 | |||
| 385e8a4262 | |||
| e358104f7c | |||
| c8e9194ab6 | |||
| bebe29815d | |||
| 7f40502c94 | |||
| 5fb5215680 | |||
| 19b80ff042 | |||
| 2dfdba2acf | |||
| f7842789de | |||
| 59bdd5f065 | |||
| 8da1bc71a6 | |||
| 0e0259691c | |||
| ac7831fa3d | |||
| db32cb392f | |||
| e5cb01bd88 | |||
| cbef4c2a69 | |||
| 86948b70de | |||
| dfe955ca7c | |||
| a7ee1f189f | |||
| 46a590311b | |||
| 0faeffb624 | |||
| d9727a336a | |||
| 330e0db668 | |||
| f5b7f73199 | |||
| c02f070307 | |||
| bc489e050a | |||
| 3062059ed5 | |||
| f27ede6b2c | |||
| e460ff5100 | |||
| 4699c0014b | |||
| 52f69be05d | |||
| 619c0983a4 | |||
| 964fb8d4f4 | |||
| 5391c60bba |
@@ -0,0 +1,216 @@
|
|||||||
|
---
|
||||||
|
description:
|
||||||
|
globs:
|
||||||
|
alwaysApply: false
|
||||||
|
---
|
||||||
|
# Component Migration Automation Rule
|
||||||
|
|
||||||
|
## Overview
|
||||||
|
This rule automates the migration of deprecated components to new component systems in React/TypeScript codebases.
|
||||||
|
|
||||||
|
## Trigger
|
||||||
|
When the user requests component migration (e.g., "migrate [DeprecatedComponent] to [NewComponent]" or "component migration").
|
||||||
|
|
||||||
|
## Process
|
||||||
|
|
||||||
|
### Step 1: Discovery and Planning
|
||||||
|
1. **Identify migration parameters:**
|
||||||
|
- Ask user for deprecated component name (e.g., "Modal")
|
||||||
|
- Ask user for new component name(s) (e.g., "Dialog")
|
||||||
|
- Ask for any components to exclude (e.g., "ModalWithTabs")
|
||||||
|
- Ask for specific import paths if needed
|
||||||
|
|
||||||
|
2. **Scan codebase** for deprecated components:
|
||||||
|
- Search for `import.*[DeprecatedComponent]` patterns
|
||||||
|
- Exclude specified components that should not be migrated
|
||||||
|
- List all found components with file paths
|
||||||
|
- Present numbered list to user for confirmation
|
||||||
|
|
||||||
|
### Step 2: Component-by-Component Migration
|
||||||
|
For each component, follow this exact sequence:
|
||||||
|
|
||||||
|
#### 2.1 Component Migration
|
||||||
|
- **Import changes:**
|
||||||
|
- Ask user to provide the new import structure
|
||||||
|
- Example transformation pattern:
|
||||||
|
```typescript
|
||||||
|
// FROM:
|
||||||
|
import { [DeprecatedComponent] } from "@/components/ui/[DeprecatedComponent]"
|
||||||
|
|
||||||
|
// TO:
|
||||||
|
import {
|
||||||
|
[NewComponent],
|
||||||
|
[NewComponentPart1],
|
||||||
|
[NewComponentPart2],
|
||||||
|
// ... other parts
|
||||||
|
} from "@/components/ui/[NewComponent]"
|
||||||
|
```
|
||||||
|
|
||||||
|
- **Props transformation:**
|
||||||
|
- Ask user for prop mapping rules (e.g., `open` → `open`, `setOpen` → `onOpenChange`)
|
||||||
|
- Ask for props to remove (e.g., `noPadding`, `closeOnOutsideClick`, `size`)
|
||||||
|
- Apply transformations based on user specifications
|
||||||
|
|
||||||
|
- **Structure transformation:**
|
||||||
|
- Ask user for the new component structure pattern
|
||||||
|
- Apply the transformation maintaining all functionality
|
||||||
|
- Preserve all existing logic, state management, and event handlers
|
||||||
|
|
||||||
|
#### 2.2 Wait for User Approval
|
||||||
|
- Present the migration changes
|
||||||
|
- Wait for explicit user approval before proceeding
|
||||||
|
- If rejected, ask for specific feedback and iterate
|
||||||
|
#### 2.3 Re-read and Apply Additional Changes
|
||||||
|
- Re-read the component file to capture any user modifications
|
||||||
|
- Apply any additional improvements the user made
|
||||||
|
- Ensure all changes are incorporated
|
||||||
|
|
||||||
|
#### 2.4 Test File Updates
|
||||||
|
- **Find corresponding test file** (same name with `.test.tsx` or `.test.ts`)
|
||||||
|
- **Update test mocks:**
|
||||||
|
- Ask user for new component mock structure
|
||||||
|
- Replace old component mocks with new ones
|
||||||
|
- Example pattern:
|
||||||
|
```typescript
|
||||||
|
// Add to test setup:
|
||||||
|
jest.mock("@/components/ui/[NewComponent]", () => ({
|
||||||
|
[NewComponent]: ({ children, [props] }: any) => ([mock implementation]),
|
||||||
|
[NewComponentPart1]: ({ children }: any) => <div data-testid="[new-component-part1]">{children}</div>,
|
||||||
|
[NewComponentPart2]: ({ children }: any) => <div data-testid="[new-component-part2]">{children}</div>,
|
||||||
|
// ... other parts
|
||||||
|
}));
|
||||||
|
```
|
||||||
|
- **Update test expectations:**
|
||||||
|
- Change test IDs from old component to new component
|
||||||
|
- Update any component-specific assertions
|
||||||
|
- Ensure all new component parts used in the component are mocked
|
||||||
|
|
||||||
|
#### 2.5 Run Tests and Optimize
|
||||||
|
- Execute `Node package manager test -- ComponentName.test.tsx`
|
||||||
|
- Fix any failing tests
|
||||||
|
- Optimize code quality (imports, formatting, etc.)
|
||||||
|
- Re-run tests until all pass
|
||||||
|
- **Maximum 3 iterations** - if still failing, ask user for guidance
|
||||||
|
|
||||||
|
#### 2.6 Wait for Final Approval
|
||||||
|
- Present test results and any optimizations made
|
||||||
|
- Wait for user approval of the complete migration
|
||||||
|
- If rejected, iterate based on feedback
|
||||||
|
|
||||||
|
#### 2.7 Git Commit
|
||||||
|
- Run: `git add .`
|
||||||
|
- Run: `git commit -m "migrate [ComponentName] from [DeprecatedComponent] to [NewComponent]"`
|
||||||
|
- Confirm commit was successful
|
||||||
|
|
||||||
|
### Step 3: Final Report Generation
|
||||||
|
After all components are migrated, generate a comprehensive GitHub PR report:
|
||||||
|
|
||||||
|
#### PR Title
|
||||||
|
```
|
||||||
|
feat: migrate [DeprecatedComponent] components to [NewComponent] system
|
||||||
|
```
|
||||||
|
|
||||||
|
#### PR Description Template
|
||||||
|
```markdown
|
||||||
|
## 🔄 [DeprecatedComponent] to [NewComponent] Migration
|
||||||
|
|
||||||
|
### Overview
|
||||||
|
Migrated [X] [DeprecatedComponent] components to the new [NewComponent] component system to modernize the UI architecture and improve consistency.
|
||||||
|
|
||||||
|
### Components Migrated
|
||||||
|
[List each component with file path]
|
||||||
|
|
||||||
|
### Technical Changes
|
||||||
|
- **Imports:** Replaced `[DeprecatedComponent]` with `[NewComponent], [NewComponentParts...]`
|
||||||
|
- **Props:** [List prop transformations]
|
||||||
|
- **Structure:** Implemented proper [NewComponent] component hierarchy
|
||||||
|
- **Styling:** [Describe styling changes]
|
||||||
|
- **Tests:** Updated all test mocks and expectations
|
||||||
|
|
||||||
|
### Migration Pattern
|
||||||
|
```typescript
|
||||||
|
// Before
|
||||||
|
<[DeprecatedComponent] [oldProps]>
|
||||||
|
[oldStructure]
|
||||||
|
</[DeprecatedComponent]>
|
||||||
|
|
||||||
|
// After
|
||||||
|
<[NewComponent] [newProps]>
|
||||||
|
[newStructure]
|
||||||
|
</[NewComponent]>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Testing
|
||||||
|
- ✅ All existing tests updated and passing
|
||||||
|
- ✅ Component functionality preserved
|
||||||
|
- ✅ UI/UX behavior maintained
|
||||||
|
|
||||||
|
### How to Test This PR
|
||||||
|
1. **Functional Testing:**
|
||||||
|
- Navigate to each migrated component's usage
|
||||||
|
- Verify [component] opens and closes correctly
|
||||||
|
- Test all interactive elements within [components]
|
||||||
|
- Confirm styling and layout are preserved
|
||||||
|
|
||||||
|
2. **Automated Testing:**
|
||||||
|
```bash
|
||||||
|
Node package manager test
|
||||||
|
```
|
||||||
|
|
||||||
|
3. **Visual Testing:**
|
||||||
|
- Check that all [components] maintain proper styling
|
||||||
|
- Verify responsive behavior
|
||||||
|
- Test keyboard navigation and accessibility
|
||||||
|
|
||||||
|
### Breaking Changes
|
||||||
|
[List any breaking changes or state "None - this is a drop-in replacement maintaining all existing functionality."]
|
||||||
|
|
||||||
|
### Notes
|
||||||
|
- [Any excluded components] were preserved as they already use [NewComponent] internally
|
||||||
|
- All form validation and complex state management preserved
|
||||||
|
- Enhanced code quality with better imports and formatting
|
||||||
|
```
|
||||||
|
|
||||||
|
## Special Considerations
|
||||||
|
|
||||||
|
### Excluded Components
|
||||||
|
- **DO NOT MIGRATE** components specified by user as exclusions
|
||||||
|
- They may already use the new component internally or have other reasons
|
||||||
|
- Inform user these are skipped and why
|
||||||
|
|
||||||
|
### Complex Components
|
||||||
|
- Preserve all existing functionality (forms, validation, state management)
|
||||||
|
- Maintain prop interfaces
|
||||||
|
- Keep all event handlers and callbacks
|
||||||
|
- Preserve accessibility features
|
||||||
|
|
||||||
|
### Test Coverage
|
||||||
|
- Ensure all new component parts are mocked when used
|
||||||
|
- Mock all new component parts that appear in the component
|
||||||
|
- Update test IDs from old component to new component
|
||||||
|
- Maintain all existing test scenarios
|
||||||
|
|
||||||
|
### Error Handling
|
||||||
|
- If tests fail after 3 iterations, stop and ask user for guidance
|
||||||
|
- If component is too complex, ask user for specific guidance
|
||||||
|
- If unsure about functionality preservation, ask for clarification
|
||||||
|
|
||||||
|
### Migration Patterns
|
||||||
|
- Always ask user for specific migration patterns before starting
|
||||||
|
- Confirm import structures, prop mappings, and component hierarchies
|
||||||
|
- Adapt to different component architectures (simple replacements, complex restructuring, etc.)
|
||||||
|
|
||||||
|
## Success Criteria
|
||||||
|
- All deprecated components successfully migrated to new components
|
||||||
|
- All tests passing
|
||||||
|
- No functionality lost
|
||||||
|
- Code quality maintained or improved
|
||||||
|
- User approval on each component
|
||||||
|
- Successful git commits for each migration
|
||||||
|
- Comprehensive PR report generated
|
||||||
|
|
||||||
|
## Usage Examples
|
||||||
|
- "migrate Modal to Dialog"
|
||||||
|
- "migrate Button to NewButton"
|
||||||
|
- "migrate Card to ModernCard"
|
||||||
|
- "component migration" (will prompt for details)
|
||||||
@@ -0,0 +1,101 @@
|
|||||||
|
---
|
||||||
|
description: >
|
||||||
|
This rule provides comprehensive knowledge about the Formbricks database structure, relationships,
|
||||||
|
and data patterns. It should be used **only when the agent explicitly requests database schema-level
|
||||||
|
details** to support tasks such as: writing/debugging Prisma queries, designing/reviewing data models,
|
||||||
|
investigating multi-tenancy behavior, creating API endpoints, or understanding data relationships.
|
||||||
|
globs: []
|
||||||
|
alwaysApply: agent-requested
|
||||||
|
---
|
||||||
|
# Formbricks Database Schema Reference
|
||||||
|
|
||||||
|
This rule provides a reference to the Formbricks database structure. For the most up-to-date and complete schema definitions, please refer to the schema.prisma file directly.
|
||||||
|
|
||||||
|
## Database Overview
|
||||||
|
|
||||||
|
Formbricks uses PostgreSQL with Prisma ORM. The schema is designed for multi-tenancy with strong data isolation between organizations.
|
||||||
|
|
||||||
|
### Core Hierarchy
|
||||||
|
```
|
||||||
|
Organization
|
||||||
|
└── Project
|
||||||
|
└── Environment (production/development)
|
||||||
|
├── Survey
|
||||||
|
├── Contact
|
||||||
|
├── ActionClass
|
||||||
|
└── Integration
|
||||||
|
```
|
||||||
|
|
||||||
|
## Schema Reference
|
||||||
|
|
||||||
|
For the complete and up-to-date database schema, please refer to:
|
||||||
|
- Main schema: `packages/database/schema.prisma`
|
||||||
|
- JSON type definitions: `packages/database/json-types.ts`
|
||||||
|
|
||||||
|
The schema.prisma file contains all model definitions, relationships, enums, and field types. The json-types.ts file contains TypeScript type definitions for JSON fields.
|
||||||
|
|
||||||
|
## Data Access Patterns
|
||||||
|
|
||||||
|
### Multi-tenancy
|
||||||
|
- All data is scoped by Organization
|
||||||
|
- Environment-level isolation for surveys and contacts
|
||||||
|
- Project-level grouping for related surveys
|
||||||
|
|
||||||
|
### Soft Deletion
|
||||||
|
Some models use soft deletion patterns:
|
||||||
|
- Check `isActive` fields where present
|
||||||
|
- Use proper filtering in queries
|
||||||
|
|
||||||
|
### Cascading Deletes
|
||||||
|
Configured cascade relationships:
|
||||||
|
- Organization deletion cascades to all child entities
|
||||||
|
- Survey deletion removes responses, displays, triggers
|
||||||
|
- Contact deletion removes attributes and responses
|
||||||
|
|
||||||
|
## Common Query Patterns
|
||||||
|
|
||||||
|
### Survey with Responses
|
||||||
|
```typescript
|
||||||
|
// Include response count and latest responses
|
||||||
|
const survey = await prisma.survey.findUnique({
|
||||||
|
where: { id: surveyId },
|
||||||
|
include: {
|
||||||
|
responses: {
|
||||||
|
take: 10,
|
||||||
|
orderBy: { createdAt: 'desc' }
|
||||||
|
},
|
||||||
|
_count: {
|
||||||
|
select: { responses: true }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
### Environment Scoping
|
||||||
|
```typescript
|
||||||
|
// Always scope by environment
|
||||||
|
const surveys = await prisma.survey.findMany({
|
||||||
|
where: {
|
||||||
|
environmentId: environmentId,
|
||||||
|
// Additional filters...
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
### Contact with Attributes
|
||||||
|
```typescript
|
||||||
|
const contact = await prisma.contact.findUnique({
|
||||||
|
where: { id: contactId },
|
||||||
|
include: {
|
||||||
|
attributes: {
|
||||||
|
include: {
|
||||||
|
attributeKey: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
This schema supports Formbricks' core functionality: multi-tenant survey management, user targeting, response collection, and analysis, all while maintaining strict data isolation and security.
|
||||||
|
|
||||||
|
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
---
|
||||||
|
description: Guideline for writing end-user facing documentation in the apps/docs folder
|
||||||
|
globs:
|
||||||
|
alwaysApply: false
|
||||||
|
---
|
||||||
|
Follow these instructions and guidelines when asked to write documentation in the apps/docs folder
|
||||||
|
|
||||||
|
Follow this structure to write the title, describtion and pick a matching icon and insert it at the top of the MDX file:
|
||||||
|
|
||||||
|
---
|
||||||
|
title: "FEATURE NAME"
|
||||||
|
description: "1 concise sentence to describe WHEN the feature is being used and FOR WHAT BENEFIT."
|
||||||
|
icon: "link"
|
||||||
|
---
|
||||||
|
|
||||||
|
- Description: 1 concise sentence to describe WHEN the feature is being used and FOR WHAT BENEFIT.
|
||||||
|
- Make ample use of the Mintlify components you can find here https://mintlify.com/docs/llms.txt
|
||||||
|
- In all Headlines, only capitalize the current feature and nothing else, to Camel Case
|
||||||
|
- If a feature is part of the Enterprise Edition, use this note:
|
||||||
|
|
||||||
|
<Note>
|
||||||
|
FEATURE NAME is part of the @Enterprise Edition.
|
||||||
|
</Note>
|
||||||
+2
-2
@@ -80,8 +80,8 @@ S3_ENDPOINT_URL=
|
|||||||
# Force path style for S3 compatible storage (0 for disabled, 1 for enabled)
|
# Force path style for S3 compatible storage (0 for disabled, 1 for enabled)
|
||||||
S3_FORCE_PATH_STYLE=0
|
S3_FORCE_PATH_STYLE=0
|
||||||
|
|
||||||
# Set this URL to add a custom domain to your survey links(default is WEBAPP_URL)
|
# Set this URL to add a public domain for all your client facing routes(default is WEBAPP_URL)
|
||||||
# SURVEY_URL=https://survey.example.com
|
# PUBLIC_URL=https://survey.example.com
|
||||||
|
|
||||||
#####################
|
#####################
|
||||||
# Disable Features #
|
# Disable Features #
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
name: Feature request
|
name: Feature request
|
||||||
description: "Suggest an idea for this project \U0001F680"
|
description: "Suggest an idea for this project \U0001F680"
|
||||||
type: feature
|
type: feature
|
||||||
|
projects: "formbricks/21"
|
||||||
body:
|
body:
|
||||||
- type: textarea
|
- type: textarea
|
||||||
id: problem-description
|
id: problem-description
|
||||||
|
|||||||
@@ -1,11 +0,0 @@
|
|||||||
name: Task (internal)
|
|
||||||
description: "Template for creating a task. Used by the Formbricks Team only \U0001f4e5"
|
|
||||||
type: task
|
|
||||||
body:
|
|
||||||
- type: textarea
|
|
||||||
id: task-summary
|
|
||||||
attributes:
|
|
||||||
label: Task description
|
|
||||||
description: A clear detailed-rich description of the task.
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
@@ -4,16 +4,16 @@ on:
|
|||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
inputs:
|
inputs:
|
||||||
VERSION:
|
VERSION:
|
||||||
description: 'The version of the Docker image to release, full image tag if image tag is v0.0.0 enter v0.0.0.'
|
description: "The version of the Docker image to release, full image tag if image tag is v0.0.0 enter v0.0.0."
|
||||||
required: true
|
required: true
|
||||||
type: string
|
type: string
|
||||||
REPOSITORY:
|
REPOSITORY:
|
||||||
description: 'The repository to use for the Docker image'
|
description: "The repository to use for the Docker image"
|
||||||
required: false
|
required: false
|
||||||
type: string
|
type: string
|
||||||
default: 'ghcr.io/formbricks/formbricks'
|
default: "ghcr.io/formbricks/formbricks"
|
||||||
ENVIRONMENT:
|
ENVIRONMENT:
|
||||||
description: 'The environment to deploy to'
|
description: "The environment to deploy to"
|
||||||
required: true
|
required: true
|
||||||
type: choice
|
type: choice
|
||||||
options:
|
options:
|
||||||
@@ -22,16 +22,16 @@ on:
|
|||||||
workflow_call:
|
workflow_call:
|
||||||
inputs:
|
inputs:
|
||||||
VERSION:
|
VERSION:
|
||||||
description: 'The version of the Docker image to release'
|
description: "The version of the Docker image to release"
|
||||||
required: true
|
required: true
|
||||||
type: string
|
type: string
|
||||||
REPOSITORY:
|
REPOSITORY:
|
||||||
description: 'The repository to use for the Docker image'
|
description: "The repository to use for the Docker image"
|
||||||
required: false
|
required: false
|
||||||
type: string
|
type: string
|
||||||
default: 'ghcr.io/formbricks/formbricks'
|
default: "ghcr.io/formbricks/formbricks"
|
||||||
ENVIRONMENT:
|
ENVIRONMENT:
|
||||||
description: 'The environment to deploy to'
|
description: "The environment to deploy to"
|
||||||
required: true
|
required: true
|
||||||
type: string
|
type: string
|
||||||
|
|
||||||
@@ -75,7 +75,7 @@ jobs:
|
|||||||
FORMBRICKS_INGRESS_CERT_ARN: ${{ secrets.FORMBRICKS_INGRESS_CERT_ARN }}
|
FORMBRICKS_INGRESS_CERT_ARN: ${{ secrets.FORMBRICKS_INGRESS_CERT_ARN }}
|
||||||
FORMBRICKS_ROLE_ARN: ${{ secrets.FORMBRICKS_ROLE_ARN }}
|
FORMBRICKS_ROLE_ARN: ${{ secrets.FORMBRICKS_ROLE_ARN }}
|
||||||
with:
|
with:
|
||||||
helmfile-version: 'v1.0.0'
|
helmfile-version: "v1.0.0"
|
||||||
helm-plugins: >
|
helm-plugins: >
|
||||||
https://github.com/databus23/helm-diff,
|
https://github.com/databus23/helm-diff,
|
||||||
https://github.com/jkroepke/helm-secrets
|
https://github.com/jkroepke/helm-secrets
|
||||||
@@ -92,7 +92,7 @@ jobs:
|
|||||||
FORMBRICKS_INGRESS_CERT_ARN: ${{ secrets.STAGE_FORMBRICKS_INGRESS_CERT_ARN }}
|
FORMBRICKS_INGRESS_CERT_ARN: ${{ secrets.STAGE_FORMBRICKS_INGRESS_CERT_ARN }}
|
||||||
FORMBRICKS_ROLE_ARN: ${{ secrets.STAGE_FORMBRICKS_ROLE_ARN }}
|
FORMBRICKS_ROLE_ARN: ${{ secrets.STAGE_FORMBRICKS_ROLE_ARN }}
|
||||||
with:
|
with:
|
||||||
helmfile-version: 'v1.0.0'
|
helmfile-version: "v1.0.0"
|
||||||
helm-plugins: >
|
helm-plugins: >
|
||||||
https://github.com/databus23/helm-diff,
|
https://github.com/databus23/helm-diff,
|
||||||
https://github.com/jkroepke/helm-secrets
|
https://github.com/jkroepke/helm-secrets
|
||||||
@@ -100,3 +100,43 @@ jobs:
|
|||||||
helmfile-auto-init: "false"
|
helmfile-auto-init: "false"
|
||||||
helmfile-workdirectory: infra/formbricks-cloud-helm
|
helmfile-workdirectory: infra/formbricks-cloud-helm
|
||||||
|
|
||||||
|
- name: Purge Cloudflare Cache
|
||||||
|
if: ${{ inputs.ENVIRONMENT == 'prod' || inputs.ENVIRONMENT == 'stage' }}
|
||||||
|
env:
|
||||||
|
CF_ZONE_ID: ${{ secrets.CLOUDFLARE_ZONE_ID }}
|
||||||
|
CF_API_TOKEN: ${{ secrets.CLOUDFLARE_API_TOKEN }}
|
||||||
|
run: |
|
||||||
|
# Set hostname based on environment
|
||||||
|
if [[ "${{ inputs.ENVIRONMENT }}" == "prod" ]]; then
|
||||||
|
PURGE_HOST="app.formbricks.com"
|
||||||
|
else
|
||||||
|
PURGE_HOST="stage.app.formbricks.com"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Purging Cloudflare cache for host: $PURGE_HOST (environment: ${{ inputs.ENVIRONMENT }}, zone: $CF_ZONE_ID)"
|
||||||
|
|
||||||
|
# Prepare JSON payload for selective cache purge
|
||||||
|
json_payload=$(cat << EOF
|
||||||
|
{
|
||||||
|
"hosts": ["$PURGE_HOST"]
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
)
|
||||||
|
|
||||||
|
# Make API call to Cloudflare
|
||||||
|
response=$(curl -s -X POST \
|
||||||
|
"https://api.cloudflare.com/client/v4/zones/$CF_ZONE_ID/purge_cache" \
|
||||||
|
-H "Authorization: Bearer $CF_API_TOKEN" \
|
||||||
|
-H "Content-Type: application/json" \
|
||||||
|
--data "$json_payload")
|
||||||
|
|
||||||
|
echo "Cloudflare API response: $response"
|
||||||
|
|
||||||
|
# Verify the operation was successful
|
||||||
|
if [[ "$(echo "$response" | jq -r .success)" == "true" ]]; then
|
||||||
|
echo "✅ Successfully purged cache for $PURGE_HOST"
|
||||||
|
else
|
||||||
|
echo "❌ Cloudflare cache purge failed"
|
||||||
|
echo "Error details: $(echo "$response" | jq -r .errors)"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|||||||
@@ -73,3 +73,4 @@ infra/terraform/.terraform/
|
|||||||
/.idea/
|
/.idea/
|
||||||
/*.iml
|
/*.iml
|
||||||
packages/ios/FormbricksSDK/FormbricksSDK.xcodeproj/project.xcworkspace/xcuserdata
|
packages/ios/FormbricksSDK/FormbricksSDK.xcodeproj/project.xcworkspace/xcuserdata
|
||||||
|
.cursorrules
|
||||||
|
|||||||
+4
-4
@@ -27,7 +27,7 @@ describe("ConnectWithFormbricks", () => {
|
|||||||
render(
|
render(
|
||||||
<ConnectWithFormbricks
|
<ConnectWithFormbricks
|
||||||
environment={environment}
|
environment={environment}
|
||||||
webAppUrl={webAppUrl}
|
publicDomain={webAppUrl}
|
||||||
widgetSetupCompleted={false}
|
widgetSetupCompleted={false}
|
||||||
channel={channel}
|
channel={channel}
|
||||||
/>
|
/>
|
||||||
@@ -40,7 +40,7 @@ describe("ConnectWithFormbricks", () => {
|
|||||||
render(
|
render(
|
||||||
<ConnectWithFormbricks
|
<ConnectWithFormbricks
|
||||||
environment={environment}
|
environment={environment}
|
||||||
webAppUrl={webAppUrl}
|
publicDomain={webAppUrl}
|
||||||
widgetSetupCompleted={true}
|
widgetSetupCompleted={true}
|
||||||
channel={channel}
|
channel={channel}
|
||||||
/>
|
/>
|
||||||
@@ -53,7 +53,7 @@ describe("ConnectWithFormbricks", () => {
|
|||||||
render(
|
render(
|
||||||
<ConnectWithFormbricks
|
<ConnectWithFormbricks
|
||||||
environment={environment}
|
environment={environment}
|
||||||
webAppUrl={webAppUrl}
|
publicDomain={webAppUrl}
|
||||||
widgetSetupCompleted={true}
|
widgetSetupCompleted={true}
|
||||||
channel={channel}
|
channel={channel}
|
||||||
/>
|
/>
|
||||||
@@ -67,7 +67,7 @@ describe("ConnectWithFormbricks", () => {
|
|||||||
render(
|
render(
|
||||||
<ConnectWithFormbricks
|
<ConnectWithFormbricks
|
||||||
environment={environment}
|
environment={environment}
|
||||||
webAppUrl={webAppUrl}
|
publicDomain={webAppUrl}
|
||||||
widgetSetupCompleted={false}
|
widgetSetupCompleted={false}
|
||||||
channel={channel}
|
channel={channel}
|
||||||
/>
|
/>
|
||||||
|
|||||||
+3
-3
@@ -12,14 +12,14 @@ import { OnboardingSetupInstructions } from "./OnboardingSetupInstructions";
|
|||||||
|
|
||||||
interface ConnectWithFormbricksProps {
|
interface ConnectWithFormbricksProps {
|
||||||
environment: TEnvironment;
|
environment: TEnvironment;
|
||||||
webAppUrl: string;
|
publicDomain: string;
|
||||||
widgetSetupCompleted: boolean;
|
widgetSetupCompleted: boolean;
|
||||||
channel: TProjectConfigChannel;
|
channel: TProjectConfigChannel;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const ConnectWithFormbricks = ({
|
export const ConnectWithFormbricks = ({
|
||||||
environment,
|
environment,
|
||||||
webAppUrl,
|
publicDomain,
|
||||||
widgetSetupCompleted,
|
widgetSetupCompleted,
|
||||||
channel,
|
channel,
|
||||||
}: ConnectWithFormbricksProps) => {
|
}: ConnectWithFormbricksProps) => {
|
||||||
@@ -49,7 +49,7 @@ export const ConnectWithFormbricks = ({
|
|||||||
<div className="flex w-1/2 flex-col space-y-4">
|
<div className="flex w-1/2 flex-col space-y-4">
|
||||||
<OnboardingSetupInstructions
|
<OnboardingSetupInstructions
|
||||||
environmentId={environment.id}
|
environmentId={environment.id}
|
||||||
webAppUrl={webAppUrl}
|
publicDomain={publicDomain}
|
||||||
channel={channel}
|
channel={channel}
|
||||||
widgetSetupCompleted={widgetSetupCompleted}
|
widgetSetupCompleted={widgetSetupCompleted}
|
||||||
/>
|
/>
|
||||||
|
|||||||
+1
-1
@@ -33,7 +33,7 @@ describe("OnboardingSetupInstructions", () => {
|
|||||||
// Provide some default props for testing
|
// Provide some default props for testing
|
||||||
const defaultProps = {
|
const defaultProps = {
|
||||||
environmentId: "env-123",
|
environmentId: "env-123",
|
||||||
webAppUrl: "https://example.com",
|
publicDomain: "https://example.com",
|
||||||
channel: "app" as const, // Assuming channel is either "app" or "website"
|
channel: "app" as const, // Assuming channel is either "app" or "website"
|
||||||
widgetSetupCompleted: false,
|
widgetSetupCompleted: false,
|
||||||
};
|
};
|
||||||
|
|||||||
+6
-6
@@ -18,14 +18,14 @@ const tabs = [
|
|||||||
|
|
||||||
interface OnboardingSetupInstructionsProps {
|
interface OnboardingSetupInstructionsProps {
|
||||||
environmentId: string;
|
environmentId: string;
|
||||||
webAppUrl: string;
|
publicDomain: string;
|
||||||
channel: TProjectConfigChannel;
|
channel: TProjectConfigChannel;
|
||||||
widgetSetupCompleted: boolean;
|
widgetSetupCompleted: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const OnboardingSetupInstructions = ({
|
export const OnboardingSetupInstructions = ({
|
||||||
environmentId,
|
environmentId,
|
||||||
webAppUrl,
|
publicDomain,
|
||||||
channel,
|
channel,
|
||||||
widgetSetupCompleted,
|
widgetSetupCompleted,
|
||||||
}: OnboardingSetupInstructionsProps) => {
|
}: OnboardingSetupInstructionsProps) => {
|
||||||
@@ -34,7 +34,7 @@ export const OnboardingSetupInstructions = ({
|
|||||||
const htmlSnippetForAppSurveys = `<!-- START Formbricks Surveys -->
|
const htmlSnippetForAppSurveys = `<!-- START Formbricks Surveys -->
|
||||||
<script type="text/javascript">
|
<script type="text/javascript">
|
||||||
!function(){
|
!function(){
|
||||||
var appUrl = "${webAppUrl}";
|
var appUrl = "${publicDomain}";
|
||||||
var environmentId = "${environmentId}";
|
var environmentId = "${environmentId}";
|
||||||
var t=document.createElement("script");t.type="text/javascript",t.async=!0,t.src=appUrl+"/js/formbricks.umd.cjs",t.onload=function(){window.formbricks?window.formbricks.setup({environmentId:environmentId,appUrl:appUrl}):console.error("Formbricks library failed to load properly. The formbricks object is not available.");};var e=document.getElementsByTagName("script")[0];e.parentNode.insertBefore(t,e)}();
|
var t=document.createElement("script");t.type="text/javascript",t.async=!0,t.src=appUrl+"/js/formbricks.umd.cjs",t.onload=function(){window.formbricks?window.formbricks.setup({environmentId:environmentId,appUrl:appUrl}):console.error("Formbricks library failed to load properly. The formbricks object is not available.");};var e=document.getElementsByTagName("script")[0];e.parentNode.insertBefore(t,e)}();
|
||||||
</script>
|
</script>
|
||||||
@@ -44,7 +44,7 @@ export const OnboardingSetupInstructions = ({
|
|||||||
const htmlSnippetForWebsiteSurveys = `<!-- START Formbricks Surveys -->
|
const htmlSnippetForWebsiteSurveys = `<!-- START Formbricks Surveys -->
|
||||||
<script type="text/javascript">
|
<script type="text/javascript">
|
||||||
!function(){
|
!function(){
|
||||||
var appUrl = "${webAppUrl}";
|
var appUrl = "${publicDomain}";
|
||||||
var environmentId = "${environmentId}";
|
var environmentId = "${environmentId}";
|
||||||
var t=document.createElement("script");t.type="text/javascript",t.async=!0,t.src=appUrl+"/js/formbricks.umd.cjs",t.onload=function(){window.formbricks?window.formbricks.setup({environmentId:environmentId,appUrl:appUrl}):console.error("Formbricks library failed to load properly. The formbricks object is not available.");};var e=document.getElementsByTagName("script")[0];e.parentNode.insertBefore(t,e)}();
|
var t=document.createElement("script");t.type="text/javascript",t.async=!0,t.src=appUrl+"/js/formbricks.umd.cjs",t.onload=function(){window.formbricks?window.formbricks.setup({environmentId:environmentId,appUrl:appUrl}):console.error("Formbricks library failed to load properly. The formbricks object is not available.");};var e=document.getElementsByTagName("script")[0];e.parentNode.insertBefore(t,e)}();
|
||||||
</script>
|
</script>
|
||||||
@@ -57,7 +57,7 @@ export const OnboardingSetupInstructions = ({
|
|||||||
if (typeof window !== "undefined") {
|
if (typeof window !== "undefined") {
|
||||||
formbricks.setup({
|
formbricks.setup({
|
||||||
environmentId: "${environmentId}",
|
environmentId: "${environmentId}",
|
||||||
appUrl: "${webAppUrl}",
|
appUrl: "${publicDomain}",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -75,7 +75,7 @@ export const OnboardingSetupInstructions = ({
|
|||||||
if (typeof window !== "undefined") {
|
if (typeof window !== "undefined") {
|
||||||
formbricks.setup({
|
formbricks.setup({
|
||||||
environmentId: "${environmentId}",
|
environmentId: "${environmentId}",
|
||||||
appUrl: "${webAppUrl}",
|
appUrl: "${publicDomain}",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { ConnectWithFormbricks } from "@/app/(app)/(onboarding)/environments/[environmentId]/connect/components/ConnectWithFormbricks";
|
import { ConnectWithFormbricks } from "@/app/(app)/(onboarding)/environments/[environmentId]/connect/components/ConnectWithFormbricks";
|
||||||
import { WEBAPP_URL } from "@/lib/constants";
|
|
||||||
import { getEnvironment } from "@/lib/environment/service";
|
import { getEnvironment } from "@/lib/environment/service";
|
||||||
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
||||||
import { Button } from "@/modules/ui/components/button";
|
import { Button } from "@/modules/ui/components/button";
|
||||||
import { Header } from "@/modules/ui/components/header";
|
import { Header } from "@/modules/ui/components/header";
|
||||||
@@ -30,6 +30,8 @@ const Page = async (props: ConnectPageProps) => {
|
|||||||
|
|
||||||
const channel = project.config.channel || null;
|
const channel = project.config.channel || null;
|
||||||
|
|
||||||
|
const publicDomain = getPublicDomain();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex min-h-full flex-col items-center justify-center py-10">
|
<div className="flex min-h-full flex-col items-center justify-center py-10">
|
||||||
<Header title={t("environments.connect.headline")} subtitle={t("environments.connect.subtitle")} />
|
<Header title={t("environments.connect.headline")} subtitle={t("environments.connect.subtitle")} />
|
||||||
@@ -39,7 +41,7 @@ const Page = async (props: ConnectPageProps) => {
|
|||||||
</div>
|
</div>
|
||||||
<ConnectWithFormbricks
|
<ConnectWithFormbricks
|
||||||
environment={environment}
|
environment={environment}
|
||||||
webAppUrl={WEBAPP_URL}
|
publicDomain={publicDomain}
|
||||||
widgetSetupCompleted={environment.appSetupCompleted}
|
widgetSetupCompleted={environment.appSetupCompleted}
|
||||||
channel={channel}
|
channel={channel}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
IS_DEVELOPMENT: true,
|
IS_DEVELOPMENT: true,
|
||||||
E2E_TESTING: false,
|
E2E_TESTING: false,
|
||||||
WEBAPP_URL: "http://localhost:3000",
|
WEBAPP_URL: "http://localhost:3000",
|
||||||
SURVEY_URL: "http://localhost:3000/survey",
|
PUBLIC_URL: "http://localhost:3000/survey",
|
||||||
ENCRYPTION_KEY: "mock-encryption-key",
|
ENCRYPTION_KEY: "mock-encryption-key",
|
||||||
CRON_SECRET: "mock-cron-secret",
|
CRON_SECRET: "mock-cron-secret",
|
||||||
DEFAULT_BRAND_COLOR: "#64748b",
|
DEFAULT_BRAND_COLOR: "#64748b",
|
||||||
|
|||||||
+1
@@ -94,6 +94,7 @@ describe("LandingSidebar component", () => {
|
|||||||
organizationId: "o1",
|
organizationId: "o1",
|
||||||
redirect: true,
|
redirect: true,
|
||||||
callbackUrl: "/auth/login",
|
callbackUrl: "/auth/login",
|
||||||
|
clearEnvironmentId: true,
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
+1
@@ -130,6 +130,7 @@ export const LandingSidebar = ({
|
|||||||
organizationId: organization.id,
|
organizationId: organization.id,
|
||||||
redirect: true,
|
redirect: true,
|
||||||
callbackUrl: "/auth/login",
|
callbackUrl: "/auth/login",
|
||||||
|
clearEnvironmentId: true,
|
||||||
});
|
});
|
||||||
}}
|
}}
|
||||||
icon={<LogOutIcon className="mr-2 h-4 w-4" strokeWidth={1.5} />}>
|
icon={<LogOutIcon className="mr-2 h-4 w-4" strokeWidth={1.5} />}>
|
||||||
|
|||||||
+1
-1
@@ -14,7 +14,7 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
IS_DEVELOPMENT: true,
|
IS_DEVELOPMENT: true,
|
||||||
E2E_TESTING: false,
|
E2E_TESTING: false,
|
||||||
WEBAPP_URL: "http://localhost:3000",
|
WEBAPP_URL: "http://localhost:3000",
|
||||||
SURVEY_URL: "http://localhost:3000/survey",
|
PUBLIC_URL: "http://localhost:3000/survey",
|
||||||
ENCRYPTION_KEY: "mock-encryption-key",
|
ENCRYPTION_KEY: "mock-encryption-key",
|
||||||
CRON_SECRET: "mock-cron-secret",
|
CRON_SECRET: "mock-cron-secret",
|
||||||
DEFAULT_BRAND_COLOR: "#64748b",
|
DEFAULT_BRAND_COLOR: "#64748b",
|
||||||
|
|||||||
@@ -23,7 +23,6 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
IS_DEVELOPMENT: true,
|
IS_DEVELOPMENT: true,
|
||||||
E2E_TESTING: false,
|
E2E_TESTING: false,
|
||||||
WEBAPP_URL: "http://localhost:3000",
|
WEBAPP_URL: "http://localhost:3000",
|
||||||
SURVEY_URL: "http://localhost:3000/survey",
|
|
||||||
ENCRYPTION_KEY: "mock-encryption-key",
|
ENCRYPTION_KEY: "mock-encryption-key",
|
||||||
CRON_SECRET: "mock-cron-secret",
|
CRON_SECRET: "mock-cron-secret",
|
||||||
DEFAULT_BRAND_COLOR: "#64748b",
|
DEFAULT_BRAND_COLOR: "#64748b",
|
||||||
|
|||||||
+6
@@ -30,6 +30,12 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
REDIS_URL: "redis://localhost:6379",
|
REDIS_URL: "redis://localhost:6379",
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/env", () => ({
|
||||||
|
env: {
|
||||||
|
PUBLIC_URL: "https://public-domain.com",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
describe("Contact Page Re-export", () => {
|
describe("Contact Page Re-export", () => {
|
||||||
test("should re-export SingleContactPage", () => {
|
test("should re-export SingleContactPage", () => {
|
||||||
expect(Page).toBe(SingleContactPage);
|
expect(Page).toBe(SingleContactPage);
|
||||||
|
|||||||
+7
-8
@@ -11,22 +11,21 @@ export const ActionClassDataRow = ({
|
|||||||
locale: TUserLocale;
|
locale: TUserLocale;
|
||||||
}) => {
|
}) => {
|
||||||
return (
|
return (
|
||||||
<div className="m-2 grid h-16 grid-cols-6 content-center rounded-lg transition-colors ease-in-out hover:bg-slate-100">
|
<div className="m-2 grid grid-cols-6 content-center rounded-lg transition-colors ease-in-out hover:bg-slate-100">
|
||||||
<div className="col-span-4 flex items-center pl-6 text-sm">
|
<div className="col-span-4 flex items-start py-3 pl-6 text-sm">
|
||||||
<div className="flex items-center">
|
<div className="flex w-full items-center gap-4">
|
||||||
<div className="h-5 w-5 flex-shrink-0 text-slate-500">
|
<div className="mt-1 h-5 w-5 flex-shrink-0 text-slate-500">
|
||||||
{ACTION_TYPE_ICON_LOOKUP[actionClass.type]}
|
{ACTION_TYPE_ICON_LOOKUP[actionClass.type]}
|
||||||
</div>
|
</div>
|
||||||
<div className="ml-4 text-left">
|
<div className="text-left">
|
||||||
<div className="font-medium text-slate-900">{actionClass.name}</div>
|
<div className="break-words font-medium text-slate-900">{actionClass.name}</div>
|
||||||
<div className="text-xs text-slate-400">{actionClass.description}</div>
|
<div className="break-words text-xs text-slate-400">{actionClass.description}</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="col-span-2 my-auto whitespace-nowrap text-center text-sm text-slate-500">
|
<div className="col-span-2 my-auto whitespace-nowrap text-center text-sm text-slate-500">
|
||||||
{timeSince(actionClass.createdAt.toString(), locale)}
|
{timeSince(actionClass.createdAt.toString(), locale)}
|
||||||
</div>
|
</div>
|
||||||
<div className="text-center"></div>
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -220,6 +220,8 @@ describe("MainNavigation", () => {
|
|||||||
const mockSignOut = vi.fn().mockResolvedValue({ url: "/auth/login" });
|
const mockSignOut = vi.fn().mockResolvedValue({ url: "/auth/login" });
|
||||||
vi.mocked(useSignOut).mockReturnValue({ signOut: mockSignOut });
|
vi.mocked(useSignOut).mockReturnValue({ signOut: mockSignOut });
|
||||||
|
|
||||||
|
// Set up localStorage spy on the mocked localStorage
|
||||||
|
|
||||||
render(<MainNavigation {...defaultProps} />);
|
render(<MainNavigation {...defaultProps} />);
|
||||||
|
|
||||||
// Find the avatar and get its parent div which acts as the trigger
|
// Find the avatar and get its parent div which acts as the trigger
|
||||||
@@ -246,7 +248,9 @@ describe("MainNavigation", () => {
|
|||||||
organizationId: "org1",
|
organizationId: "org1",
|
||||||
redirect: false,
|
redirect: false,
|
||||||
callbackUrl: "/auth/login",
|
callbackUrl: "/auth/login",
|
||||||
|
clearEnvironmentId: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
await waitFor(() => {
|
await waitFor(() => {
|
||||||
expect(mockRouterPush).toHaveBeenCalledWith("/auth/login");
|
expect(mockRouterPush).toHaveBeenCalledWith("/auth/login");
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -396,6 +396,7 @@ export const MainNavigation = ({
|
|||||||
organizationId: organization.id,
|
organizationId: organization.id,
|
||||||
redirect: false,
|
redirect: false,
|
||||||
callbackUrl: "/auth/login",
|
callbackUrl: "/auth/login",
|
||||||
|
clearEnvironmentId: true,
|
||||||
});
|
});
|
||||||
router.push(route?.url || "/auth/login"); // NOSONAR // We want to check for empty strings
|
router.push(route?.url || "/auth/login"); // NOSONAR // We want to check for empty strings
|
||||||
}}
|
}}
|
||||||
|
|||||||
+6
@@ -29,6 +29,12 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
AUDIT_LOG_ENABLED: true,
|
AUDIT_LOG_ENABLED: true,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/env", () => ({
|
||||||
|
env: {
|
||||||
|
PUBLIC_URL: "https://example.com",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
describe("AppConnectionPage Re-export", () => {
|
describe("AppConnectionPage Re-export", () => {
|
||||||
test("should re-export AppConnectionPage correctly", () => {
|
test("should re-export AppConnectionPage correctly", () => {
|
||||||
expect(AppConnectionPage).toBe(OriginalAppConnectionPage);
|
expect(AppConnectionPage).toBe(OriginalAppConnectionPage);
|
||||||
|
|||||||
@@ -29,6 +29,12 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
AUDIT_LOG_ENABLED: 1,
|
AUDIT_LOG_ENABLED: 1,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/env", () => ({
|
||||||
|
env: {
|
||||||
|
PUBLIC_URL: "https://public-domain.com",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
describe("GeneralSettingsPage re-export", () => {
|
describe("GeneralSettingsPage re-export", () => {
|
||||||
test("should re-export GeneralSettingsPage component", () => {
|
test("should re-export GeneralSettingsPage component", () => {
|
||||||
expect(Page).toBe(GeneralSettingsPage);
|
expect(Page).toBe(GeneralSettingsPage);
|
||||||
|
|||||||
@@ -29,6 +29,12 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
AUDIT_LOG_ENABLED: 1,
|
AUDIT_LOG_ENABLED: 1,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/env", () => ({
|
||||||
|
env: {
|
||||||
|
PUBLIC_URL: "https://public-domain.com",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
describe("ProjectLookSettingsPage re-export", () => {
|
describe("ProjectLookSettingsPage re-export", () => {
|
||||||
test("should re-export ProjectLookSettingsPage component", () => {
|
test("should re-export ProjectLookSettingsPage component", () => {
|
||||||
expect(Page).toBe(ProjectLookSettingsPage);
|
expect(Page).toBe(ProjectLookSettingsPage);
|
||||||
|
|||||||
+201
-1
@@ -20,7 +20,7 @@ vi.mock("@/modules/ui/components/switch", () => ({
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
vi.mock("../actions", () => ({
|
vi.mock("../actions", () => ({
|
||||||
updateNotificationSettingsAction: vi.fn(() => Promise.resolve()),
|
updateNotificationSettingsAction: vi.fn(() => Promise.resolve({ data: true })),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
const surveyId = "survey1";
|
const surveyId = "survey1";
|
||||||
@@ -246,4 +246,204 @@ describe("NotificationSwitch", () => {
|
|||||||
});
|
});
|
||||||
expect(updateNotificationSettingsAction).not.toHaveBeenCalled();
|
expect(updateNotificationSettingsAction).not.toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("shows error toast when updateNotificationSettingsAction fails for 'alert' type", async () => {
|
||||||
|
const mockErrorResponse = { serverError: "Failed to update notification settings" };
|
||||||
|
vi.mocked(updateNotificationSettingsAction).mockResolvedValueOnce(mockErrorResponse);
|
||||||
|
|
||||||
|
const initialSettings = { ...baseNotificationSettings, alert: { [surveyId]: false } };
|
||||||
|
renderSwitch({ notificationSettings: initialSettings, notificationType: "alert" });
|
||||||
|
const switchInput = screen.getByLabelText("toggle notification settings for alert");
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
await user.click(switchInput);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updateNotificationSettingsAction).toHaveBeenCalledWith({
|
||||||
|
notificationSettings: { ...initialSettings, alert: { [surveyId]: true } },
|
||||||
|
});
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("Failed to update notification settings", {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
expect(toast.success).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("shows error toast when updateNotificationSettingsAction fails for 'weeklySummary' type", async () => {
|
||||||
|
const mockErrorResponse = { serverError: "Database connection failed" };
|
||||||
|
vi.mocked(updateNotificationSettingsAction).mockResolvedValueOnce(mockErrorResponse);
|
||||||
|
|
||||||
|
const initialSettings = { ...baseNotificationSettings, weeklySummary: { [projectId]: true } };
|
||||||
|
renderSwitch({
|
||||||
|
surveyOrProjectOrOrganizationId: projectId,
|
||||||
|
notificationSettings: initialSettings,
|
||||||
|
notificationType: "weeklySummary",
|
||||||
|
});
|
||||||
|
const switchInput = screen.getByLabelText("toggle notification settings for weeklySummary");
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
await user.click(switchInput);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updateNotificationSettingsAction).toHaveBeenCalledWith({
|
||||||
|
notificationSettings: { ...initialSettings, weeklySummary: { [projectId]: false } },
|
||||||
|
});
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("Database connection failed", {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
expect(toast.success).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("shows error toast when updateNotificationSettingsAction fails for 'unsubscribedOrganizationIds' type", async () => {
|
||||||
|
const mockErrorResponse = { serverError: "Permission denied" };
|
||||||
|
vi.mocked(updateNotificationSettingsAction).mockResolvedValueOnce(mockErrorResponse);
|
||||||
|
|
||||||
|
const initialSettings = { ...baseNotificationSettings, unsubscribedOrganizationIds: [] };
|
||||||
|
renderSwitch({
|
||||||
|
surveyOrProjectOrOrganizationId: organizationId,
|
||||||
|
notificationSettings: initialSettings,
|
||||||
|
notificationType: "unsubscribedOrganizationIds",
|
||||||
|
});
|
||||||
|
const switchInput = screen.getByLabelText("toggle notification settings for unsubscribedOrganizationIds");
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
await user.click(switchInput);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updateNotificationSettingsAction).toHaveBeenCalledWith({
|
||||||
|
notificationSettings: { ...initialSettings, unsubscribedOrganizationIds: [organizationId] },
|
||||||
|
});
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("Permission denied", {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
expect(toast.success).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("shows error toast when updateNotificationSettingsAction returns null", async () => {
|
||||||
|
const mockErrorResponse = { serverError: "An error occurred" };
|
||||||
|
vi.mocked(updateNotificationSettingsAction).mockResolvedValueOnce(mockErrorResponse);
|
||||||
|
|
||||||
|
const initialSettings = { ...baseNotificationSettings, alert: { [surveyId]: false } };
|
||||||
|
renderSwitch({ notificationSettings: initialSettings, notificationType: "alert" });
|
||||||
|
const switchInput = screen.getByLabelText("toggle notification settings for alert");
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
await user.click(switchInput);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updateNotificationSettingsAction).toHaveBeenCalledWith({
|
||||||
|
notificationSettings: { ...initialSettings, alert: { [surveyId]: true } },
|
||||||
|
});
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("An error occurred", {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
expect(toast.success).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("shows error toast when updateNotificationSettingsAction returns undefined", async () => {
|
||||||
|
const mockErrorResponse = { serverError: "An error occurred" };
|
||||||
|
vi.mocked(updateNotificationSettingsAction).mockResolvedValueOnce(mockErrorResponse);
|
||||||
|
|
||||||
|
const initialSettings = { ...baseNotificationSettings, alert: { [surveyId]: false } };
|
||||||
|
renderSwitch({ notificationSettings: initialSettings, notificationType: "alert" });
|
||||||
|
const switchInput = screen.getByLabelText("toggle notification settings for alert");
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
await user.click(switchInput);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updateNotificationSettingsAction).toHaveBeenCalledWith({
|
||||||
|
notificationSettings: { ...initialSettings, alert: { [surveyId]: true } },
|
||||||
|
});
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("An error occurred", {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
expect(toast.success).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("shows error toast when updateNotificationSettingsAction returns response without data property", async () => {
|
||||||
|
const mockErrorResponse = { validationErrors: { _errors: ["Invalid input"] } };
|
||||||
|
vi.mocked(updateNotificationSettingsAction).mockResolvedValueOnce(mockErrorResponse);
|
||||||
|
|
||||||
|
const initialSettings = { ...baseNotificationSettings, alert: { [surveyId]: false } };
|
||||||
|
renderSwitch({ notificationSettings: initialSettings, notificationType: "alert" });
|
||||||
|
const switchInput = screen.getByLabelText("toggle notification settings for alert");
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
await user.click(switchInput);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updateNotificationSettingsAction).toHaveBeenCalledWith({
|
||||||
|
notificationSettings: { ...initialSettings, alert: { [surveyId]: true } },
|
||||||
|
});
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("Invalid input", {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
expect(toast.success).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("shows error toast when updateNotificationSettingsAction throws an exception", async () => {
|
||||||
|
const mockErrorResponse = { serverError: "Network error" };
|
||||||
|
vi.mocked(updateNotificationSettingsAction).mockResolvedValueOnce(mockErrorResponse);
|
||||||
|
|
||||||
|
const initialSettings = { ...baseNotificationSettings, alert: { [surveyId]: false } };
|
||||||
|
renderSwitch({ notificationSettings: initialSettings, notificationType: "alert" });
|
||||||
|
const switchInput = screen.getByLabelText("toggle notification settings for alert");
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
await user.click(switchInput);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updateNotificationSettingsAction).toHaveBeenCalledWith({
|
||||||
|
notificationSettings: { ...initialSettings, alert: { [surveyId]: true } },
|
||||||
|
});
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("Network error", {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
expect(toast.success).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("switch remains enabled after error occurs", async () => {
|
||||||
|
const mockErrorResponse = { serverError: "Failed to update" };
|
||||||
|
vi.mocked(updateNotificationSettingsAction).mockResolvedValueOnce(mockErrorResponse);
|
||||||
|
|
||||||
|
const initialSettings = { ...baseNotificationSettings, alert: { [surveyId]: false } };
|
||||||
|
renderSwitch({ notificationSettings: initialSettings, notificationType: "alert" });
|
||||||
|
const switchInput = screen.getByLabelText("toggle notification settings for alert");
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
await user.click(switchInput);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("Failed to update", {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
expect(switchInput).toBeEnabled(); // Switch should be re-enabled after error
|
||||||
|
});
|
||||||
|
|
||||||
|
test("shows error toast with validation errors for specific fields", async () => {
|
||||||
|
const mockErrorResponse = {
|
||||||
|
validationErrors: {
|
||||||
|
notificationSettings: {
|
||||||
|
_errors: ["Invalid notification settings"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
vi.mocked(updateNotificationSettingsAction).mockResolvedValueOnce(mockErrorResponse);
|
||||||
|
|
||||||
|
const initialSettings = { ...baseNotificationSettings, alert: { [surveyId]: false } };
|
||||||
|
renderSwitch({ notificationSettings: initialSettings, notificationType: "alert" });
|
||||||
|
const switchInput = screen.getByLabelText("toggle notification settings for alert");
|
||||||
|
|
||||||
|
await act(async () => {
|
||||||
|
await user.click(switchInput);
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(updateNotificationSettingsAction).toHaveBeenCalledWith({
|
||||||
|
notificationSettings: { ...initialSettings, alert: { [surveyId]: true } },
|
||||||
|
});
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("notificationSettingsInvalid notification settings", {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
expect(toast.success).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
+17
-4
@@ -1,7 +1,9 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import { getFormattedErrorMessage } from "@/lib/utils/helper";
|
||||||
import { Switch } from "@/modules/ui/components/switch";
|
import { Switch } from "@/modules/ui/components/switch";
|
||||||
import { useTranslate } from "@tolgee/react";
|
import { useTranslate } from "@tolgee/react";
|
||||||
|
import { useRouter } from "next/navigation";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import toast from "react-hot-toast";
|
import toast from "react-hot-toast";
|
||||||
import { TUserNotificationSettings } from "@formbricks/types/user";
|
import { TUserNotificationSettings } from "@formbricks/types/user";
|
||||||
@@ -24,6 +26,7 @@ export const NotificationSwitch = ({
|
|||||||
}: NotificationSwitchProps) => {
|
}: NotificationSwitchProps) => {
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const { t } = useTranslate();
|
const { t } = useTranslate();
|
||||||
|
const router = useRouter();
|
||||||
const isChecked =
|
const isChecked =
|
||||||
notificationType === "unsubscribedOrganizationIds"
|
notificationType === "unsubscribedOrganizationIds"
|
||||||
? !notificationSettings.unsubscribedOrganizationIds?.includes(surveyOrProjectOrOrganizationId)
|
? !notificationSettings.unsubscribedOrganizationIds?.includes(surveyOrProjectOrOrganizationId)
|
||||||
@@ -50,7 +53,20 @@ export const NotificationSwitch = ({
|
|||||||
!updatedNotificationSettings[notificationType][surveyOrProjectOrOrganizationId];
|
!updatedNotificationSettings[notificationType][surveyOrProjectOrOrganizationId];
|
||||||
}
|
}
|
||||||
|
|
||||||
await updateNotificationSettingsAction({ notificationSettings: updatedNotificationSettings });
|
const updatedNotificationSettingsActionResponse = await updateNotificationSettingsAction({
|
||||||
|
notificationSettings: updatedNotificationSettings,
|
||||||
|
});
|
||||||
|
if (updatedNotificationSettingsActionResponse?.data) {
|
||||||
|
toast.success(t("environments.settings.notifications.notification_settings_updated"), {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
router.refresh();
|
||||||
|
} else {
|
||||||
|
const errorMessage = getFormattedErrorMessage(updatedNotificationSettingsActionResponse);
|
||||||
|
toast.error(errorMessage, {
|
||||||
|
id: "notification-switch",
|
||||||
|
});
|
||||||
|
}
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -104,9 +120,6 @@ export const NotificationSwitch = ({
|
|||||||
disabled={isLoading}
|
disabled={isLoading}
|
||||||
onCheckedChange={async () => {
|
onCheckedChange={async () => {
|
||||||
await handleSwitchChange();
|
await handleSwitchChange();
|
||||||
toast.success(t("environments.settings.notifications.notification_settings_updated"), {
|
|
||||||
id: "notification-switch",
|
|
||||||
});
|
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
|||||||
+19
-1
@@ -13,7 +13,7 @@ import { AuthenticatedActionClientCtx } from "@/lib/utils/action-client/types/co
|
|||||||
import { rateLimit } from "@/lib/utils/rate-limit";
|
import { rateLimit } from "@/lib/utils/rate-limit";
|
||||||
import { updateBrevoCustomer } from "@/modules/auth/lib/brevo";
|
import { updateBrevoCustomer } from "@/modules/auth/lib/brevo";
|
||||||
import { withAuditLogging } from "@/modules/ee/audit-logs/lib/handler";
|
import { withAuditLogging } from "@/modules/ee/audit-logs/lib/handler";
|
||||||
import { sendVerificationNewEmail } from "@/modules/email";
|
import { sendForgotPasswordEmail, sendVerificationNewEmail } from "@/modules/email";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { ZId } from "@formbricks/types/common";
|
import { ZId } from "@formbricks/types/common";
|
||||||
import {
|
import {
|
||||||
@@ -162,3 +162,21 @@ export const removeAvatarAction = authenticatedActionClient.schema(ZRemoveAvatar
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
export const resetPasswordAction = authenticatedActionClient.action(
|
||||||
|
withAuditLogging(
|
||||||
|
"passwordReset",
|
||||||
|
"user",
|
||||||
|
async ({ ctx }: { ctx: AuthenticatedActionClientCtx; parsedInput: undefined }) => {
|
||||||
|
if (ctx.user.identityProvider !== "email") {
|
||||||
|
throw new OperationNotAllowedError("auth.reset-password.not-allowed");
|
||||||
|
}
|
||||||
|
|
||||||
|
await sendForgotPasswordEmail(ctx.user);
|
||||||
|
|
||||||
|
ctx.auditLoggingCtx.userId = ctx.user.id;
|
||||||
|
|
||||||
|
return { success: true };
|
||||||
|
}
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|||||||
+93
-4
@@ -3,7 +3,7 @@ import userEvent from "@testing-library/user-event";
|
|||||||
import toast from "react-hot-toast";
|
import toast from "react-hot-toast";
|
||||||
import { afterEach, beforeEach, describe, expect, test, vi } from "vitest";
|
import { afterEach, beforeEach, describe, expect, test, vi } from "vitest";
|
||||||
import { TUser } from "@formbricks/types/user";
|
import { TUser } from "@formbricks/types/user";
|
||||||
import { updateUserAction } from "../actions";
|
import { resetPasswordAction, updateUserAction } from "../actions";
|
||||||
import { EditProfileDetailsForm } from "./EditProfileDetailsForm";
|
import { EditProfileDetailsForm } from "./EditProfileDetailsForm";
|
||||||
|
|
||||||
const mockUser = {
|
const mockUser = {
|
||||||
@@ -24,6 +24,8 @@ const mockUser = {
|
|||||||
objective: "other",
|
objective: "other",
|
||||||
} as unknown as TUser;
|
} as unknown as TUser;
|
||||||
|
|
||||||
|
vi.mock("next-auth/react", () => ({ signOut: vi.fn() }));
|
||||||
|
|
||||||
// Mock window.location.reload
|
// Mock window.location.reload
|
||||||
const originalLocation = window.location;
|
const originalLocation = window.location;
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
@@ -35,6 +37,11 @@ beforeEach(() => {
|
|||||||
|
|
||||||
vi.mock("@/app/(app)/environments/[environmentId]/settings/(account)/profile/actions", () => ({
|
vi.mock("@/app/(app)/environments/[environmentId]/settings/(account)/profile/actions", () => ({
|
||||||
updateUserAction: vi.fn(),
|
updateUserAction: vi.fn(),
|
||||||
|
resetPasswordAction: vi.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/modules/auth/forgot-password/actions", () => ({
|
||||||
|
forgotPasswordAction: vi.fn(),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
@@ -50,7 +57,13 @@ describe("EditProfileDetailsForm", () => {
|
|||||||
test("renders with initial user data and updates successfully", async () => {
|
test("renders with initial user data and updates successfully", async () => {
|
||||||
vi.mocked(updateUserAction).mockResolvedValue({ ...mockUser, name: "New Name" } as any);
|
vi.mocked(updateUserAction).mockResolvedValue({ ...mockUser, name: "New Name" } as any);
|
||||||
|
|
||||||
render(<EditProfileDetailsForm user={mockUser} emailVerificationDisabled={true} />);
|
render(
|
||||||
|
<EditProfileDetailsForm
|
||||||
|
user={mockUser}
|
||||||
|
emailVerificationDisabled={true}
|
||||||
|
isPasswordResetEnabled={false}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
const nameInput = screen.getByPlaceholderText("common.full_name");
|
const nameInput = screen.getByPlaceholderText("common.full_name");
|
||||||
expect(nameInput).toHaveValue(mockUser.name);
|
expect(nameInput).toHaveValue(mockUser.name);
|
||||||
@@ -91,7 +104,13 @@ describe("EditProfileDetailsForm", () => {
|
|||||||
const errorMessage = "Update failed";
|
const errorMessage = "Update failed";
|
||||||
vi.mocked(updateUserAction).mockRejectedValue(new Error(errorMessage));
|
vi.mocked(updateUserAction).mockRejectedValue(new Error(errorMessage));
|
||||||
|
|
||||||
render(<EditProfileDetailsForm user={mockUser} emailVerificationDisabled={false} />);
|
render(
|
||||||
|
<EditProfileDetailsForm
|
||||||
|
user={mockUser}
|
||||||
|
emailVerificationDisabled={false}
|
||||||
|
isPasswordResetEnabled={false}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
const nameInput = screen.getByPlaceholderText("common.full_name");
|
const nameInput = screen.getByPlaceholderText("common.full_name");
|
||||||
await userEvent.clear(nameInput);
|
await userEvent.clear(nameInput);
|
||||||
@@ -109,7 +128,13 @@ describe("EditProfileDetailsForm", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test("update button is disabled initially and enables on change", async () => {
|
test("update button is disabled initially and enables on change", async () => {
|
||||||
render(<EditProfileDetailsForm user={mockUser} emailVerificationDisabled={false} />);
|
render(
|
||||||
|
<EditProfileDetailsForm
|
||||||
|
user={mockUser}
|
||||||
|
emailVerificationDisabled={false}
|
||||||
|
isPasswordResetEnabled={false}
|
||||||
|
/>
|
||||||
|
);
|
||||||
const updateButton = screen.getByText("common.update");
|
const updateButton = screen.getByText("common.update");
|
||||||
expect(updateButton).toBeDisabled();
|
expect(updateButton).toBeDisabled();
|
||||||
|
|
||||||
@@ -117,4 +142,68 @@ describe("EditProfileDetailsForm", () => {
|
|||||||
await userEvent.type(nameInput, " updated");
|
await userEvent.type(nameInput, " updated");
|
||||||
expect(updateButton).toBeEnabled();
|
expect(updateButton).toBeEnabled();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("reset password button works", async () => {
|
||||||
|
vi.mocked(resetPasswordAction).mockResolvedValue({ data: { success: true } });
|
||||||
|
|
||||||
|
render(
|
||||||
|
<EditProfileDetailsForm
|
||||||
|
user={mockUser}
|
||||||
|
emailVerificationDisabled={false}
|
||||||
|
isPasswordResetEnabled={true}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
const resetButton = screen.getByRole("button", { name: "auth.forgot-password.reset_password" });
|
||||||
|
await userEvent.click(resetButton);
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(resetPasswordAction).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(toast.success).toHaveBeenCalledWith("auth.forgot-password.email-sent.heading");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("reset password button handles error correctly", async () => {
|
||||||
|
const errorMessage = "Reset failed";
|
||||||
|
vi.mocked(resetPasswordAction).mockResolvedValue({ serverError: errorMessage });
|
||||||
|
|
||||||
|
render(
|
||||||
|
<EditProfileDetailsForm
|
||||||
|
user={mockUser}
|
||||||
|
emailVerificationDisabled={false}
|
||||||
|
isPasswordResetEnabled={true}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
const resetButton = screen.getByRole("button", { name: "auth.forgot-password.reset_password" });
|
||||||
|
await userEvent.click(resetButton);
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(resetPasswordAction).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(toast.error).toHaveBeenCalledWith(errorMessage);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("reset password button shows loading state", async () => {
|
||||||
|
vi.mocked(resetPasswordAction).mockImplementation(() => new Promise(() => {})); // Never resolves
|
||||||
|
|
||||||
|
render(
|
||||||
|
<EditProfileDetailsForm
|
||||||
|
user={mockUser}
|
||||||
|
emailVerificationDisabled={false}
|
||||||
|
isPasswordResetEnabled={true}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
const resetButton = screen.getByRole("button", { name: "auth.forgot-password.reset_password" });
|
||||||
|
await userEvent.click(resetButton);
|
||||||
|
|
||||||
|
expect(resetButton).toBeDisabled();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
+55
-5
@@ -14,6 +14,7 @@ import {
|
|||||||
} from "@/modules/ui/components/dropdown-menu";
|
} from "@/modules/ui/components/dropdown-menu";
|
||||||
import { FormControl, FormError, FormField, FormItem, FormLabel } from "@/modules/ui/components/form";
|
import { FormControl, FormError, FormField, FormItem, FormLabel } from "@/modules/ui/components/form";
|
||||||
import { Input } from "@/modules/ui/components/input";
|
import { Input } from "@/modules/ui/components/input";
|
||||||
|
import { Label } from "@/modules/ui/components/label";
|
||||||
import { zodResolver } from "@hookform/resolvers/zod";
|
import { zodResolver } from "@hookform/resolvers/zod";
|
||||||
import { useTranslate } from "@tolgee/react";
|
import { useTranslate } from "@tolgee/react";
|
||||||
import { ChevronDownIcon } from "lucide-react";
|
import { ChevronDownIcon } from "lucide-react";
|
||||||
@@ -22,7 +23,7 @@ import { FormProvider, SubmitHandler, useForm } from "react-hook-form";
|
|||||||
import toast from "react-hot-toast";
|
import toast from "react-hot-toast";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
import { TUser, TUserUpdateInput, ZUser, ZUserEmail } from "@formbricks/types/user";
|
import { TUser, TUserUpdateInput, ZUser, ZUserEmail } from "@formbricks/types/user";
|
||||||
import { updateUserAction } from "../actions";
|
import { resetPasswordAction, updateUserAction } from "../actions";
|
||||||
|
|
||||||
// Schema & types
|
// Schema & types
|
||||||
const ZEditProfileNameFormSchema = ZUser.pick({ name: true, locale: true, email: true }).extend({
|
const ZEditProfileNameFormSchema = ZUser.pick({ name: true, locale: true, email: true }).extend({
|
||||||
@@ -30,13 +31,17 @@ const ZEditProfileNameFormSchema = ZUser.pick({ name: true, locale: true, email:
|
|||||||
});
|
});
|
||||||
type TEditProfileNameForm = z.infer<typeof ZEditProfileNameFormSchema>;
|
type TEditProfileNameForm = z.infer<typeof ZEditProfileNameFormSchema>;
|
||||||
|
|
||||||
|
interface IEditProfileDetailsFormProps {
|
||||||
|
user: TUser;
|
||||||
|
isPasswordResetEnabled?: boolean;
|
||||||
|
emailVerificationDisabled: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
export const EditProfileDetailsForm = ({
|
export const EditProfileDetailsForm = ({
|
||||||
user,
|
user,
|
||||||
|
isPasswordResetEnabled,
|
||||||
emailVerificationDisabled,
|
emailVerificationDisabled,
|
||||||
}: {
|
}: IEditProfileDetailsFormProps) => {
|
||||||
user: TUser;
|
|
||||||
emailVerificationDisabled: boolean;
|
|
||||||
}) => {
|
|
||||||
const { t } = useTranslate();
|
const { t } = useTranslate();
|
||||||
|
|
||||||
const form = useForm<TEditProfileNameForm>({
|
const form = useForm<TEditProfileNameForm>({
|
||||||
@@ -50,6 +55,8 @@ export const EditProfileDetailsForm = ({
|
|||||||
});
|
});
|
||||||
|
|
||||||
const { isSubmitting, isDirty } = form.formState;
|
const { isSubmitting, isDirty } = form.formState;
|
||||||
|
|
||||||
|
const [isResettingPassword, setIsResettingPassword] = useState(false);
|
||||||
const [showModal, setShowModal] = useState(false);
|
const [showModal, setShowModal] = useState(false);
|
||||||
const { signOut: signOutWithAudit } = useSignOut({ id: user.id, email: user.email });
|
const { signOut: signOutWithAudit } = useSignOut({ id: user.id, email: user.email });
|
||||||
|
|
||||||
@@ -90,6 +97,7 @@ export const EditProfileDetailsForm = ({
|
|||||||
redirectUrl: "/email-change-without-verification-success",
|
redirectUrl: "/email-change-without-verification-success",
|
||||||
redirect: true,
|
redirect: true,
|
||||||
callbackUrl: "/email-change-without-verification-success",
|
callbackUrl: "/email-change-without-verification-success",
|
||||||
|
clearEnvironmentId: true,
|
||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -121,6 +129,28 @@ export const EditProfileDetailsForm = ({
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleResetPassword = async () => {
|
||||||
|
setIsResettingPassword(true);
|
||||||
|
|
||||||
|
const result = await resetPasswordAction();
|
||||||
|
if (result?.data) {
|
||||||
|
toast.success(t("auth.forgot-password.email-sent.heading"));
|
||||||
|
|
||||||
|
await signOutWithAudit({
|
||||||
|
reason: "password_reset",
|
||||||
|
redirectUrl: "/auth/login",
|
||||||
|
redirect: true,
|
||||||
|
callbackUrl: "/auth/login",
|
||||||
|
clearEnvironmentId: true,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
const errorMessage = getFormattedErrorMessage(result);
|
||||||
|
toast.error(t(errorMessage));
|
||||||
|
}
|
||||||
|
|
||||||
|
setIsResettingPassword(false);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<FormProvider {...form}>
|
<FormProvider {...form}>
|
||||||
@@ -205,6 +235,26 @@ export const EditProfileDetailsForm = ({
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
{isPasswordResetEnabled && (
|
||||||
|
<div className="mt-4 space-y-2">
|
||||||
|
<Label htmlFor="reset-password">{t("auth.forgot-password.reset_password")}</Label>
|
||||||
|
<p className="mt-1 text-sm text-slate-500">
|
||||||
|
{t("auth.forgot-password.reset_password_description")}
|
||||||
|
</p>
|
||||||
|
<div className="flex items-center justify-between gap-2">
|
||||||
|
<Input type="email" id="reset-password" defaultValue={user.email} disabled />
|
||||||
|
<Button
|
||||||
|
onClick={handleResetPassword}
|
||||||
|
loading={isResettingPassword}
|
||||||
|
disabled={isResettingPassword}
|
||||||
|
size="default"
|
||||||
|
variant="secondary">
|
||||||
|
{t("auth.forgot-password.reset_password")}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
<Button
|
<Button
|
||||||
type="submit"
|
type="submit"
|
||||||
className="mt-4"
|
className="mt-4"
|
||||||
|
|||||||
+2
-1
@@ -12,7 +12,8 @@ import Page from "./page";
|
|||||||
|
|
||||||
// Mock services and utils
|
// Mock services and utils
|
||||||
vi.mock("@/lib/constants", () => ({
|
vi.mock("@/lib/constants", () => ({
|
||||||
IS_FORMBRICKS_CLOUD: true,
|
IS_FORMBRICKS_CLOUD: 1,
|
||||||
|
PASSWORD_RESET_DISABLED: 1,
|
||||||
EMAIL_VERIFICATION_DISABLED: true,
|
EMAIL_VERIFICATION_DISABLED: true,
|
||||||
}));
|
}));
|
||||||
vi.mock("@/lib/organization/service", () => ({
|
vi.mock("@/lib/organization/service", () => ({
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { AccountSettingsNavbar } from "@/app/(app)/environments/[environmentId]/settings/(account)/components/AccountSettingsNavbar";
|
import { AccountSettingsNavbar } from "@/app/(app)/environments/[environmentId]/settings/(account)/components/AccountSettingsNavbar";
|
||||||
import { AccountSecurity } from "@/app/(app)/environments/[environmentId]/settings/(account)/profile/components/AccountSecurity";
|
import { AccountSecurity } from "@/app/(app)/environments/[environmentId]/settings/(account)/profile/components/AccountSecurity";
|
||||||
import { EMAIL_VERIFICATION_DISABLED, IS_FORMBRICKS_CLOUD } from "@/lib/constants";
|
import { EMAIL_VERIFICATION_DISABLED, IS_FORMBRICKS_CLOUD, PASSWORD_RESET_DISABLED } from "@/lib/constants";
|
||||||
import { getOrganizationsWhereUserIsSingleOwner } from "@/lib/organization/service";
|
import { getOrganizationsWhereUserIsSingleOwner } from "@/lib/organization/service";
|
||||||
import { getUser } from "@/lib/user/service";
|
import { getUser } from "@/lib/user/service";
|
||||||
import { getIsMultiOrgEnabled, getIsTwoFactorAuthEnabled } from "@/modules/ee/license-check/lib/utils";
|
import { getIsMultiOrgEnabled, getIsTwoFactorAuthEnabled } from "@/modules/ee/license-check/lib/utils";
|
||||||
@@ -32,6 +32,8 @@ const Page = async (props: { params: Promise<{ environmentId: string }> }) => {
|
|||||||
throw new Error(t("common.user_not_found"));
|
throw new Error(t("common.user_not_found"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const isPasswordResetEnabled = !PASSWORD_RESET_DISABLED && user.identityProvider === "email";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<PageContentWrapper>
|
<PageContentWrapper>
|
||||||
<PageHeader pageTitle={t("common.account_settings")}>
|
<PageHeader pageTitle={t("common.account_settings")}>
|
||||||
@@ -42,7 +44,11 @@ const Page = async (props: { params: Promise<{ environmentId: string }> }) => {
|
|||||||
<SettingsCard
|
<SettingsCard
|
||||||
title={t("environments.settings.profile.personal_information")}
|
title={t("environments.settings.profile.personal_information")}
|
||||||
description={t("environments.settings.profile.update_personal_info")}>
|
description={t("environments.settings.profile.update_personal_info")}>
|
||||||
<EditProfileDetailsForm emailVerificationDisabled={EMAIL_VERIFICATION_DISABLED} user={user} />
|
<EditProfileDetailsForm
|
||||||
|
user={user}
|
||||||
|
emailVerificationDisabled={EMAIL_VERIFICATION_DISABLED}
|
||||||
|
isPasswordResetEnabled={isPasswordResetEnabled}
|
||||||
|
/>
|
||||||
</SettingsCard>
|
</SettingsCard>
|
||||||
<SettingsCard
|
<SettingsCard
|
||||||
title={t("common.avatar")}
|
title={t("common.avatar")}
|
||||||
|
|||||||
+6
@@ -34,6 +34,12 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
AUDIT_LOG_ENABLED: 1,
|
AUDIT_LOG_ENABLED: 1,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/env", () => ({
|
||||||
|
env: {
|
||||||
|
PUBLIC_URL: "https://public-domain.com",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
describe("TeamsPage re-export", () => {
|
describe("TeamsPage re-export", () => {
|
||||||
test("should re-export TeamsPage component", () => {
|
test("should re-export TeamsPage component", () => {
|
||||||
expect(Page).toBe(TeamsPage);
|
expect(Page).toBe(TeamsPage);
|
||||||
|
|||||||
+6
@@ -49,6 +49,12 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
AUDIT_LOG_ENABLED: true,
|
AUDIT_LOG_ENABLED: true,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/env", () => ({
|
||||||
|
env: {
|
||||||
|
PUBLIC_URL: "https://public-domain.com",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
vi.mock("@/app/(app)/environments/[environmentId]/components/ResponseFilterContext");
|
vi.mock("@/app/(app)/environments/[environmentId]/components/ResponseFilterContext");
|
||||||
vi.mock("@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/actions");
|
vi.mock("@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/actions");
|
||||||
vi.mock("@/app/lib/surveys/surveys");
|
vi.mock("@/app/lib/surveys/surveys");
|
||||||
|
|||||||
+3
-3
@@ -20,7 +20,7 @@ interface ResponsePageProps {
|
|||||||
environment: TEnvironment;
|
environment: TEnvironment;
|
||||||
survey: TSurvey;
|
survey: TSurvey;
|
||||||
surveyId: string;
|
surveyId: string;
|
||||||
webAppUrl: string;
|
publicDomain: string;
|
||||||
user?: TUser;
|
user?: TUser;
|
||||||
environmentTags: TTag[];
|
environmentTags: TTag[];
|
||||||
responsesPerPage: number;
|
responsesPerPage: number;
|
||||||
@@ -32,7 +32,7 @@ export const ResponsePage = ({
|
|||||||
environment,
|
environment,
|
||||||
survey,
|
survey,
|
||||||
surveyId,
|
surveyId,
|
||||||
webAppUrl,
|
publicDomain,
|
||||||
user,
|
user,
|
||||||
environmentTags,
|
environmentTags,
|
||||||
responsesPerPage,
|
responsesPerPage,
|
||||||
@@ -155,7 +155,7 @@ export const ResponsePage = ({
|
|||||||
<>
|
<>
|
||||||
<div className="flex gap-1.5">
|
<div className="flex gap-1.5">
|
||||||
<CustomFilter survey={surveyMemoized} />
|
<CustomFilter survey={surveyMemoized} />
|
||||||
{!isReadOnly && !isSharingPage && <ResultsShareButton survey={survey} webAppUrl={webAppUrl} />}
|
{!isReadOnly && !isSharingPage && <ResultsShareButton survey={survey} publicDomain={publicDomain} />}
|
||||||
</div>
|
</div>
|
||||||
<ResponseDataView
|
<ResponseDataView
|
||||||
survey={survey}
|
survey={survey}
|
||||||
|
|||||||
+7
-7
@@ -3,7 +3,7 @@ import { SurveyAnalysisNavigation } from "@/app/(app)/environments/[environmentI
|
|||||||
import { ResponsePage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/responses/components/ResponsePage";
|
import { ResponsePage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/responses/components/ResponsePage";
|
||||||
import Page from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/responses/page";
|
import Page from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/responses/page";
|
||||||
import { SurveyAnalysisCTA } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SurveyAnalysisCTA";
|
import { SurveyAnalysisCTA } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SurveyAnalysisCTA";
|
||||||
import { getSurveyDomain } from "@/lib/getSurveyUrl";
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getResponseCountBySurveyId } from "@/lib/response/service";
|
import { getResponseCountBySurveyId } from "@/lib/response/service";
|
||||||
import { getSurvey } from "@/lib/survey/service";
|
import { getSurvey } from "@/lib/survey/service";
|
||||||
import { getTagsByEnvironmentId } from "@/lib/tag/service";
|
import { getTagsByEnvironmentId } from "@/lib/tag/service";
|
||||||
@@ -65,8 +65,8 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
SESSION_MAX_AGE: 1000,
|
SESSION_MAX_AGE: 1000,
|
||||||
}));
|
}));
|
||||||
|
|
||||||
vi.mock("@/lib/getSurveyUrl", () => ({
|
vi.mock("@/lib/getPublicUrl", () => ({
|
||||||
getSurveyDomain: vi.fn(),
|
getPublicDomain: vi.fn(),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
vi.mock("@/lib/response/service", () => ({
|
vi.mock("@/lib/response/service", () => ({
|
||||||
@@ -160,7 +160,7 @@ const mockEnvironment = {
|
|||||||
|
|
||||||
const mockTags: TTag[] = [{ id: "tag1", name: "Tag 1", environmentId: mockEnvironmentId } as unknown as TTag];
|
const mockTags: TTag[] = [{ id: "tag1", name: "Tag 1", environmentId: mockEnvironmentId } as unknown as TTag];
|
||||||
const mockLocale: TUserLocale = "en-US";
|
const mockLocale: TUserLocale = "en-US";
|
||||||
const mockSurveyDomain = "http://customdomain.com";
|
const mockPublicDomain = "http://customdomain.com";
|
||||||
|
|
||||||
const mockParams = {
|
const mockParams = {
|
||||||
environmentId: mockEnvironmentId,
|
environmentId: mockEnvironmentId,
|
||||||
@@ -179,7 +179,7 @@ describe("ResponsesPage", () => {
|
|||||||
vi.mocked(getTagsByEnvironmentId).mockResolvedValue(mockTags);
|
vi.mocked(getTagsByEnvironmentId).mockResolvedValue(mockTags);
|
||||||
vi.mocked(getResponseCountBySurveyId).mockResolvedValue(10);
|
vi.mocked(getResponseCountBySurveyId).mockResolvedValue(10);
|
||||||
vi.mocked(findMatchingLocale).mockResolvedValue(mockLocale);
|
vi.mocked(findMatchingLocale).mockResolvedValue(mockLocale);
|
||||||
vi.mocked(getSurveyDomain).mockReturnValue(mockSurveyDomain);
|
vi.mocked(getPublicDomain).mockReturnValue(mockPublicDomain);
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
@@ -205,7 +205,7 @@ describe("ResponsesPage", () => {
|
|||||||
survey: mockSurvey,
|
survey: mockSurvey,
|
||||||
isReadOnly: false,
|
isReadOnly: false,
|
||||||
user: mockUser,
|
user: mockUser,
|
||||||
surveyDomain: mockSurveyDomain,
|
publicDomain: mockPublicDomain,
|
||||||
}),
|
}),
|
||||||
undefined
|
undefined
|
||||||
);
|
);
|
||||||
@@ -224,7 +224,7 @@ describe("ResponsesPage", () => {
|
|||||||
environment: mockEnvironment,
|
environment: mockEnvironment,
|
||||||
survey: mockSurvey,
|
survey: mockSurvey,
|
||||||
surveyId: mockSurveyId,
|
surveyId: mockSurveyId,
|
||||||
webAppUrl: "http://localhost:3000",
|
publicDomain: mockPublicDomain,
|
||||||
environmentTags: mockTags,
|
environmentTags: mockTags,
|
||||||
user: mockUser,
|
user: mockUser,
|
||||||
responsesPerPage: 10,
|
responsesPerPage: 10,
|
||||||
|
|||||||
+5
-5
@@ -1,8 +1,8 @@
|
|||||||
import { SurveyAnalysisNavigation } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/components/SurveyAnalysisNavigation";
|
import { SurveyAnalysisNavigation } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/components/SurveyAnalysisNavigation";
|
||||||
import { ResponsePage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/responses/components/ResponsePage";
|
import { ResponsePage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/responses/components/ResponsePage";
|
||||||
import { SurveyAnalysisCTA } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SurveyAnalysisCTA";
|
import { SurveyAnalysisCTA } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SurveyAnalysisCTA";
|
||||||
import { RESPONSES_PER_PAGE, WEBAPP_URL } from "@/lib/constants";
|
import { RESPONSES_PER_PAGE } from "@/lib/constants";
|
||||||
import { getSurveyDomain } from "@/lib/getSurveyUrl";
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getResponseCountBySurveyId } from "@/lib/response/service";
|
import { getResponseCountBySurveyId } from "@/lib/response/service";
|
||||||
import { getSurvey } from "@/lib/survey/service";
|
import { getSurvey } from "@/lib/survey/service";
|
||||||
import { getTagsByEnvironmentId } from "@/lib/tag/service";
|
import { getTagsByEnvironmentId } from "@/lib/tag/service";
|
||||||
@@ -37,7 +37,7 @@ const Page = async (props) => {
|
|||||||
const responseCount = await getResponseCountBySurveyId(params.surveyId);
|
const responseCount = await getResponseCountBySurveyId(params.surveyId);
|
||||||
|
|
||||||
const locale = await findMatchingLocale();
|
const locale = await findMatchingLocale();
|
||||||
const surveyDomain = getSurveyDomain();
|
const publicDomain = getPublicDomain();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<PageContentWrapper>
|
<PageContentWrapper>
|
||||||
@@ -49,7 +49,7 @@ const Page = async (props) => {
|
|||||||
survey={survey}
|
survey={survey}
|
||||||
isReadOnly={isReadOnly}
|
isReadOnly={isReadOnly}
|
||||||
user={user}
|
user={user}
|
||||||
surveyDomain={surveyDomain}
|
publicDomain={publicDomain}
|
||||||
responseCount={responseCount}
|
responseCount={responseCount}
|
||||||
/>
|
/>
|
||||||
}>
|
}>
|
||||||
@@ -59,7 +59,7 @@ const Page = async (props) => {
|
|||||||
environment={environment}
|
environment={environment}
|
||||||
survey={survey}
|
survey={survey}
|
||||||
surveyId={params.surveyId}
|
surveyId={params.surveyId}
|
||||||
webAppUrl={WEBAPP_URL}
|
publicDomain={publicDomain}
|
||||||
environmentTags={tags}
|
environmentTags={tags}
|
||||||
user={user}
|
user={user}
|
||||||
responsesPerPage={RESPONSES_PER_PAGE}
|
responsesPerPage={RESPONSES_PER_PAGE}
|
||||||
|
|||||||
+21
-9
@@ -149,7 +149,7 @@ describe("ShareEmbedSurvey", () => {
|
|||||||
|
|
||||||
const defaultProps = {
|
const defaultProps = {
|
||||||
survey: mockSurveyWeb,
|
survey: mockSurveyWeb,
|
||||||
surveyDomain: "test.com",
|
publicDomain: "https://public-domain.com",
|
||||||
open: true,
|
open: true,
|
||||||
modalView: "start" as "start" | "embed" | "panel",
|
modalView: "start" as "start" | "embed" | "panel",
|
||||||
setOpen: mockSetOpen,
|
setOpen: mockSetOpen,
|
||||||
@@ -158,7 +158,7 @@ describe("ShareEmbedSurvey", () => {
|
|||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
mockEmbedViewComponent.mockImplementation(
|
mockEmbedViewComponent.mockImplementation(
|
||||||
({ handleInitialPageButton, tabs, activeId, survey, email, surveyUrl, surveyDomain, locale }) => (
|
({ handleInitialPageButton, tabs, activeId, survey, email, surveyUrl, publicDomain, locale }) => (
|
||||||
<div>
|
<div>
|
||||||
<button onClick={() => handleInitialPageButton()}>EmbedViewMockContent</button>
|
<button onClick={() => handleInitialPageButton()}>EmbedViewMockContent</button>
|
||||||
<div data-testid="embedview-tabs">{JSON.stringify(tabs)}</div>
|
<div data-testid="embedview-tabs">{JSON.stringify(tabs)}</div>
|
||||||
@@ -166,7 +166,7 @@ describe("ShareEmbedSurvey", () => {
|
|||||||
<div data-testid="embedview-survey-id">{survey.id}</div>
|
<div data-testid="embedview-survey-id">{survey.id}</div>
|
||||||
<div data-testid="embedview-email">{email}</div>
|
<div data-testid="embedview-email">{email}</div>
|
||||||
<div data-testid="embedview-surveyUrl">{surveyUrl}</div>
|
<div data-testid="embedview-surveyUrl">{surveyUrl}</div>
|
||||||
<div data-testid="embedview-surveyDomain">{surveyDomain}</div>
|
<div data-testid="embedview-publicDomain">{publicDomain}</div>
|
||||||
<div data-testid="embedview-locale">{locale}</div>
|
<div data-testid="embedview-locale">{locale}</div>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
@@ -176,8 +176,8 @@ describe("ShareEmbedSurvey", () => {
|
|||||||
));
|
));
|
||||||
});
|
});
|
||||||
|
|
||||||
test("renders initial 'start' view correctly when open and modalView is 'start'", () => {
|
test("renders initial 'start' view correctly when open and modalView is 'start' for link survey", () => {
|
||||||
render(<ShareEmbedSurvey {...defaultProps} />);
|
render(<ShareEmbedSurvey {...defaultProps} survey={mockSurveyLink} />);
|
||||||
expect(screen.getByText("environments.surveys.summary.your_survey_is_public 🎉")).toBeInTheDocument();
|
expect(screen.getByText("environments.surveys.summary.your_survey_is_public 🎉")).toBeInTheDocument();
|
||||||
expect(screen.getByText("ShareSurveyLinkMock")).toBeInTheDocument();
|
expect(screen.getByText("ShareSurveyLinkMock")).toBeInTheDocument();
|
||||||
expect(screen.getByText("environments.surveys.summary.whats_next")).toBeInTheDocument();
|
expect(screen.getByText("environments.surveys.summary.whats_next")).toBeInTheDocument();
|
||||||
@@ -188,6 +188,18 @@ describe("ShareEmbedSurvey", () => {
|
|||||||
expect(screen.getByTestId("badge-mock")).toHaveTextContent("common.new");
|
expect(screen.getByTestId("badge-mock")).toHaveTextContent("common.new");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("renders initial 'start' view correctly when open and modalView is 'start' for app survey", () => {
|
||||||
|
render(<ShareEmbedSurvey {...defaultProps} survey={mockSurveyWeb} />);
|
||||||
|
// For app surveys, ShareSurveyLink should not be rendered
|
||||||
|
expect(screen.queryByText("ShareSurveyLinkMock")).not.toBeInTheDocument();
|
||||||
|
expect(screen.getByText("environments.surveys.summary.whats_next")).toBeInTheDocument();
|
||||||
|
expect(screen.getByText("environments.surveys.summary.embed_survey")).toBeInTheDocument();
|
||||||
|
expect(screen.getByText("environments.surveys.summary.configure_alerts")).toBeInTheDocument();
|
||||||
|
expect(screen.getByText("environments.surveys.summary.setup_integrations")).toBeInTheDocument();
|
||||||
|
expect(screen.getByText("environments.surveys.summary.send_to_panel")).toBeInTheDocument();
|
||||||
|
expect(screen.getByTestId("badge-mock")).toHaveTextContent("common.new");
|
||||||
|
});
|
||||||
|
|
||||||
test("switches to 'embed' view when 'Embed survey' button is clicked", async () => {
|
test("switches to 'embed' view when 'Embed survey' button is clicked", async () => {
|
||||||
render(<ShareEmbedSurvey {...defaultProps} />);
|
render(<ShareEmbedSurvey {...defaultProps} />);
|
||||||
const embedButton = screen.getByText("environments.surveys.summary.embed_survey");
|
const embedButton = screen.getByText("environments.surveys.summary.embed_survey");
|
||||||
@@ -205,7 +217,7 @@ describe("ShareEmbedSurvey", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test("returns to 'start' view when handleInitialPageButton is triggered from EmbedView", async () => {
|
test("returns to 'start' view when handleInitialPageButton is triggered from EmbedView", async () => {
|
||||||
render(<ShareEmbedSurvey {...defaultProps} modalView="embed" />);
|
render(<ShareEmbedSurvey {...defaultProps} survey={mockSurveyLink} modalView="embed" />);
|
||||||
expect(mockEmbedViewComponent).toHaveBeenCalled();
|
expect(mockEmbedViewComponent).toHaveBeenCalled();
|
||||||
expect(screen.getByText("EmbedViewMockContent")).toBeInTheDocument();
|
expect(screen.getByText("EmbedViewMockContent")).toBeInTheDocument();
|
||||||
|
|
||||||
@@ -219,7 +231,7 @@ describe("ShareEmbedSurvey", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
test("returns to 'start' view when handleInitialPageButton is triggered from PanelInfoView", async () => {
|
test("returns to 'start' view when handleInitialPageButton is triggered from PanelInfoView", async () => {
|
||||||
render(<ShareEmbedSurvey {...defaultProps} modalView="panel" />);
|
render(<ShareEmbedSurvey {...defaultProps} survey={mockSurveyLink} modalView="panel" />);
|
||||||
expect(mockPanelInfoViewComponent).toHaveBeenCalled();
|
expect(mockPanelInfoViewComponent).toHaveBeenCalled();
|
||||||
expect(screen.getByText("PanelInfoViewMockContent")).toBeInTheDocument();
|
expect(screen.getByText("PanelInfoViewMockContent")).toBeInTheDocument();
|
||||||
|
|
||||||
@@ -257,8 +269,8 @@ describe("ShareEmbedSurvey", () => {
|
|||||||
};
|
};
|
||||||
expect(embedViewProps.tabs.length).toBe(3);
|
expect(embedViewProps.tabs.length).toBe(3);
|
||||||
expect(embedViewProps.tabs.find((tab) => tab.id === "app")).toBeUndefined();
|
expect(embedViewProps.tabs.find((tab) => tab.id === "app")).toBeUndefined();
|
||||||
expect(embedViewProps.tabs[0].id).toBe("email");
|
expect(embedViewProps.tabs[0].id).toBe("link");
|
||||||
expect(embedViewProps.activeId).toBe("email");
|
expect(embedViewProps.activeId).toBe("link");
|
||||||
});
|
});
|
||||||
|
|
||||||
test("correctly configures for 'web' survey type in embed view", () => {
|
test("correctly configures for 'web' survey type in embed view", () => {
|
||||||
|
|||||||
+30
-28
@@ -24,7 +24,7 @@ import { PanelInfoView } from "./shareEmbedModal/PanelInfoView";
|
|||||||
|
|
||||||
interface ShareEmbedSurveyProps {
|
interface ShareEmbedSurveyProps {
|
||||||
survey: TSurvey;
|
survey: TSurvey;
|
||||||
surveyDomain: string;
|
publicDomain: string;
|
||||||
open: boolean;
|
open: boolean;
|
||||||
modalView: "start" | "embed" | "panel";
|
modalView: "start" | "embed" | "panel";
|
||||||
setOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
setOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
||||||
@@ -33,7 +33,7 @@ interface ShareEmbedSurveyProps {
|
|||||||
|
|
||||||
export const ShareEmbedSurvey = ({
|
export const ShareEmbedSurvey = ({
|
||||||
survey,
|
survey,
|
||||||
surveyDomain,
|
publicDomain,
|
||||||
open,
|
open,
|
||||||
modalView,
|
modalView,
|
||||||
setOpen,
|
setOpen,
|
||||||
@@ -47,13 +47,14 @@ export const ShareEmbedSurvey = ({
|
|||||||
const tabs = useMemo(
|
const tabs = useMemo(
|
||||||
() =>
|
() =>
|
||||||
[
|
[
|
||||||
{ id: "email", label: t("environments.surveys.summary.embed_in_an_email"), icon: MailIcon },
|
|
||||||
{ id: "webpage", label: t("environments.surveys.summary.embed_on_website"), icon: Code2Icon },
|
|
||||||
{
|
{
|
||||||
id: "link",
|
id: "link",
|
||||||
label: `${isSingleUseLinkSurvey ? t("environments.surveys.summary.single_use_links") : t("environments.surveys.summary.share_the_link")}`,
|
label: `${isSingleUseLinkSurvey ? t("environments.surveys.summary.single_use_links") : t("environments.surveys.summary.share_the_link")}`,
|
||||||
icon: LinkIcon,
|
icon: LinkIcon,
|
||||||
},
|
},
|
||||||
|
{ id: "email", label: t("environments.surveys.summary.embed_in_an_email"), icon: MailIcon },
|
||||||
|
{ id: "webpage", label: t("environments.surveys.summary.embed_on_website"), icon: Code2Icon },
|
||||||
|
|
||||||
{ id: "app", label: t("environments.surveys.summary.embed_in_app"), icon: SmartphoneIcon },
|
{ id: "app", label: t("environments.surveys.summary.embed_in_app"), icon: SmartphoneIcon },
|
||||||
].filter((tab) => !(survey.type === "link" && tab.id === "app")),
|
].filter((tab) => !(survey.type === "link" && tab.id === "app")),
|
||||||
[t, isSingleUseLinkSurvey, survey.type]
|
[t, isSingleUseLinkSurvey, survey.type]
|
||||||
@@ -66,16 +67,16 @@ export const ShareEmbedSurvey = ({
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const fetchSurveyUrl = async () => {
|
const fetchSurveyUrl = async () => {
|
||||||
try {
|
try {
|
||||||
const url = await getSurveyUrl(survey, surveyDomain, "default");
|
const url = await getSurveyUrl(survey, publicDomain, "default");
|
||||||
setSurveyUrl(url);
|
setSurveyUrl(url);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Failed to fetch survey URL:", error);
|
console.error("Failed to fetch survey URL:", error);
|
||||||
// Fallback to a default URL if fetching fails
|
// Fallback to a default URL if fetching fails
|
||||||
setSurveyUrl(`${surveyDomain}/s/${survey.id}`);
|
setSurveyUrl(`${publicDomain}/s/${survey.id}`);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
fetchSurveyUrl();
|
fetchSurveyUrl();
|
||||||
}, [survey, surveyDomain]);
|
}, [survey, publicDomain]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (survey.type !== "link") {
|
if (survey.type !== "link") {
|
||||||
@@ -106,27 +107,28 @@ export const ShareEmbedSurvey = ({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog open={open} onOpenChange={handleOpenChange}>
|
<Dialog open={open} onOpenChange={handleOpenChange}>
|
||||||
<DialogTitle className="sr-only" />
|
<DialogContent className="w-full bg-white p-0 lg:h-[700px]" width="wide">
|
||||||
<DialogContent className="w-full max-w-xl bg-white p-0 md:max-w-3xl lg:h-[700px] lg:max-w-5xl">
|
|
||||||
{showView === "start" ? (
|
{showView === "start" ? (
|
||||||
<div className="h-full max-w-full overflow-hidden">
|
<div className="flex h-full max-w-full flex-col overflow-hidden">
|
||||||
<div className="flex h-[200px] w-full flex-col items-center justify-center space-y-6 p-8 text-center lg:h-2/5">
|
{survey.type === "link" && (
|
||||||
<DialogTitle>
|
<div className="flex h-2/5 w-full flex-col items-center justify-center space-y-6 p-8 text-center">
|
||||||
<p className="pt-2 text-xl font-semibold text-slate-800">
|
<DialogTitle>
|
||||||
{t("environments.surveys.summary.your_survey_is_public")} 🎉
|
<p className="pt-2 text-xl font-semibold text-slate-800">
|
||||||
</p>
|
{t("environments.surveys.summary.your_survey_is_public")} 🎉
|
||||||
</DialogTitle>
|
</p>
|
||||||
<DialogDescription className="hidden" />
|
</DialogTitle>
|
||||||
<ShareSurveyLink
|
<DialogDescription className="hidden" />
|
||||||
survey={survey}
|
<ShareSurveyLink
|
||||||
surveyUrl={surveyUrl}
|
survey={survey}
|
||||||
surveyDomain={surveyDomain}
|
surveyUrl={surveyUrl}
|
||||||
setSurveyUrl={setSurveyUrl}
|
publicDomain={publicDomain}
|
||||||
locale={user.locale}
|
setSurveyUrl={setSurveyUrl}
|
||||||
/>
|
locale={user.locale}
|
||||||
</div>
|
/>
|
||||||
<div className="flex h-[300px] flex-col items-center justify-center gap-8 rounded-b-lg bg-slate-50 px-8 lg:h-3/5">
|
</div>
|
||||||
<p className="-mt-8 text-sm text-slate-500">{t("environments.surveys.summary.whats_next")}</p>
|
)}
|
||||||
|
<div className="flex h-full flex-col items-center justify-center gap-4 rounded-b-lg bg-slate-50 px-8">
|
||||||
|
<p className="text-sm text-slate-500">{t("environments.surveys.summary.whats_next")}</p>
|
||||||
<div className="grid grid-cols-4 gap-2">
|
<div className="grid grid-cols-4 gap-2">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
@@ -174,7 +176,7 @@ export const ShareEmbedSurvey = ({
|
|||||||
survey={survey}
|
survey={survey}
|
||||||
email={email}
|
email={email}
|
||||||
surveyUrl={surveyUrl}
|
surveyUrl={surveyUrl}
|
||||||
surveyDomain={surveyDomain}
|
publicDomain={publicDomain}
|
||||||
setSurveyUrl={setSurveyUrl}
|
setSurveyUrl={setSurveyUrl}
|
||||||
locale={user.locale}
|
locale={user.locale}
|
||||||
/>
|
/>
|
||||||
|
|||||||
+7
-5
@@ -104,13 +104,15 @@ describe("SummaryDropOffs", () => {
|
|||||||
|
|
||||||
// Check drop-off counts and percentages
|
// Check drop-off counts and percentages
|
||||||
expect(screen.getByText("20")).toBeInTheDocument();
|
expect(screen.getByText("20")).toBeInTheDocument();
|
||||||
expect(screen.getByText("(20%)")).toBeInTheDocument();
|
|
||||||
|
|
||||||
expect(screen.getByText("15")).toBeInTheDocument();
|
expect(screen.getByText("15")).toBeInTheDocument();
|
||||||
expect(screen.getByText("(19%)")).toBeInTheDocument(); // 18.75% rounded to 19%
|
|
||||||
|
|
||||||
expect(screen.getByText("10")).toBeInTheDocument();
|
expect(screen.getByText("10")).toBeInTheDocument();
|
||||||
expect(screen.getByText("(15%)")).toBeInTheDocument(); // 15.38% rounded to 15%
|
|
||||||
|
// Check percentage values
|
||||||
|
const percentageElements = screen.getAllByText(/\d+%/);
|
||||||
|
expect(percentageElements).toHaveLength(3);
|
||||||
|
expect(percentageElements[0]).toHaveTextContent("20%");
|
||||||
|
expect(percentageElements[1]).toHaveTextContent("19%");
|
||||||
|
expect(percentageElements[2]).toHaveTextContent("15%");
|
||||||
});
|
});
|
||||||
|
|
||||||
test("renders empty state when dropOff array is empty", () => {
|
test("renders empty state when dropOff array is empty", () => {
|
||||||
|
|||||||
+19
-13
@@ -23,9 +23,9 @@ export const SummaryDropOffs = ({ dropOff, survey }: SummaryDropOffsProps) => {
|
|||||||
return (
|
return (
|
||||||
<div className="rounded-xl border border-slate-200 bg-white shadow-sm">
|
<div className="rounded-xl border border-slate-200 bg-white shadow-sm">
|
||||||
<div className="">
|
<div className="">
|
||||||
<div className="grid h-10 grid-cols-6 items-center border-y border-slate-200 bg-slate-100 text-sm font-semibold text-slate-600">
|
<div className="grid min-h-10 grid-cols-6 items-center rounded-t-xl border-b border-slate-200 bg-slate-100 text-sm font-semibold text-slate-600">
|
||||||
<div className="col-span-3 pl-4 md:pl-6">{t("common.questions")}</div>
|
<div className="col-span-3 px-4 md:px-6">{t("common.questions")}</div>
|
||||||
<div className="flex justify-center">
|
<div className="flex justify-end px-4 md:px-6">
|
||||||
<TooltipProvider delayDuration={50}>
|
<TooltipProvider delayDuration={50}>
|
||||||
<Tooltip>
|
<Tooltip>
|
||||||
<TooltipTrigger>
|
<TooltipTrigger>
|
||||||
@@ -37,14 +37,16 @@ export const SummaryDropOffs = ({ dropOff, survey }: SummaryDropOffsProps) => {
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
</TooltipProvider>
|
</TooltipProvider>
|
||||||
</div>
|
</div>
|
||||||
<div className="px-4 text-center md:px-6">{t("environments.surveys.summary.impressions")}</div>
|
<div className="px-4 text-right md:px-6">{t("environments.surveys.summary.impressions")}</div>
|
||||||
<div className="pr-6 text-center md:pl-6">{t("environments.surveys.summary.drop_offs")}</div>
|
<div className="px-4 text-right md:mr-1 md:pl-6 md:pr-6">
|
||||||
|
{t("environments.surveys.summary.drop_offs")}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{dropOff.map((quesDropOff) => (
|
{dropOff.map((quesDropOff) => (
|
||||||
<div
|
<div
|
||||||
key={quesDropOff.questionId}
|
key={quesDropOff.questionId}
|
||||||
className="grid grid-cols-6 items-center border-b border-slate-100 py-2 text-sm text-slate-800 md:text-base">
|
className="grid grid-cols-6 items-start border-b border-slate-100 text-xs text-slate-800 md:text-sm">
|
||||||
<div className="col-span-3 flex gap-3 pl-4 md:pl-6">
|
<div className="col-span-3 flex gap-3 px-4 py-2 md:px-6">
|
||||||
{getIcon(quesDropOff.questionType)}
|
{getIcon(quesDropOff.questionType)}
|
||||||
<p>
|
<p>
|
||||||
{formatTextWithSlashes(
|
{formatTextWithSlashes(
|
||||||
@@ -57,17 +59,21 @@ export const SummaryDropOffs = ({ dropOff, survey }: SummaryDropOffsProps) => {
|
|||||||
"default"
|
"default"
|
||||||
)["default"],
|
)["default"],
|
||||||
"@",
|
"@",
|
||||||
["text-lg"]
|
["text-sm"]
|
||||||
)}
|
)}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<div className="whitespace-pre-wrap text-center font-semibold">
|
<div className="whitespace-pre-wrap px-4 py-2 text-right font-mono font-medium md:px-6">
|
||||||
{quesDropOff.ttc > 0 ? (quesDropOff.ttc / 1000).toFixed(2) + "s" : "N/A"}
|
{quesDropOff.ttc > 0 ? (quesDropOff.ttc / 1000).toFixed(2) + "s" : "N/A"}
|
||||||
</div>
|
</div>
|
||||||
<div className="whitespace-pre-wrap text-center font-semibold">{quesDropOff.impressions}</div>
|
<div className="whitespace-pre-wrap px-4 py-2 text-right font-mono font-medium md:px-6">
|
||||||
<div className="pl-6 text-center md:px-6">
|
{quesDropOff.impressions}
|
||||||
<span className="mr-1.5 font-semibold">{quesDropOff.dropOffCount}</span>
|
</div>
|
||||||
<span>({Math.round(quesDropOff.dropOffPercentage)}%)</span>
|
<div className="px-4 py-2 text-right md:px-6">
|
||||||
|
<span className="mr-1 inline-block w-fit rounded-xl bg-slate-100 px-2 py-1 text-left text-xs">
|
||||||
|
{Math.round(quesDropOff.dropOffPercentage)}%
|
||||||
|
</span>
|
||||||
|
<span className="mr-1 font-mono font-medium">{quesDropOff.dropOffCount}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
))}
|
))}
|
||||||
|
|||||||
+3
-2
@@ -1,5 +1,6 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import { Button } from "@/modules/ui/components/button";
|
||||||
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from "@/modules/ui/components/tooltip";
|
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from "@/modules/ui/components/tooltip";
|
||||||
import { useTranslate } from "@tolgee/react";
|
import { useTranslate } from "@tolgee/react";
|
||||||
import { ChevronDownIcon, ChevronUpIcon } from "lucide-react";
|
import { ChevronDownIcon, ChevronUpIcon } from "lucide-react";
|
||||||
@@ -117,13 +118,13 @@ export const SummaryMetadata = ({
|
|||||||
)}
|
)}
|
||||||
</span>
|
</span>
|
||||||
{!isLoading && (
|
{!isLoading && (
|
||||||
<span className="ml-1 flex items-center rounded-md bg-slate-800 px-2 py-1 text-xs text-slate-50 group-hover:bg-slate-700">
|
<Button variant="secondary" className="h-6 w-6">
|
||||||
{showDropOffs ? (
|
{showDropOffs ? (
|
||||||
<ChevronUpIcon className="h-4 w-4" />
|
<ChevronUpIcon className="h-4 w-4" />
|
||||||
) : (
|
) : (
|
||||||
<ChevronDownIcon className="h-4 w-4" />
|
<ChevronDownIcon className="h-4 w-4" />
|
||||||
)}
|
)}
|
||||||
</span>
|
</Button>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
+3
-3
@@ -36,7 +36,7 @@ interface SummaryPageProps {
|
|||||||
environment: TEnvironment;
|
environment: TEnvironment;
|
||||||
survey: TSurvey;
|
survey: TSurvey;
|
||||||
surveyId: string;
|
surveyId: string;
|
||||||
webAppUrl: string;
|
publicDomain: string;
|
||||||
locale: TUserLocale;
|
locale: TUserLocale;
|
||||||
isReadOnly: boolean;
|
isReadOnly: boolean;
|
||||||
initialSurveySummary?: TSurveySummary;
|
initialSurveySummary?: TSurveySummary;
|
||||||
@@ -46,7 +46,7 @@ export const SummaryPage = ({
|
|||||||
environment,
|
environment,
|
||||||
survey,
|
survey,
|
||||||
surveyId,
|
surveyId,
|
||||||
webAppUrl,
|
publicDomain,
|
||||||
locale,
|
locale,
|
||||||
isReadOnly,
|
isReadOnly,
|
||||||
initialSurveySummary,
|
initialSurveySummary,
|
||||||
@@ -133,7 +133,7 @@ export const SummaryPage = ({
|
|||||||
<div className="flex gap-1.5">
|
<div className="flex gap-1.5">
|
||||||
<CustomFilter survey={surveyMemoized} />
|
<CustomFilter survey={surveyMemoized} />
|
||||||
{!isReadOnly && !isSharingPage && (
|
{!isReadOnly && !isSharingPage && (
|
||||||
<ResultsShareButton survey={surveyMemoized} webAppUrl={webAppUrl} />
|
<ResultsShareButton survey={surveyMemoized} publicDomain={publicDomain} />
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
<ScrollToTop containerId="mainContent" />
|
<ScrollToTop containerId="mainContent" />
|
||||||
|
|||||||
+255
-245
@@ -21,6 +21,8 @@ vi.mock("@/modules/ee/audit-logs/lib/utils", () => ({
|
|||||||
}),
|
}),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
const mockPublicDomain = "https://public-domain.com";
|
||||||
|
|
||||||
// Mock constants
|
// Mock constants
|
||||||
vi.mock("@/lib/constants", () => ({
|
vi.mock("@/lib/constants", () => ({
|
||||||
IS_FORMBRICKS_CLOUD: false,
|
IS_FORMBRICKS_CLOUD: false,
|
||||||
@@ -49,6 +51,12 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
REDIS_URL: "mock-url",
|
REDIS_URL: "mock-url",
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/env", () => ({
|
||||||
|
env: {
|
||||||
|
PUBLIC_URL: "https://public-domain.com",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
// Create a spy for refreshSingleUseId so we can override it in tests
|
// Create a spy for refreshSingleUseId so we can override it in tests
|
||||||
const refreshSingleUseIdSpy = vi.fn(() => Promise.resolve("newSingleUseId"));
|
const refreshSingleUseIdSpy = vi.fn(() => Promise.resolve("newSingleUseId"));
|
||||||
|
|
||||||
@@ -61,18 +69,18 @@ vi.mock("@/modules/survey/hooks/useSingleUseId", () => ({
|
|||||||
|
|
||||||
const mockSearchParams = new URLSearchParams();
|
const mockSearchParams = new URLSearchParams();
|
||||||
const mockPush = vi.fn();
|
const mockPush = vi.fn();
|
||||||
|
const mockReplace = vi.fn();
|
||||||
|
|
||||||
// Mock next/navigation
|
// Mock next/navigation
|
||||||
vi.mock("next/navigation", () => ({
|
vi.mock("next/navigation", () => ({
|
||||||
useRouter: () => ({ push: mockPush }),
|
useRouter: () => ({ push: mockPush, replace: mockReplace }),
|
||||||
useSearchParams: () => mockSearchParams,
|
useSearchParams: () => mockSearchParams,
|
||||||
usePathname: () => "/current",
|
usePathname: () => "/current-path",
|
||||||
useParams: () => ({ environmentId: "env123", surveyId: "survey123" }),
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Mock copySurveyLink to return a predictable string
|
// Mock copySurveyLink to return a predictable string
|
||||||
vi.mock("@/modules/survey/lib/client-utils", () => ({
|
vi.mock("@/modules/survey/lib/client-utils", () => ({
|
||||||
copySurveyLink: vi.fn((url: string, id: string) => `${url}?id=${id}`),
|
copySurveyLink: vi.fn((url: string, suId: string) => `${url}?suId=${suId}`),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
// Mock the copy survey action
|
// Mock the copy survey action
|
||||||
@@ -103,6 +111,10 @@ vi.mock("@/app/share/[sharingKey]/actions", () => ({
|
|||||||
getResponseCountBySurveySharingKeyAction: vi.fn(() => Promise.resolve({ data: 5 })),
|
getResponseCountBySurveySharingKeyAction: vi.fn(() => Promise.resolve({ data: 5 })),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/getPublicUrl", () => ({
|
||||||
|
getPublicDomain: vi.fn(() => mockPublicDomain),
|
||||||
|
}));
|
||||||
|
|
||||||
vi.spyOn(toast, "success");
|
vi.spyOn(toast, "success");
|
||||||
vi.spyOn(toast, "error");
|
vi.spyOn(toast, "error");
|
||||||
|
|
||||||
@@ -119,283 +131,281 @@ const dummySurvey = {
|
|||||||
id: "survey123",
|
id: "survey123",
|
||||||
type: "link",
|
type: "link",
|
||||||
environmentId: "env123",
|
environmentId: "env123",
|
||||||
status: "active",
|
status: "inProgress",
|
||||||
|
resultShareKey: null,
|
||||||
} as unknown as TSurvey;
|
} as unknown as TSurvey;
|
||||||
|
|
||||||
|
const dummyAppSurvey = {
|
||||||
|
id: "survey123",
|
||||||
|
type: "app",
|
||||||
|
environmentId: "env123",
|
||||||
|
status: "inProgress",
|
||||||
|
} as unknown as TSurvey;
|
||||||
|
|
||||||
const dummyEnvironment = { id: "env123", appSetupCompleted: true } as TEnvironment;
|
const dummyEnvironment = { id: "env123", appSetupCompleted: true } as TEnvironment;
|
||||||
const dummyUser = { id: "user123", name: "Test User" } as TUser;
|
const dummyUser = { id: "user123", name: "Test User" } as TUser;
|
||||||
const surveyDomain = "https://surveys.test.formbricks.com";
|
|
||||||
|
|
||||||
describe("SurveyAnalysisCTA - handleCopyLink", () => {
|
describe("SurveyAnalysisCTA", () => {
|
||||||
afterEach(() => {
|
|
||||||
cleanup();
|
|
||||||
});
|
|
||||||
|
|
||||||
test("calls copySurveyLink and clipboard.writeText on success", async () => {
|
|
||||||
render(
|
|
||||||
<SurveyAnalysisCTA
|
|
||||||
survey={dummySurvey}
|
|
||||||
environment={dummyEnvironment}
|
|
||||||
isReadOnly={false}
|
|
||||||
surveyDomain={surveyDomain}
|
|
||||||
user={dummyUser}
|
|
||||||
responseCount={5}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
|
|
||||||
const copyButton = screen.getByRole("button", { name: "common.copy_link" });
|
|
||||||
fireEvent.click(copyButton);
|
|
||||||
|
|
||||||
await waitFor(() => {
|
|
||||||
expect(refreshSingleUseIdSpy).toHaveBeenCalled();
|
|
||||||
expect(writeTextMock).toHaveBeenCalledWith(
|
|
||||||
"https://surveys.test.formbricks.com/s/survey123?id=newSingleUseId"
|
|
||||||
);
|
|
||||||
expect(toast.success).toHaveBeenCalledWith("common.copied_to_clipboard");
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
test("shows error toast on failure", async () => {
|
|
||||||
refreshSingleUseIdSpy.mockImplementationOnce(() => Promise.reject(new Error("fail")));
|
|
||||||
render(
|
|
||||||
<SurveyAnalysisCTA
|
|
||||||
survey={dummySurvey}
|
|
||||||
environment={dummyEnvironment}
|
|
||||||
isReadOnly={false}
|
|
||||||
surveyDomain={surveyDomain}
|
|
||||||
user={dummyUser}
|
|
||||||
responseCount={5}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
|
|
||||||
const copyButton = screen.getByRole("button", { name: "common.copy_link" });
|
|
||||||
fireEvent.click(copyButton);
|
|
||||||
|
|
||||||
await waitFor(() => {
|
|
||||||
expect(refreshSingleUseIdSpy).toHaveBeenCalled();
|
|
||||||
expect(writeTextMock).not.toHaveBeenCalled();
|
|
||||||
expect(toast.error).toHaveBeenCalledWith("environments.surveys.summary.failed_to_copy_link");
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// New tests for squarePenIcon and edit functionality
|
|
||||||
describe("SurveyAnalysisCTA - Edit functionality", () => {
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
vi.resetAllMocks();
|
vi.resetAllMocks();
|
||||||
|
mockSearchParams.delete("share"); // reset params
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
cleanup();
|
cleanup();
|
||||||
});
|
});
|
||||||
|
|
||||||
test("opens EditPublicSurveyAlertDialog when edit icon is clicked and response count > 0", async () => {
|
describe("Edit functionality", () => {
|
||||||
render(
|
test("opens EditPublicSurveyAlertDialog when edit icon is clicked and response count > 0", async () => {
|
||||||
<SurveyAnalysisCTA
|
render(
|
||||||
survey={dummySurvey}
|
<SurveyAnalysisCTA
|
||||||
environment={dummyEnvironment}
|
survey={dummySurvey}
|
||||||
isReadOnly={false}
|
environment={dummyEnvironment}
|
||||||
surveyDomain={surveyDomain}
|
isReadOnly={false}
|
||||||
user={dummyUser}
|
publicDomain={mockPublicDomain}
|
||||||
responseCount={5}
|
user={dummyUser}
|
||||||
/>
|
responseCount={5}
|
||||||
);
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
// Find the edit button
|
// Find the edit button
|
||||||
const editButton = screen.getByRole("button", { name: "common.edit" });
|
const editButton = screen.getByRole("button", { name: "common.edit" });
|
||||||
await fireEvent.click(editButton);
|
await fireEvent.click(editButton);
|
||||||
|
|
||||||
// Check if dialog is shown
|
// Check if dialog is shown
|
||||||
const dialogTitle = screen.getByText("environments.surveys.edit.caution_edit_published_survey");
|
const dialogTitle = screen.getByText("environments.surveys.edit.caution_edit_published_survey");
|
||||||
expect(dialogTitle).toBeInTheDocument();
|
expect(dialogTitle).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("navigates directly to edit page when response count = 0", async () => {
|
||||||
|
render(
|
||||||
|
<SurveyAnalysisCTA
|
||||||
|
survey={dummySurvey}
|
||||||
|
environment={dummyEnvironment}
|
||||||
|
isReadOnly={false}
|
||||||
|
publicDomain={mockPublicDomain}
|
||||||
|
user={dummyUser}
|
||||||
|
responseCount={0}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
// Find the edit button
|
||||||
|
const editButton = screen.getByRole("button", { name: "common.edit" });
|
||||||
|
await fireEvent.click(editButton);
|
||||||
|
|
||||||
|
// Should navigate directly to edit page
|
||||||
|
expect(mockPush).toHaveBeenCalledWith(
|
||||||
|
`/environments/${dummyEnvironment.id}/surveys/${dummySurvey.id}/edit`
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("doesn't show edit button when isReadOnly is true", () => {
|
||||||
|
render(
|
||||||
|
<SurveyAnalysisCTA
|
||||||
|
survey={dummySurvey}
|
||||||
|
environment={dummyEnvironment}
|
||||||
|
isReadOnly={true}
|
||||||
|
publicDomain={mockPublicDomain}
|
||||||
|
user={dummyUser}
|
||||||
|
responseCount={5}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
const editButton = screen.queryByRole("button", { name: "common.edit" });
|
||||||
|
expect(editButton).not.toBeInTheDocument();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test("navigates directly to edit page when response count = 0", async () => {
|
describe("Duplicate functionality", () => {
|
||||||
render(
|
test("duplicates survey and redirects on primary button click", async () => {
|
||||||
<SurveyAnalysisCTA
|
mockCopySurveyToOtherEnvironmentAction.mockResolvedValue({
|
||||||
survey={dummySurvey}
|
data: { id: "newSurvey456" },
|
||||||
environment={dummyEnvironment}
|
});
|
||||||
isReadOnly={false}
|
|
||||||
surveyDomain={surveyDomain}
|
|
||||||
user={dummyUser}
|
|
||||||
responseCount={0}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
|
|
||||||
// Find the edit button
|
render(
|
||||||
const editButton = screen.getByRole("button", { name: "common.edit" });
|
<SurveyAnalysisCTA
|
||||||
await fireEvent.click(editButton);
|
survey={dummySurvey}
|
||||||
|
environment={dummyEnvironment}
|
||||||
|
isReadOnly={false}
|
||||||
|
publicDomain={mockPublicDomain}
|
||||||
|
user={dummyUser}
|
||||||
|
responseCount={5}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
// Should navigate directly to edit page
|
const editButton = screen.getByRole("button", { name: "common.edit" });
|
||||||
expect(mockPush).toHaveBeenCalledWith(
|
fireEvent.click(editButton);
|
||||||
`/environments/${dummyEnvironment.id}/surveys/${dummySurvey.id}/edit`
|
|
||||||
);
|
const primaryButton = await screen.findByText("environments.surveys.edit.caution_edit_duplicate");
|
||||||
|
fireEvent.click(primaryButton);
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(mockCopySurveyToOtherEnvironmentAction).toHaveBeenCalledWith({
|
||||||
|
environmentId: "env123",
|
||||||
|
surveyId: "survey123",
|
||||||
|
targetEnvironmentId: "env123",
|
||||||
|
});
|
||||||
|
expect(mockPush).toHaveBeenCalledWith("/environments/env123/surveys/newSurvey456/edit");
|
||||||
|
expect(toast.success).toHaveBeenCalledWith("environments.surveys.survey_duplicated_successfully");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("shows error toast on duplication failure", async () => {
|
||||||
|
const error = { error: "Duplication failed" };
|
||||||
|
mockCopySurveyToOtherEnvironmentAction.mockResolvedValue(error);
|
||||||
|
render(
|
||||||
|
<SurveyAnalysisCTA
|
||||||
|
survey={dummySurvey}
|
||||||
|
environment={dummyEnvironment}
|
||||||
|
isReadOnly={false}
|
||||||
|
publicDomain={mockPublicDomain}
|
||||||
|
user={dummyUser}
|
||||||
|
responseCount={5}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
const editButton = screen.getByRole("button", { name: "common.edit" });
|
||||||
|
fireEvent.click(editButton);
|
||||||
|
|
||||||
|
const primaryButton = await screen.findByText("environments.surveys.edit.caution_edit_duplicate");
|
||||||
|
fireEvent.click(primaryButton);
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
expect(toast.error).toHaveBeenCalledWith("Duplication failed");
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test("doesn't show edit button when isReadOnly is true", () => {
|
describe("Share button and modal", () => {
|
||||||
render(
|
test("opens share modal when 'Share survey' button is clicked", async () => {
|
||||||
<SurveyAnalysisCTA
|
render(
|
||||||
survey={dummySurvey}
|
<SurveyAnalysisCTA
|
||||||
environment={dummyEnvironment}
|
survey={dummySurvey}
|
||||||
isReadOnly={true}
|
environment={dummyEnvironment}
|
||||||
surveyDomain={surveyDomain}
|
isReadOnly={false}
|
||||||
user={dummyUser}
|
publicDomain={mockPublicDomain}
|
||||||
responseCount={5}
|
user={dummyUser}
|
||||||
/>
|
responseCount={5}
|
||||||
);
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
// Try to find the edit button (it shouldn't exist)
|
const shareButton = screen.getByText("environments.surveys.summary.share_survey");
|
||||||
const editButton = screen.queryByRole("button", { name: "common.edit" });
|
fireEvent.click(shareButton);
|
||||||
expect(editButton).not.toBeInTheDocument();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
// Updated test description to mention EditPublicSurveyAlertDialog
|
// The share button opens the embed modal, not a URL
|
||||||
describe("SurveyAnalysisCTA - duplicateSurveyAndRoute and EditPublicSurveyAlertDialog", () => {
|
// We can verify this by checking that the ShareEmbedSurvey component is rendered
|
||||||
afterEach(() => {
|
// with the embed modal open
|
||||||
cleanup();
|
expect(screen.getByText("environments.surveys.summary.share_survey")).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("renders ShareEmbedSurvey component when share modal is open", async () => {
|
||||||
|
mockSearchParams.set("share", "true");
|
||||||
|
render(
|
||||||
|
<SurveyAnalysisCTA
|
||||||
|
survey={dummySurvey}
|
||||||
|
environment={dummyEnvironment}
|
||||||
|
isReadOnly={false}
|
||||||
|
publicDomain={mockPublicDomain}
|
||||||
|
user={dummyUser}
|
||||||
|
responseCount={5}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
// Assuming ShareEmbedSurvey renders a dialog with a specific title when open
|
||||||
|
const dialog = await screen.findByRole("dialog");
|
||||||
|
expect(dialog).toBeInTheDocument();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test("duplicates survey successfully and navigates to edit page", async () => {
|
describe("General UI and visibility", () => {
|
||||||
// Mock the API response
|
test("shows public results badge when resultShareKey is present", () => {
|
||||||
mockCopySurveyToOtherEnvironmentAction.mockResolvedValueOnce({
|
const surveyWithShareKey = { ...dummySurvey, resultShareKey: "someKey" } as TSurvey;
|
||||||
data: { id: "duplicated-survey-456" },
|
render(
|
||||||
|
<SurveyAnalysisCTA
|
||||||
|
survey={surveyWithShareKey}
|
||||||
|
environment={dummyEnvironment}
|
||||||
|
isReadOnly={false}
|
||||||
|
publicDomain={mockPublicDomain}
|
||||||
|
user={dummyUser}
|
||||||
|
responseCount={5}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(screen.getByText("environments.surveys.summary.results_are_public")).toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
render(
|
test("shows SurveyStatusDropdown for non-draft surveys", () => {
|
||||||
<SurveyAnalysisCTA
|
render(
|
||||||
survey={dummySurvey}
|
<SurveyAnalysisCTA
|
||||||
environment={dummyEnvironment}
|
survey={dummySurvey}
|
||||||
isReadOnly={false}
|
environment={dummyEnvironment}
|
||||||
surveyDomain={surveyDomain}
|
isReadOnly={false}
|
||||||
user={dummyUser}
|
publicDomain={mockPublicDomain}
|
||||||
responseCount={5}
|
user={dummyUser}
|
||||||
/>
|
responseCount={5}
|
||||||
);
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
// Find and click the edit button to show dialog
|
expect(screen.getByRole("combobox")).toBeInTheDocument();
|
||||||
const editButton = screen.getByRole("button", { name: "common.edit" });
|
|
||||||
await fireEvent.click(editButton);
|
|
||||||
|
|
||||||
// Find and click the duplicate button in dialog
|
|
||||||
const duplicateButton = screen.getByRole("button", {
|
|
||||||
name: "environments.surveys.edit.caution_edit_duplicate",
|
|
||||||
});
|
|
||||||
await fireEvent.click(duplicateButton);
|
|
||||||
|
|
||||||
// Verify the API was called with correct parameters
|
|
||||||
expect(mockCopySurveyToOtherEnvironmentAction).toHaveBeenCalledWith({
|
|
||||||
environmentId: dummyEnvironment.id,
|
|
||||||
surveyId: dummySurvey.id,
|
|
||||||
targetEnvironmentId: dummyEnvironment.id,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Verify success toast was shown
|
test("does not show SurveyStatusDropdown for draft surveys", () => {
|
||||||
expect(toast.success).toHaveBeenCalledWith("environments.surveys.survey_duplicated_successfully");
|
const draftSurvey = { ...dummySurvey, status: "draft" } as TSurvey;
|
||||||
|
render(
|
||||||
// Verify navigation to edit page
|
<SurveyAnalysisCTA
|
||||||
expect(mockPush).toHaveBeenCalledWith(
|
survey={draftSurvey}
|
||||||
`/environments/${dummyEnvironment.id}/surveys/duplicated-survey-456/edit`
|
environment={dummyEnvironment}
|
||||||
);
|
isReadOnly={false}
|
||||||
});
|
publicDomain={mockPublicDomain}
|
||||||
|
user={dummyUser}
|
||||||
test("shows error toast when duplication fails with error object", async () => {
|
responseCount={5}
|
||||||
// Mock API failure with error object
|
/>
|
||||||
mockCopySurveyToOtherEnvironmentAction.mockResolvedValueOnce({
|
);
|
||||||
error: "Test error message",
|
expect(screen.queryByRole("combobox")).not.toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
render(
|
test("hides status dropdown and edit actions when isReadOnly is true", () => {
|
||||||
<SurveyAnalysisCTA
|
render(
|
||||||
survey={dummySurvey}
|
<SurveyAnalysisCTA
|
||||||
environment={dummyEnvironment}
|
survey={dummySurvey}
|
||||||
isReadOnly={false}
|
environment={dummyEnvironment}
|
||||||
surveyDomain={surveyDomain}
|
isReadOnly={true}
|
||||||
user={dummyUser}
|
publicDomain={mockPublicDomain}
|
||||||
responseCount={5}
|
user={dummyUser}
|
||||||
/>
|
responseCount={5}
|
||||||
);
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
// Open dialog
|
expect(screen.queryByRole("combobox")).not.toBeInTheDocument();
|
||||||
const editButton = screen.getByRole("button", { name: "common.edit" });
|
expect(screen.queryByRole("button", { name: "common.edit" })).not.toBeInTheDocument();
|
||||||
await fireEvent.click(editButton);
|
|
||||||
|
|
||||||
// Click duplicate
|
|
||||||
const duplicateButton = screen.getByRole("button", {
|
|
||||||
name: "environments.surveys.edit.caution_edit_duplicate",
|
|
||||||
});
|
|
||||||
await fireEvent.click(duplicateButton);
|
|
||||||
|
|
||||||
// Verify error toast
|
|
||||||
expect(toast.error).toHaveBeenCalledWith("Test error message");
|
|
||||||
});
|
|
||||||
|
|
||||||
test("navigates to edit page when cancel button is clicked in dialog", async () => {
|
|
||||||
render(
|
|
||||||
<SurveyAnalysisCTA
|
|
||||||
survey={dummySurvey}
|
|
||||||
environment={dummyEnvironment}
|
|
||||||
isReadOnly={false}
|
|
||||||
surveyDomain={surveyDomain}
|
|
||||||
user={dummyUser}
|
|
||||||
responseCount={5}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
|
|
||||||
// Open dialog
|
|
||||||
const editButton = screen.getByRole("button", { name: "common.edit" });
|
|
||||||
await fireEvent.click(editButton);
|
|
||||||
|
|
||||||
// Click edit (cancel) button
|
|
||||||
const editButtonInDialog = screen.getByRole("button", { name: "common.edit" });
|
|
||||||
await fireEvent.click(editButtonInDialog);
|
|
||||||
|
|
||||||
// Verify navigation
|
|
||||||
expect(mockPush).toHaveBeenCalledWith(
|
|
||||||
`/environments/${dummyEnvironment.id}/surveys/${dummySurvey.id}/edit`
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
test("shows loading state when duplicating survey", async () => {
|
|
||||||
// Create a promise that we can resolve manually
|
|
||||||
let resolvePromise: (value: any) => void;
|
|
||||||
const promise = new Promise((resolve) => {
|
|
||||||
resolvePromise = resolve;
|
|
||||||
});
|
});
|
||||||
|
|
||||||
mockCopySurveyToOtherEnvironmentAction.mockImplementation(() => promise);
|
test("shows preview button for link surveys", () => {
|
||||||
|
render(
|
||||||
render(
|
<SurveyAnalysisCTA
|
||||||
<SurveyAnalysisCTA
|
survey={dummySurvey}
|
||||||
survey={dummySurvey}
|
environment={dummyEnvironment}
|
||||||
environment={dummyEnvironment}
|
isReadOnly={false}
|
||||||
isReadOnly={false}
|
publicDomain={mockPublicDomain}
|
||||||
surveyDomain={surveyDomain}
|
user={dummyUser}
|
||||||
user={dummyUser}
|
responseCount={5}
|
||||||
responseCount={5}
|
/>
|
||||||
/>
|
);
|
||||||
);
|
expect(screen.getByRole("button", { name: "common.preview" })).toBeInTheDocument();
|
||||||
|
|
||||||
// Open dialog
|
|
||||||
const editButton = screen.getByRole("button", { name: "common.edit" });
|
|
||||||
await fireEvent.click(editButton);
|
|
||||||
|
|
||||||
// Click duplicate
|
|
||||||
const duplicateButton = screen.getByRole("button", {
|
|
||||||
name: "environments.surveys.edit.caution_edit_duplicate",
|
|
||||||
});
|
|
||||||
await fireEvent.click(duplicateButton);
|
|
||||||
|
|
||||||
// Button should now be in loading state
|
|
||||||
// expect(duplicateButton).toHaveAttribute("data-state", "loading");
|
|
||||||
|
|
||||||
// Resolve the promise
|
|
||||||
resolvePromise!({
|
|
||||||
data: { id: "duplicated-survey-456" },
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Wait for the promise to resolve
|
test("hides preview button for app surveys", () => {
|
||||||
await waitFor(() => {
|
render(
|
||||||
expect(mockPush).toHaveBeenCalled();
|
<SurveyAnalysisCTA
|
||||||
|
survey={dummyAppSurvey}
|
||||||
|
environment={dummyEnvironment}
|
||||||
|
isReadOnly={false}
|
||||||
|
publicDomain={mockPublicDomain}
|
||||||
|
user={dummyUser}
|
||||||
|
responseCount={5}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
expect(screen.queryByRole("button", { name: "common.preview" })).not.toBeInTheDocument();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
+17
-49
@@ -5,13 +5,12 @@ import { SuccessMessage } from "@/app/(app)/environments/[environmentId]/surveys
|
|||||||
import { SurveyStatusDropdown } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/components/SurveyStatusDropdown";
|
import { SurveyStatusDropdown } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/components/SurveyStatusDropdown";
|
||||||
import { getFormattedErrorMessage } from "@/lib/utils/helper";
|
import { getFormattedErrorMessage } from "@/lib/utils/helper";
|
||||||
import { EditPublicSurveyAlertDialog } from "@/modules/survey/components/edit-public-survey-alert-dialog";
|
import { EditPublicSurveyAlertDialog } from "@/modules/survey/components/edit-public-survey-alert-dialog";
|
||||||
import { useSingleUseId } from "@/modules/survey/hooks/useSingleUseId";
|
|
||||||
import { copySurveyLink } from "@/modules/survey/lib/client-utils";
|
|
||||||
import { copySurveyToOtherEnvironmentAction } from "@/modules/survey/list/actions";
|
import { copySurveyToOtherEnvironmentAction } from "@/modules/survey/list/actions";
|
||||||
import { Badge } from "@/modules/ui/components/badge";
|
import { Badge } from "@/modules/ui/components/badge";
|
||||||
|
import { Button } from "@/modules/ui/components/button";
|
||||||
import { IconBar } from "@/modules/ui/components/iconbar";
|
import { IconBar } from "@/modules/ui/components/iconbar";
|
||||||
import { useTranslate } from "@tolgee/react";
|
import { useTranslate } from "@tolgee/react";
|
||||||
import { BellRing, Code2Icon, Eye, LinkIcon, SquarePenIcon, UsersRound } from "lucide-react";
|
import { BellRing, Eye, SquarePenIcon } from "lucide-react";
|
||||||
import { usePathname, useRouter, useSearchParams } from "next/navigation";
|
import { usePathname, useRouter, useSearchParams } from "next/navigation";
|
||||||
import { useEffect, useMemo, useState } from "react";
|
import { useEffect, useMemo, useState } from "react";
|
||||||
import toast from "react-hot-toast";
|
import toast from "react-hot-toast";
|
||||||
@@ -24,7 +23,7 @@ interface SurveyAnalysisCTAProps {
|
|||||||
environment: TEnvironment;
|
environment: TEnvironment;
|
||||||
isReadOnly: boolean;
|
isReadOnly: boolean;
|
||||||
user: TUser;
|
user: TUser;
|
||||||
surveyDomain: string;
|
publicDomain: string;
|
||||||
responseCount: number;
|
responseCount: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -40,7 +39,7 @@ export const SurveyAnalysisCTA = ({
|
|||||||
environment,
|
environment,
|
||||||
isReadOnly,
|
isReadOnly,
|
||||||
user,
|
user,
|
||||||
surveyDomain,
|
publicDomain,
|
||||||
responseCount,
|
responseCount,
|
||||||
}: SurveyAnalysisCTAProps) => {
|
}: SurveyAnalysisCTAProps) => {
|
||||||
const { t } = useTranslate();
|
const { t } = useTranslate();
|
||||||
@@ -56,8 +55,7 @@ export const SurveyAnalysisCTA = ({
|
|||||||
dropdown: false,
|
dropdown: false,
|
||||||
});
|
});
|
||||||
|
|
||||||
const surveyUrl = useMemo(() => `${surveyDomain}/s/${survey.id}`, [survey.id, surveyDomain]);
|
const surveyUrl = useMemo(() => `${publicDomain}/s/${survey.id}`, [survey.id, publicDomain]);
|
||||||
const { refreshSingleUseId } = useSingleUseId(survey);
|
|
||||||
|
|
||||||
const widgetSetupCompleted = survey.type === "app" && environment.appSetupCompleted;
|
const widgetSetupCompleted = survey.type === "app" && environment.appSetupCompleted;
|
||||||
|
|
||||||
@@ -79,22 +77,6 @@ export const SurveyAnalysisCTA = ({
|
|||||||
setModalState((prev) => ({ ...prev, share: open }));
|
setModalState((prev) => ({ ...prev, share: open }));
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleCopyLink = () => {
|
|
||||||
refreshSingleUseId()
|
|
||||||
.then((newId) => {
|
|
||||||
const linkToCopy = copySurveyLink(surveyUrl, newId);
|
|
||||||
return navigator.clipboard.writeText(linkToCopy);
|
|
||||||
})
|
|
||||||
.then(() => {
|
|
||||||
toast.success(t("common.copied_to_clipboard"));
|
|
||||||
})
|
|
||||||
.catch((err) => {
|
|
||||||
toast.error(t("environments.surveys.summary.failed_to_copy_link"));
|
|
||||||
console.error(err);
|
|
||||||
});
|
|
||||||
setModalState((prev) => ({ ...prev, dropdown: false }));
|
|
||||||
};
|
|
||||||
|
|
||||||
const duplicateSurveyAndRoute = async (surveyId: string) => {
|
const duplicateSurveyAndRoute = async (surveyId: string) => {
|
||||||
setLoading(true);
|
setLoading(true);
|
||||||
const duplicatedSurveyResponse = await copySurveyToOtherEnvironmentAction({
|
const duplicatedSurveyResponse = await copySurveyToOtherEnvironmentAction({
|
||||||
@@ -134,24 +116,6 @@ export const SurveyAnalysisCTA = ({
|
|||||||
const [isCautionDialogOpen, setIsCautionDialogOpen] = useState(false);
|
const [isCautionDialogOpen, setIsCautionDialogOpen] = useState(false);
|
||||||
|
|
||||||
const iconActions = [
|
const iconActions = [
|
||||||
{
|
|
||||||
icon: Eye,
|
|
||||||
tooltip: t("common.preview"),
|
|
||||||
onClick: () => window.open(getPreviewUrl(), "_blank"),
|
|
||||||
isVisible: survey.type === "link",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
icon: LinkIcon,
|
|
||||||
tooltip: t("common.copy_link"),
|
|
||||||
onClick: handleCopyLink,
|
|
||||||
isVisible: survey.type === "link",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
icon: Code2Icon,
|
|
||||||
tooltip: t("common.embed"),
|
|
||||||
onClick: () => handleModalState("embed")(true),
|
|
||||||
isVisible: !isReadOnly,
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
icon: BellRing,
|
icon: BellRing,
|
||||||
tooltip: t("environments.surveys.summary.configure_alerts"),
|
tooltip: t("environments.surveys.summary.configure_alerts"),
|
||||||
@@ -159,13 +123,10 @@ export const SurveyAnalysisCTA = ({
|
|||||||
isVisible: !isReadOnly,
|
isVisible: !isReadOnly,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
icon: UsersRound,
|
icon: Eye,
|
||||||
tooltip: t("environments.surveys.summary.send_to_panel"),
|
tooltip: t("common.preview"),
|
||||||
onClick: () => {
|
onClick: () => window.open(getPreviewUrl(), "_blank"),
|
||||||
handleModalState("panel")(true);
|
isVisible: survey.type === "link",
|
||||||
setModalState((prev) => ({ ...prev, dropdown: false }));
|
|
||||||
},
|
|
||||||
isVisible: !isReadOnly,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
icon: SquarePenIcon,
|
icon: SquarePenIcon,
|
||||||
@@ -195,6 +156,13 @@ export const SurveyAnalysisCTA = ({
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
<IconBar actions={iconActions} />
|
<IconBar actions={iconActions} />
|
||||||
|
<Button
|
||||||
|
className="h-10"
|
||||||
|
onClick={() => {
|
||||||
|
setModalState((prev) => ({ ...prev, embed: true }));
|
||||||
|
}}>
|
||||||
|
{t("environments.surveys.summary.share_survey")}
|
||||||
|
</Button>
|
||||||
|
|
||||||
{user && (
|
{user && (
|
||||||
<>
|
<>
|
||||||
@@ -202,7 +170,7 @@ export const SurveyAnalysisCTA = ({
|
|||||||
<ShareEmbedSurvey
|
<ShareEmbedSurvey
|
||||||
key={key}
|
key={key}
|
||||||
survey={survey}
|
survey={survey}
|
||||||
surveyDomain={surveyDomain}
|
publicDomain={publicDomain}
|
||||||
open={modalState[key as keyof ModalState]}
|
open={modalState[key as keyof ModalState]}
|
||||||
setOpen={setOpen}
|
setOpen={setOpen}
|
||||||
user={user}
|
user={user}
|
||||||
|
|||||||
+1
-1
@@ -64,7 +64,7 @@ const defaultProps = {
|
|||||||
survey: mockSurveyLink,
|
survey: mockSurveyLink,
|
||||||
email: "test@example.com",
|
email: "test@example.com",
|
||||||
surveyUrl: "http://example.com/survey1",
|
surveyUrl: "http://example.com/survey1",
|
||||||
surveyDomain: "http://example.com",
|
publicDomain: "http://example.com",
|
||||||
setSurveyUrl: vi.fn(),
|
setSurveyUrl: vi.fn(),
|
||||||
locale: "en" as any,
|
locale: "en" as any,
|
||||||
disableBack: false,
|
disableBack: false,
|
||||||
|
|||||||
+3
-3
@@ -20,7 +20,7 @@ interface EmbedViewProps {
|
|||||||
survey: any;
|
survey: any;
|
||||||
email: string;
|
email: string;
|
||||||
surveyUrl: string;
|
surveyUrl: string;
|
||||||
surveyDomain: string;
|
publicDomain: string;
|
||||||
setSurveyUrl: React.Dispatch<React.SetStateAction<string>>;
|
setSurveyUrl: React.Dispatch<React.SetStateAction<string>>;
|
||||||
locale: TUserLocale;
|
locale: TUserLocale;
|
||||||
}
|
}
|
||||||
@@ -35,7 +35,7 @@ export const EmbedView = ({
|
|||||||
survey,
|
survey,
|
||||||
email,
|
email,
|
||||||
surveyUrl,
|
surveyUrl,
|
||||||
surveyDomain,
|
publicDomain,
|
||||||
setSurveyUrl,
|
setSurveyUrl,
|
||||||
locale,
|
locale,
|
||||||
}: EmbedViewProps) => {
|
}: EmbedViewProps) => {
|
||||||
@@ -83,7 +83,7 @@ export const EmbedView = ({
|
|||||||
<LinkTab
|
<LinkTab
|
||||||
survey={survey}
|
survey={survey}
|
||||||
surveyUrl={surveyUrl}
|
surveyUrl={surveyUrl}
|
||||||
surveyDomain={surveyDomain}
|
publicDomain={publicDomain}
|
||||||
setSurveyUrl={setSurveyUrl}
|
setSurveyUrl={setSurveyUrl}
|
||||||
locale={locale}
|
locale={locale}
|
||||||
/>
|
/>
|
||||||
|
|||||||
+8
-8
@@ -6,12 +6,12 @@ import { LinkTab } from "./LinkTab";
|
|||||||
|
|
||||||
// Mock ShareSurveyLink
|
// Mock ShareSurveyLink
|
||||||
vi.mock("@/modules/analysis/components/ShareSurveyLink", () => ({
|
vi.mock("@/modules/analysis/components/ShareSurveyLink", () => ({
|
||||||
ShareSurveyLink: vi.fn(({ survey, surveyUrl, surveyDomain, locale }) => (
|
ShareSurveyLink: vi.fn(({ survey, surveyUrl, publicDomain, locale }) => (
|
||||||
<div data-testid="share-survey-link">
|
<div data-testid="share-survey-link">
|
||||||
Mocked ShareSurveyLink
|
Mocked ShareSurveyLink
|
||||||
<span data-testid="survey-id">{survey.id}</span>
|
<span data-testid="survey-id">{survey.id}</span>
|
||||||
<span data-testid="survey-url">{surveyUrl}</span>
|
<span data-testid="survey-url">{surveyUrl}</span>
|
||||||
<span data-testid="survey-domain">{surveyDomain}</span>
|
<span data-testid="public-domain">{publicDomain}</span>
|
||||||
<span data-testid="locale">{locale}</span>
|
<span data-testid="locale">{locale}</span>
|
||||||
</div>
|
</div>
|
||||||
)),
|
)),
|
||||||
@@ -49,7 +49,7 @@ const mockSurvey: TSurvey = {
|
|||||||
} as unknown as TSurvey;
|
} as unknown as TSurvey;
|
||||||
|
|
||||||
const mockSurveyUrl = "https://app.formbricks.com/s/survey1";
|
const mockSurveyUrl = "https://app.formbricks.com/s/survey1";
|
||||||
const mockSurveyDomain = "https://app.formbricks.com";
|
const mockPublicDomain = "https://app.formbricks.com";
|
||||||
const mockSetSurveyUrl = vi.fn();
|
const mockSetSurveyUrl = vi.fn();
|
||||||
const mockLocale: TUserLocale = "en-US";
|
const mockLocale: TUserLocale = "en-US";
|
||||||
|
|
||||||
@@ -82,7 +82,7 @@ describe("LinkTab", () => {
|
|||||||
<LinkTab
|
<LinkTab
|
||||||
survey={mockSurvey}
|
survey={mockSurvey}
|
||||||
surveyUrl={mockSurveyUrl}
|
surveyUrl={mockSurveyUrl}
|
||||||
surveyDomain={mockSurveyDomain}
|
publicDomain={mockPublicDomain}
|
||||||
setSurveyUrl={mockSetSurveyUrl}
|
setSurveyUrl={mockSetSurveyUrl}
|
||||||
locale={mockLocale}
|
locale={mockLocale}
|
||||||
/>
|
/>
|
||||||
@@ -97,7 +97,7 @@ describe("LinkTab", () => {
|
|||||||
<LinkTab
|
<LinkTab
|
||||||
survey={mockSurvey}
|
survey={mockSurvey}
|
||||||
surveyUrl={mockSurveyUrl}
|
surveyUrl={mockSurveyUrl}
|
||||||
surveyDomain={mockSurveyDomain}
|
publicDomain={mockPublicDomain}
|
||||||
setSurveyUrl={mockSetSurveyUrl}
|
setSurveyUrl={mockSetSurveyUrl}
|
||||||
locale={mockLocale}
|
locale={mockLocale}
|
||||||
/>
|
/>
|
||||||
@@ -105,7 +105,7 @@ describe("LinkTab", () => {
|
|||||||
expect(screen.getByTestId("share-survey-link")).toBeInTheDocument();
|
expect(screen.getByTestId("share-survey-link")).toBeInTheDocument();
|
||||||
expect(screen.getByTestId("survey-id")).toHaveTextContent(mockSurvey.id);
|
expect(screen.getByTestId("survey-id")).toHaveTextContent(mockSurvey.id);
|
||||||
expect(screen.getByTestId("survey-url")).toHaveTextContent(mockSurveyUrl);
|
expect(screen.getByTestId("survey-url")).toHaveTextContent(mockSurveyUrl);
|
||||||
expect(screen.getByTestId("survey-domain")).toHaveTextContent(mockSurveyDomain);
|
expect(screen.getByTestId("public-domain")).toHaveTextContent(mockPublicDomain);
|
||||||
expect(screen.getByTestId("locale")).toHaveTextContent(mockLocale);
|
expect(screen.getByTestId("locale")).toHaveTextContent(mockLocale);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -114,7 +114,7 @@ describe("LinkTab", () => {
|
|||||||
<LinkTab
|
<LinkTab
|
||||||
survey={mockSurvey}
|
survey={mockSurvey}
|
||||||
surveyUrl={mockSurveyUrl}
|
surveyUrl={mockSurveyUrl}
|
||||||
surveyDomain={mockSurveyDomain}
|
publicDomain={mockPublicDomain}
|
||||||
setSurveyUrl={mockSetSurveyUrl}
|
setSurveyUrl={mockSetSurveyUrl}
|
||||||
locale={mockLocale}
|
locale={mockLocale}
|
||||||
/>
|
/>
|
||||||
@@ -129,7 +129,7 @@ describe("LinkTab", () => {
|
|||||||
<LinkTab
|
<LinkTab
|
||||||
survey={mockSurvey}
|
survey={mockSurvey}
|
||||||
surveyUrl={mockSurveyUrl}
|
surveyUrl={mockSurveyUrl}
|
||||||
surveyDomain={mockSurveyDomain}
|
publicDomain={mockPublicDomain}
|
||||||
setSurveyUrl={mockSetSurveyUrl}
|
setSurveyUrl={mockSetSurveyUrl}
|
||||||
locale={mockLocale}
|
locale={mockLocale}
|
||||||
/>
|
/>
|
||||||
|
|||||||
+3
-3
@@ -9,12 +9,12 @@ import { TUserLocale } from "@formbricks/types/user";
|
|||||||
interface LinkTabProps {
|
interface LinkTabProps {
|
||||||
survey: TSurvey;
|
survey: TSurvey;
|
||||||
surveyUrl: string;
|
surveyUrl: string;
|
||||||
surveyDomain: string;
|
publicDomain: string;
|
||||||
setSurveyUrl: (url: string) => void;
|
setSurveyUrl: (url: string) => void;
|
||||||
locale: TUserLocale;
|
locale: TUserLocale;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const LinkTab = ({ survey, surveyUrl, surveyDomain, setSurveyUrl, locale }: LinkTabProps) => {
|
export const LinkTab = ({ survey, surveyUrl, publicDomain, setSurveyUrl, locale }: LinkTabProps) => {
|
||||||
const { t } = useTranslate();
|
const { t } = useTranslate();
|
||||||
|
|
||||||
const docsLinks = [
|
const docsLinks = [
|
||||||
@@ -44,7 +44,7 @@ export const LinkTab = ({ survey, surveyUrl, surveyDomain, setSurveyUrl, locale
|
|||||||
<ShareSurveyLink
|
<ShareSurveyLink
|
||||||
survey={survey}
|
survey={survey}
|
||||||
surveyUrl={surveyUrl}
|
surveyUrl={surveyUrl}
|
||||||
surveyDomain={surveyDomain}
|
publicDomain={publicDomain}
|
||||||
setSurveyUrl={setSurveyUrl}
|
setSurveyUrl={setSurveyUrl}
|
||||||
locale={locale}
|
locale={locale}
|
||||||
/>
|
/>
|
||||||
|
|||||||
+15
-7
@@ -1,9 +1,8 @@
|
|||||||
import { getSurveyDomain } from "@/lib/getSurveyUrl";
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
||||||
import { getSurvey } from "@/lib/survey/service";
|
import { getSurvey } from "@/lib/survey/service";
|
||||||
import { getStyling } from "@/lib/utils/styling";
|
import { getStyling } from "@/lib/utils/styling";
|
||||||
import { getPreviewEmailTemplateHtml } from "@/modules/email/components/preview-email-template";
|
import { getPreviewEmailTemplateHtml } from "@/modules/email/components/preview-email-template";
|
||||||
import { getTranslate } from "@/tolgee/server";
|
|
||||||
import { cleanup } from "@testing-library/react";
|
import { cleanup } from "@testing-library/react";
|
||||||
import { afterEach, beforeEach, describe, expect, test, vi } from "vitest";
|
import { afterEach, beforeEach, describe, expect, test, vi } from "vitest";
|
||||||
import { TEnvironment } from "@formbricks/types/environment";
|
import { TEnvironment } from "@formbricks/types/environment";
|
||||||
@@ -35,7 +34,16 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
SENTRY_DSN: "mock-sentry-dsn",
|
SENTRY_DSN: "mock-sentry-dsn",
|
||||||
}));
|
}));
|
||||||
|
|
||||||
vi.mock("@/lib/getSurveyUrl");
|
vi.mock("@/lib/env", () => ({
|
||||||
|
env: {
|
||||||
|
PUBLIC_URL: "https://public-domain.com",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/getPublicUrl", () => ({
|
||||||
|
getPublicDomain: vi.fn().mockReturnValue("https://public-domain.com"),
|
||||||
|
}));
|
||||||
|
|
||||||
vi.mock("@/lib/project/service");
|
vi.mock("@/lib/project/service");
|
||||||
vi.mock("@/lib/survey/service");
|
vi.mock("@/lib/survey/service");
|
||||||
vi.mock("@/lib/utils/styling");
|
vi.mock("@/lib/utils/styling");
|
||||||
@@ -121,7 +129,7 @@ const mockComputedStyling = {
|
|||||||
thankYouCardIconBgColor: "#DDDDDD",
|
thankYouCardIconBgColor: "#DDDDDD",
|
||||||
} as any;
|
} as any;
|
||||||
|
|
||||||
const mockSurveyDomain = "https://app.formbricks.com";
|
const mockPublicDomain = "https://app.formbricks.com";
|
||||||
const mockRawHtml = `${doctype}<html><body>Test Email Content for ${mockSurvey.name}</body></html>`;
|
const mockRawHtml = `${doctype}<html><body>Test Email Content for ${mockSurvey.name}</body></html>`;
|
||||||
const mockCleanedHtml = `<html><body>Test Email Content for ${mockSurvey.name}</body></html>`;
|
const mockCleanedHtml = `<html><body>Test Email Content for ${mockSurvey.name}</body></html>`;
|
||||||
|
|
||||||
@@ -136,7 +144,7 @@ describe("getEmailTemplateHtml", () => {
|
|||||||
vi.mocked(getSurvey).mockResolvedValue(mockSurvey);
|
vi.mocked(getSurvey).mockResolvedValue(mockSurvey);
|
||||||
vi.mocked(getProjectByEnvironmentId).mockResolvedValue(mockProject);
|
vi.mocked(getProjectByEnvironmentId).mockResolvedValue(mockProject);
|
||||||
vi.mocked(getStyling).mockReturnValue(mockComputedStyling);
|
vi.mocked(getStyling).mockReturnValue(mockComputedStyling);
|
||||||
vi.mocked(getSurveyDomain).mockReturnValue(mockSurveyDomain);
|
vi.mocked(getPublicDomain).mockReturnValue(mockPublicDomain);
|
||||||
vi.mocked(getPreviewEmailTemplateHtml).mockResolvedValue(mockRawHtml);
|
vi.mocked(getPreviewEmailTemplateHtml).mockResolvedValue(mockRawHtml);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -147,8 +155,8 @@ describe("getEmailTemplateHtml", () => {
|
|||||||
expect(getSurvey).toHaveBeenCalledWith(mockSurveyId);
|
expect(getSurvey).toHaveBeenCalledWith(mockSurveyId);
|
||||||
expect(getProjectByEnvironmentId).toHaveBeenCalledWith(mockSurvey.environmentId);
|
expect(getProjectByEnvironmentId).toHaveBeenCalledWith(mockSurvey.environmentId);
|
||||||
expect(getStyling).toHaveBeenCalledWith(mockProject, mockSurvey);
|
expect(getStyling).toHaveBeenCalledWith(mockProject, mockSurvey);
|
||||||
expect(getSurveyDomain).toHaveBeenCalledTimes(1);
|
expect(getPublicDomain).toHaveBeenCalledTimes(1);
|
||||||
const expectedSurveyUrl = `${mockSurveyDomain}/s/${mockSurvey.id}`;
|
const expectedSurveyUrl = `${mockPublicDomain}/s/${mockSurvey.id}`;
|
||||||
expect(getPreviewEmailTemplateHtml).toHaveBeenCalledWith(
|
expect(getPreviewEmailTemplateHtml).toHaveBeenCalledWith(
|
||||||
mockSurvey,
|
mockSurvey,
|
||||||
expectedSurveyUrl,
|
expectedSurveyUrl,
|
||||||
|
|||||||
+2
-2
@@ -1,4 +1,4 @@
|
|||||||
import { getSurveyDomain } from "@/lib/getSurveyUrl";
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
||||||
import { getSurvey } from "@/lib/survey/service";
|
import { getSurvey } from "@/lib/survey/service";
|
||||||
import { getStyling } from "@/lib/utils/styling";
|
import { getStyling } from "@/lib/utils/styling";
|
||||||
@@ -17,7 +17,7 @@ export const getEmailTemplateHtml = async (surveyId: string, locale: string) =>
|
|||||||
}
|
}
|
||||||
|
|
||||||
const styling = getStyling(project, survey);
|
const styling = getStyling(project, survey);
|
||||||
const surveyUrl = getSurveyDomain() + "/s/" + survey.id;
|
const surveyUrl = getPublicDomain() + "/s/" + survey.id;
|
||||||
const html = await getPreviewEmailTemplateHtml(survey, surveyUrl, styling, locale, t);
|
const html = await getPreviewEmailTemplateHtml(survey, surveyUrl, styling, locale, t);
|
||||||
const doctype =
|
const doctype =
|
||||||
'<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">';
|
'<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">';
|
||||||
|
|||||||
+7
-8
@@ -3,8 +3,8 @@ import { SurveyAnalysisNavigation } from "@/app/(app)/environments/[environmentI
|
|||||||
import { SummaryPage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SummaryPage";
|
import { SummaryPage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SummaryPage";
|
||||||
import { getSurveySummary } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/lib/surveySummary";
|
import { getSurveySummary } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/lib/surveySummary";
|
||||||
import SurveyPage from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/page";
|
import SurveyPage from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/page";
|
||||||
import { DEFAULT_LOCALE, WEBAPP_URL } from "@/lib/constants";
|
import { DEFAULT_LOCALE } from "@/lib/constants";
|
||||||
import { getSurveyDomain } from "@/lib/getSurveyUrl";
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getResponseCountBySurveyId } from "@/lib/response/service";
|
import { getResponseCountBySurveyId } from "@/lib/response/service";
|
||||||
import { getSurvey } from "@/lib/survey/service";
|
import { getSurvey } from "@/lib/survey/service";
|
||||||
import { getUser } from "@/lib/user/service";
|
import { getUser } from "@/lib/user/service";
|
||||||
@@ -38,7 +38,6 @@ vi.mock("@/lib/constants", () => ({
|
|||||||
OIDC_SIGNING_ALGORITHM: "test-oidc-signing-algorithm",
|
OIDC_SIGNING_ALGORITHM: "test-oidc-signing-algorithm",
|
||||||
IS_PRODUCTION: false,
|
IS_PRODUCTION: false,
|
||||||
SENTRY_DSN: "mock-sentry-dsn",
|
SENTRY_DSN: "mock-sentry-dsn",
|
||||||
WEBAPP_URL: "http://localhost:3000",
|
|
||||||
RESPONSES_PER_PAGE: 10,
|
RESPONSES_PER_PAGE: 10,
|
||||||
SESSION_MAX_AGE: 1000,
|
SESSION_MAX_AGE: 1000,
|
||||||
}));
|
}));
|
||||||
@@ -64,8 +63,8 @@ vi.mock(
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
vi.mock("@/lib/getSurveyUrl", () => ({
|
vi.mock("@/lib/getPublicUrl", () => ({
|
||||||
getSurveyDomain: vi.fn(),
|
getPublicDomain: vi.fn(),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
vi.mock("@/lib/response/service", () => ({
|
vi.mock("@/lib/response/service", () => ({
|
||||||
@@ -211,7 +210,7 @@ describe("SurveyPage", () => {
|
|||||||
vi.mocked(getSurvey).mockResolvedValue(mockSurvey);
|
vi.mocked(getSurvey).mockResolvedValue(mockSurvey);
|
||||||
vi.mocked(getUser).mockResolvedValue(mockUser);
|
vi.mocked(getUser).mockResolvedValue(mockUser);
|
||||||
vi.mocked(getResponseCountBySurveyId).mockResolvedValue(10);
|
vi.mocked(getResponseCountBySurveyId).mockResolvedValue(10);
|
||||||
vi.mocked(getSurveyDomain).mockReturnValue("test.domain.com");
|
vi.mocked(getPublicDomain).mockReturnValue("http://localhost:3000");
|
||||||
vi.mocked(getSurveySummary).mockResolvedValue(mockSurveySummary);
|
vi.mocked(getSurveySummary).mockResolvedValue(mockSurveySummary);
|
||||||
vi.mocked(notFound).mockClear();
|
vi.mocked(notFound).mockClear();
|
||||||
});
|
});
|
||||||
@@ -235,7 +234,7 @@ describe("SurveyPage", () => {
|
|||||||
expect(vi.mocked(getEnvironmentAuth)).toHaveBeenCalledWith(mockEnvironmentId);
|
expect(vi.mocked(getEnvironmentAuth)).toHaveBeenCalledWith(mockEnvironmentId);
|
||||||
expect(vi.mocked(getSurvey)).toHaveBeenCalledWith(mockSurveyId);
|
expect(vi.mocked(getSurvey)).toHaveBeenCalledWith(mockSurveyId);
|
||||||
expect(vi.mocked(getUser)).toHaveBeenCalledWith(mockUserId);
|
expect(vi.mocked(getUser)).toHaveBeenCalledWith(mockUserId);
|
||||||
expect(vi.mocked(getSurveyDomain)).toHaveBeenCalled();
|
expect(vi.mocked(getPublicDomain)).toHaveBeenCalled();
|
||||||
|
|
||||||
expect(vi.mocked(SurveyAnalysisNavigation).mock.calls[0][0]).toEqual(
|
expect(vi.mocked(SurveyAnalysisNavigation).mock.calls[0][0]).toEqual(
|
||||||
expect.objectContaining({
|
expect.objectContaining({
|
||||||
@@ -250,7 +249,7 @@ describe("SurveyPage", () => {
|
|||||||
environment: mockEnvironment,
|
environment: mockEnvironment,
|
||||||
survey: mockSurvey,
|
survey: mockSurvey,
|
||||||
surveyId: mockSurveyId,
|
surveyId: mockSurveyId,
|
||||||
webAppUrl: WEBAPP_URL,
|
publicDomain: "http://localhost:3000",
|
||||||
isReadOnly: false,
|
isReadOnly: false,
|
||||||
locale: mockUser.locale ?? DEFAULT_LOCALE,
|
locale: mockUser.locale ?? DEFAULT_LOCALE,
|
||||||
initialSurveySummary: mockSurveySummary,
|
initialSurveySummary: mockSurveySummary,
|
||||||
|
|||||||
+5
-5
@@ -2,8 +2,8 @@ import { SurveyAnalysisNavigation } from "@/app/(app)/environments/[environmentI
|
|||||||
import { SummaryPage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SummaryPage";
|
import { SummaryPage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SummaryPage";
|
||||||
import { SurveyAnalysisCTA } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SurveyAnalysisCTA";
|
import { SurveyAnalysisCTA } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SurveyAnalysisCTA";
|
||||||
import { getSurveySummary } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/lib/surveySummary";
|
import { getSurveySummary } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/lib/surveySummary";
|
||||||
import { DEFAULT_LOCALE, WEBAPP_URL } from "@/lib/constants";
|
import { DEFAULT_LOCALE } from "@/lib/constants";
|
||||||
import { getSurveyDomain } from "@/lib/getSurveyUrl";
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getSurvey } from "@/lib/survey/service";
|
import { getSurvey } from "@/lib/survey/service";
|
||||||
import { getUser } from "@/lib/user/service";
|
import { getUser } from "@/lib/user/service";
|
||||||
import { getEnvironmentAuth } from "@/modules/environments/lib/utils";
|
import { getEnvironmentAuth } from "@/modules/environments/lib/utils";
|
||||||
@@ -40,7 +40,7 @@ const SurveyPage = async (props: { params: Promise<{ environmentId: string; surv
|
|||||||
// Fetch initial survey summary data on the server to prevent duplicate API calls during hydration
|
// Fetch initial survey summary data on the server to prevent duplicate API calls during hydration
|
||||||
const initialSurveySummary = await getSurveySummary(surveyId);
|
const initialSurveySummary = await getSurveySummary(surveyId);
|
||||||
|
|
||||||
const surveyDomain = getSurveyDomain();
|
const publicDomain = getPublicDomain();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<PageContentWrapper>
|
<PageContentWrapper>
|
||||||
@@ -52,7 +52,7 @@ const SurveyPage = async (props: { params: Promise<{ environmentId: string; surv
|
|||||||
survey={survey}
|
survey={survey}
|
||||||
isReadOnly={isReadOnly}
|
isReadOnly={isReadOnly}
|
||||||
user={user}
|
user={user}
|
||||||
surveyDomain={surveyDomain}
|
publicDomain={publicDomain}
|
||||||
responseCount={initialSurveySummary?.meta.totalResponses ?? 0}
|
responseCount={initialSurveySummary?.meta.totalResponses ?? 0}
|
||||||
/>
|
/>
|
||||||
}>
|
}>
|
||||||
@@ -62,7 +62,7 @@ const SurveyPage = async (props: { params: Promise<{ environmentId: string; surv
|
|||||||
environment={environment}
|
environment={environment}
|
||||||
survey={survey}
|
survey={survey}
|
||||||
surveyId={params.surveyId}
|
surveyId={params.surveyId}
|
||||||
webAppUrl={WEBAPP_URL}
|
publicDomain={publicDomain}
|
||||||
isReadOnly={isReadOnly}
|
isReadOnly={isReadOnly}
|
||||||
locale={user.locale ?? DEFAULT_LOCALE}
|
locale={user.locale ?? DEFAULT_LOCALE}
|
||||||
initialSurveySummary={initialSurveySummary}
|
initialSurveySummary={initialSurveySummary}
|
||||||
|
|||||||
+10
-6
@@ -138,7 +138,7 @@ describe("ResultsShareButton", () => {
|
|||||||
|
|
||||||
test("renders initial state and fetches sharing key (no existing key)", async () => {
|
test("renders initial state and fetches sharing key (no existing key)", async () => {
|
||||||
mockGetResultShareUrlAction.mockResolvedValue({ data: null });
|
mockGetResultShareUrlAction.mockResolvedValue({ data: null });
|
||||||
render(<ResultsShareButton survey={mockSurvey} webAppUrl={webAppUrl} />);
|
render(<ResultsShareButton survey={mockSurvey} publicDomain={webAppUrl} />);
|
||||||
|
|
||||||
expect(screen.getByTestId("dropdown-menu-trigger")).toBeInTheDocument();
|
expect(screen.getByTestId("dropdown-menu-trigger")).toBeInTheDocument();
|
||||||
expect(screen.getByTestId("link-icon")).toBeInTheDocument();
|
expect(screen.getByTestId("link-icon")).toBeInTheDocument();
|
||||||
@@ -150,7 +150,7 @@ describe("ResultsShareButton", () => {
|
|||||||
|
|
||||||
test("handles copy private link to clipboard", async () => {
|
test("handles copy private link to clipboard", async () => {
|
||||||
mockGetResultShareUrlAction.mockResolvedValue({ data: null });
|
mockGetResultShareUrlAction.mockResolvedValue({ data: null });
|
||||||
render(<ResultsShareButton survey={mockSurvey} webAppUrl={webAppUrl} />);
|
render(<ResultsShareButton survey={mockSurvey} publicDomain={webAppUrl} />);
|
||||||
|
|
||||||
fireEvent.click(screen.getByTestId("dropdown-menu-trigger")); // Open dropdown
|
fireEvent.click(screen.getByTestId("dropdown-menu-trigger")); // Open dropdown
|
||||||
const copyLinkButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
const copyLinkButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
||||||
@@ -166,7 +166,9 @@ describe("ResultsShareButton", () => {
|
|||||||
test("handles copy public link to clipboard", async () => {
|
test("handles copy public link to clipboard", async () => {
|
||||||
const shareKey = "publicShareKey";
|
const shareKey = "publicShareKey";
|
||||||
mockGetResultShareUrlAction.mockResolvedValue({ data: shareKey });
|
mockGetResultShareUrlAction.mockResolvedValue({ data: shareKey });
|
||||||
render(<ResultsShareButton survey={{ ...mockSurvey, resultShareKey: shareKey }} webAppUrl={webAppUrl} />);
|
render(
|
||||||
|
<ResultsShareButton survey={{ ...mockSurvey, resultShareKey: shareKey }} publicDomain={webAppUrl} />
|
||||||
|
);
|
||||||
|
|
||||||
fireEvent.click(screen.getByTestId("dropdown-menu-trigger")); // Open dropdown
|
fireEvent.click(screen.getByTestId("dropdown-menu-trigger")); // Open dropdown
|
||||||
const copyPublicLinkButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
const copyPublicLinkButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
||||||
@@ -184,7 +186,7 @@ describe("ResultsShareButton", () => {
|
|||||||
test("handles publish to web successfully", async () => {
|
test("handles publish to web successfully", async () => {
|
||||||
mockGetResultShareUrlAction.mockResolvedValue({ data: null });
|
mockGetResultShareUrlAction.mockResolvedValue({ data: null });
|
||||||
mockGenerateResultShareUrlAction.mockResolvedValue({ data: "newShareKey" });
|
mockGenerateResultShareUrlAction.mockResolvedValue({ data: "newShareKey" });
|
||||||
render(<ResultsShareButton survey={mockSurvey} webAppUrl={webAppUrl} />);
|
render(<ResultsShareButton survey={mockSurvey} publicDomain={webAppUrl} />);
|
||||||
|
|
||||||
fireEvent.click(screen.getByTestId("dropdown-menu-trigger"));
|
fireEvent.click(screen.getByTestId("dropdown-menu-trigger"));
|
||||||
const publishButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
const publishButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
||||||
@@ -210,7 +212,9 @@ describe("ResultsShareButton", () => {
|
|||||||
const shareKey = "toUnpublishKey";
|
const shareKey = "toUnpublishKey";
|
||||||
mockGetResultShareUrlAction.mockResolvedValue({ data: shareKey });
|
mockGetResultShareUrlAction.mockResolvedValue({ data: shareKey });
|
||||||
mockDeleteResultShareUrlAction.mockResolvedValue({ data: { id: mockSurvey.id } });
|
mockDeleteResultShareUrlAction.mockResolvedValue({ data: { id: mockSurvey.id } });
|
||||||
render(<ResultsShareButton survey={{ ...mockSurvey, resultShareKey: shareKey }} webAppUrl={webAppUrl} />);
|
render(
|
||||||
|
<ResultsShareButton survey={{ ...mockSurvey, resultShareKey: shareKey }} publicDomain={webAppUrl} />
|
||||||
|
);
|
||||||
|
|
||||||
fireEvent.click(screen.getByTestId("dropdown-menu-trigger"));
|
fireEvent.click(screen.getByTestId("dropdown-menu-trigger"));
|
||||||
const unpublishButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
const unpublishButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
||||||
@@ -234,7 +238,7 @@ describe("ResultsShareButton", () => {
|
|||||||
|
|
||||||
test("opens and closes ShareSurveyResults modal", async () => {
|
test("opens and closes ShareSurveyResults modal", async () => {
|
||||||
mockGetResultShareUrlAction.mockResolvedValue({ data: null });
|
mockGetResultShareUrlAction.mockResolvedValue({ data: null });
|
||||||
render(<ResultsShareButton survey={mockSurvey} webAppUrl={webAppUrl} />);
|
render(<ResultsShareButton survey={mockSurvey} publicDomain={webAppUrl} />);
|
||||||
|
|
||||||
fireEvent.click(screen.getByTestId("dropdown-menu-trigger"));
|
fireEvent.click(screen.getByTestId("dropdown-menu-trigger"));
|
||||||
const publishButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
const publishButton = (await screen.findAllByTestId("dropdown-menu-item")).find((item) =>
|
||||||
|
|||||||
+5
-5
@@ -21,10 +21,10 @@ import { ShareSurveyResults } from "../(analysis)/summary/components/ShareSurvey
|
|||||||
|
|
||||||
interface ResultsShareButtonProps {
|
interface ResultsShareButtonProps {
|
||||||
survey: TSurvey;
|
survey: TSurvey;
|
||||||
webAppUrl: string;
|
publicDomain: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const ResultsShareButton = ({ survey, webAppUrl }: ResultsShareButtonProps) => {
|
export const ResultsShareButton = ({ survey, publicDomain }: ResultsShareButtonProps) => {
|
||||||
const { t } = useTranslate();
|
const { t } = useTranslate();
|
||||||
const [showResultsLinkModal, setShowResultsLinkModal] = useState(false);
|
const [showResultsLinkModal, setShowResultsLinkModal] = useState(false);
|
||||||
|
|
||||||
@@ -34,7 +34,7 @@ export const ResultsShareButton = ({ survey, webAppUrl }: ResultsShareButtonProp
|
|||||||
const handlePublish = async () => {
|
const handlePublish = async () => {
|
||||||
const resultShareKeyResponse = await generateResultShareUrlAction({ surveyId: survey.id });
|
const resultShareKeyResponse = await generateResultShareUrlAction({ surveyId: survey.id });
|
||||||
if (resultShareKeyResponse?.data) {
|
if (resultShareKeyResponse?.data) {
|
||||||
setSurveyUrl(webAppUrl + "/share/" + resultShareKeyResponse.data);
|
setSurveyUrl(publicDomain + "/share/" + resultShareKeyResponse.data);
|
||||||
setShowPublishModal(true);
|
setShowPublishModal(true);
|
||||||
} else {
|
} else {
|
||||||
const errorMessage = getFormattedErrorMessage(resultShareKeyResponse);
|
const errorMessage = getFormattedErrorMessage(resultShareKeyResponse);
|
||||||
@@ -58,13 +58,13 @@ export const ResultsShareButton = ({ survey, webAppUrl }: ResultsShareButtonProp
|
|||||||
const fetchSharingKey = async () => {
|
const fetchSharingKey = async () => {
|
||||||
const resultShareUrlResponse = await getResultShareUrlAction({ surveyId: survey.id });
|
const resultShareUrlResponse = await getResultShareUrlAction({ surveyId: survey.id });
|
||||||
if (resultShareUrlResponse?.data) {
|
if (resultShareUrlResponse?.data) {
|
||||||
setSurveyUrl(webAppUrl + "/share/" + resultShareUrlResponse.data);
|
setSurveyUrl(publicDomain + "/share/" + resultShareUrlResponse.data);
|
||||||
setShowPublishModal(true);
|
setShowPublishModal(true);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
fetchSharingKey();
|
fetchSharingKey();
|
||||||
}, [survey.id, webAppUrl]);
|
}, [survey.id, publicDomain]);
|
||||||
|
|
||||||
const copyUrlToClipboard = () => {
|
const copyUrlToClipboard = () => {
|
||||||
if (typeof window !== "undefined") {
|
if (typeof window !== "undefined") {
|
||||||
|
|||||||
@@ -14,41 +14,64 @@ describe("ClientEnvironmentRedirect", () => {
|
|||||||
cleanup();
|
cleanup();
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should redirect to the provided environment ID when no last environment exists", () => {
|
test("should redirect to the first environment ID when no last environment exists", () => {
|
||||||
const mockPush = vi.fn();
|
const mockPush = vi.fn();
|
||||||
vi.mocked(useRouter).mockReturnValue({ push: mockPush } as any);
|
vi.mocked(useRouter).mockReturnValue({ push: mockPush } as any);
|
||||||
|
|
||||||
// Mock localStorage
|
// Mock localStorage
|
||||||
const localStorageMock = {
|
const localStorageMock = {
|
||||||
getItem: vi.fn().mockReturnValue(null),
|
getItem: vi.fn().mockReturnValue(null),
|
||||||
|
removeItem: vi.fn(),
|
||||||
};
|
};
|
||||||
|
|
||||||
Object.defineProperty(window, "localStorage", {
|
Object.defineProperty(window, "localStorage", {
|
||||||
value: localStorageMock,
|
value: localStorageMock,
|
||||||
});
|
});
|
||||||
|
|
||||||
render(<ClientEnvironmentRedirect environmentId="test-env-id" />);
|
render(<ClientEnvironmentRedirect userEnvironments={["test-env-id"]} />);
|
||||||
|
|
||||||
expect(mockPush).toHaveBeenCalledWith("/environments/test-env-id");
|
expect(mockPush).toHaveBeenCalledWith("/environments/test-env-id");
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should redirect to the last environment ID when it exists in localStorage", () => {
|
test("should redirect to the last environment ID when it exists in localStorage and is valid", () => {
|
||||||
const mockPush = vi.fn();
|
const mockPush = vi.fn();
|
||||||
vi.mocked(useRouter).mockReturnValue({ push: mockPush } as any);
|
vi.mocked(useRouter).mockReturnValue({ push: mockPush } as any);
|
||||||
|
|
||||||
// Mock localStorage with a last environment ID
|
// Mock localStorage with a last environment ID
|
||||||
const localStorageMock = {
|
const localStorageMock = {
|
||||||
getItem: vi.fn().mockReturnValue("last-env-id"),
|
getItem: vi.fn().mockReturnValue("last-env-id"),
|
||||||
|
removeItem: vi.fn(),
|
||||||
};
|
};
|
||||||
Object.defineProperty(window, "localStorage", {
|
Object.defineProperty(window, "localStorage", {
|
||||||
value: localStorageMock,
|
value: localStorageMock,
|
||||||
});
|
});
|
||||||
|
|
||||||
render(<ClientEnvironmentRedirect environmentId="test-env-id" />);
|
render(<ClientEnvironmentRedirect userEnvironments={["last-env-id", "other-env-id"]} />);
|
||||||
|
|
||||||
expect(localStorageMock.getItem).toHaveBeenCalledWith(FORMBRICKS_ENVIRONMENT_ID_LS);
|
expect(localStorageMock.getItem).toHaveBeenCalledWith(FORMBRICKS_ENVIRONMENT_ID_LS);
|
||||||
expect(mockPush).toHaveBeenCalledWith("/environments/last-env-id");
|
expect(mockPush).toHaveBeenCalledWith("/environments/last-env-id");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("should clear invalid environment ID and redirect to default when stored ID is not in user environments", () => {
|
||||||
|
const mockPush = vi.fn();
|
||||||
|
vi.mocked(useRouter).mockReturnValue({ push: mockPush } as any);
|
||||||
|
|
||||||
|
// Mock localStorage with an invalid environment ID
|
||||||
|
const localStorageMock = {
|
||||||
|
getItem: vi.fn().mockReturnValue("invalid-env-id"),
|
||||||
|
removeItem: vi.fn(),
|
||||||
|
};
|
||||||
|
Object.defineProperty(window, "localStorage", {
|
||||||
|
value: localStorageMock,
|
||||||
|
});
|
||||||
|
|
||||||
|
render(<ClientEnvironmentRedirect userEnvironments={["valid-env-1", "valid-env-2"]} />);
|
||||||
|
|
||||||
|
expect(localStorageMock.getItem).toHaveBeenCalledWith(FORMBRICKS_ENVIRONMENT_ID_LS);
|
||||||
|
expect(localStorageMock.removeItem).toHaveBeenCalledWith(FORMBRICKS_ENVIRONMENT_ID_LS);
|
||||||
|
expect(mockPush).toHaveBeenCalledWith("/environments/valid-env-1");
|
||||||
|
});
|
||||||
|
|
||||||
test("should update redirect when environment ID prop changes", () => {
|
test("should update redirect when environment ID prop changes", () => {
|
||||||
const mockPush = vi.fn();
|
const mockPush = vi.fn();
|
||||||
vi.mocked(useRouter).mockReturnValue({ push: mockPush } as any);
|
vi.mocked(useRouter).mockReturnValue({ push: mockPush } as any);
|
||||||
@@ -56,19 +79,20 @@ describe("ClientEnvironmentRedirect", () => {
|
|||||||
// Mock localStorage
|
// Mock localStorage
|
||||||
const localStorageMock = {
|
const localStorageMock = {
|
||||||
getItem: vi.fn().mockReturnValue(null),
|
getItem: vi.fn().mockReturnValue(null),
|
||||||
|
removeItem: vi.fn(),
|
||||||
};
|
};
|
||||||
Object.defineProperty(window, "localStorage", {
|
Object.defineProperty(window, "localStorage", {
|
||||||
value: localStorageMock,
|
value: localStorageMock,
|
||||||
});
|
});
|
||||||
|
|
||||||
const { rerender } = render(<ClientEnvironmentRedirect environmentId="initial-env-id" />);
|
const { rerender } = render(<ClientEnvironmentRedirect userEnvironments={["initial-env-id"]} />);
|
||||||
expect(mockPush).toHaveBeenCalledWith("/environments/initial-env-id");
|
expect(mockPush).toHaveBeenCalledWith("/environments/initial-env-id");
|
||||||
|
|
||||||
// Clear mock calls
|
// Clear mock calls
|
||||||
mockPush.mockClear();
|
mockPush.mockClear();
|
||||||
|
|
||||||
// Rerender with new environment ID
|
// Rerender with new environment ID
|
||||||
rerender(<ClientEnvironmentRedirect environmentId="new-env-id" />);
|
rerender(<ClientEnvironmentRedirect userEnvironments={["new-env-id"]} />);
|
||||||
expect(mockPush).toHaveBeenCalledWith("/environments/new-env-id");
|
expect(mockPush).toHaveBeenCalledWith("/environments/new-env-id");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -5,22 +5,23 @@ import { useRouter } from "next/navigation";
|
|||||||
import { useEffect } from "react";
|
import { useEffect } from "react";
|
||||||
|
|
||||||
interface ClientEnvironmentRedirectProps {
|
interface ClientEnvironmentRedirectProps {
|
||||||
environmentId: string;
|
userEnvironments: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
const ClientEnvironmentRedirect = ({ environmentId }: ClientEnvironmentRedirectProps) => {
|
const ClientEnvironmentRedirect = ({ userEnvironments }: ClientEnvironmentRedirectProps) => {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const lastEnvironmentId = localStorage.getItem(FORMBRICKS_ENVIRONMENT_ID_LS);
|
const lastEnvironmentId = localStorage.getItem(FORMBRICKS_ENVIRONMENT_ID_LS);
|
||||||
|
|
||||||
if (lastEnvironmentId) {
|
if (lastEnvironmentId && userEnvironments.includes(lastEnvironmentId)) {
|
||||||
// Redirect to the last environment the user was in
|
|
||||||
router.push(`/environments/${lastEnvironmentId}`);
|
router.push(`/environments/${lastEnvironmentId}`);
|
||||||
} else {
|
} else {
|
||||||
router.push(`/environments/${environmentId}`);
|
// If the last environmentId is not valid, remove it from localStorage and redirect to the provided environmentId
|
||||||
|
localStorage.removeItem(FORMBRICKS_ENVIRONMENT_ID_LS);
|
||||||
|
router.push(`/environments/${userEnvironments[0]}`);
|
||||||
}
|
}
|
||||||
}, [environmentId, router]);
|
}, [userEnvironments, router]);
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { responses } from "@/app/lib/api/response";
|
import { responses } from "@/app/lib/api/response";
|
||||||
import { CRON_SECRET } from "@/lib/constants";
|
import { CRON_SECRET } from "@/lib/constants";
|
||||||
|
import { env } from "@/lib/env";
|
||||||
import { captureTelemetry } from "@/lib/telemetry";
|
import { captureTelemetry } from "@/lib/telemetry";
|
||||||
import packageJson from "@/package.json";
|
import packageJson from "@/package.json";
|
||||||
import { headers } from "next/headers";
|
import { headers } from "next/headers";
|
||||||
@@ -13,6 +14,10 @@ export const POST = async () => {
|
|||||||
return responses.notAuthenticatedResponse();
|
return responses.notAuthenticatedResponse();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (env.TELEMETRY_DISABLED === "1") {
|
||||||
|
return responses.successResponse({}, true);
|
||||||
|
}
|
||||||
|
|
||||||
const [surveyCount, responseCount, userCount] = await Promise.all([
|
const [surveyCount, responseCount, userCount] = await Promise.all([
|
||||||
prisma.survey.count(),
|
prisma.survey.count(),
|
||||||
prisma.response.count(),
|
prisma.response.count(),
|
||||||
|
|||||||
@@ -274,7 +274,7 @@ describe("getEnvironmentState", () => {
|
|||||||
|
|
||||||
expect(withCache).toHaveBeenCalledWith(expect.any(Function), {
|
expect(withCache).toHaveBeenCalledWith(expect.any(Function), {
|
||||||
key: `fb:env:${environmentId}:state`,
|
key: `fb:env:${environmentId}:state`,
|
||||||
ttl: 60 * 30 * 1000, // 30 minutes in milliseconds
|
ttl: 5 * 60 * 1000, // 5 minutes in milliseconds
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -83,9 +83,8 @@ export const getEnvironmentState = async (
|
|||||||
{
|
{
|
||||||
// Use enterprise-grade cache key pattern
|
// Use enterprise-grade cache key pattern
|
||||||
key: createCacheKey.environment.state(environmentId),
|
key: createCacheKey.environment.state(environmentId),
|
||||||
// 30 minutes TTL ensures fresh data for hourly SDK checks
|
// This is a temporary fix for the invalidation issues, will be changed later with a proper solution
|
||||||
// Balances performance with freshness requirements
|
ttl: 5 * 60 * 1000, // 5 minutes in milliseconds
|
||||||
ttl: 60 * 30 * 1000, // 30 minutes in milliseconds
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
@@ -52,14 +52,6 @@ export const POST = withApiLogging(
|
|||||||
}
|
}
|
||||||
|
|
||||||
const inputValidation = ZActionClassInput.safeParse(actionClassInput);
|
const inputValidation = ZActionClassInput.safeParse(actionClassInput);
|
||||||
const environmentId = actionClassInput.environmentId;
|
|
||||||
|
|
||||||
if (!hasPermission(authentication.environmentPermissions, environmentId, "POST")) {
|
|
||||||
return {
|
|
||||||
response: responses.unauthorizedResponse(),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!inputValidation.success) {
|
if (!inputValidation.success) {
|
||||||
return {
|
return {
|
||||||
response: responses.badRequestResponse(
|
response: responses.badRequestResponse(
|
||||||
@@ -70,6 +62,14 @@ export const POST = withApiLogging(
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const environmentId = inputValidation.data.environmentId;
|
||||||
|
|
||||||
|
if (!hasPermission(authentication.environmentPermissions, environmentId, "POST")) {
|
||||||
|
return {
|
||||||
|
response: responses.unauthorizedResponse(),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
const actionClass: TActionClass = await createActionClass(environmentId, inputValidation.data);
|
const actionClass: TActionClass = await createActionClass(environmentId, inputValidation.data);
|
||||||
auditLog.targetId = actionClass.id;
|
auditLog.targetId = actionClass.id;
|
||||||
auditLog.newObject = actionClass;
|
auditLog.newObject = actionClass;
|
||||||
|
|||||||
@@ -186,6 +186,18 @@ describe("Response Lib Tests", () => {
|
|||||||
expect(logger.error).not.toHaveBeenCalled(); // Should be caught and re-thrown as DatabaseError
|
expect(logger.error).not.toHaveBeenCalled(); // Should be caught and re-thrown as DatabaseError
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("should handle RelatedRecordDoesNotExist error with specific message", async () => {
|
||||||
|
const prismaError = new Prisma.PrismaClientKnownRequestError("Related record does not exist", {
|
||||||
|
code: "P2025", // PrismaErrorType.RelatedRecordDoesNotExist
|
||||||
|
clientVersion: "2.0",
|
||||||
|
});
|
||||||
|
vi.mocked(getOrganizationByEnvironmentId).mockResolvedValue(mockOrganization);
|
||||||
|
vi.mocked(prisma.response.create).mockRejectedValue(prismaError);
|
||||||
|
|
||||||
|
await expect(createResponse(mockResponseInput)).rejects.toThrow(DatabaseError);
|
||||||
|
await expect(createResponse(mockResponseInput)).rejects.toThrow("Display ID does not exist");
|
||||||
|
});
|
||||||
|
|
||||||
test("should handle generic errors", async () => {
|
test("should handle generic errors", async () => {
|
||||||
const genericError = new Error("Something went wrong");
|
const genericError = new Error("Something went wrong");
|
||||||
vi.mocked(getOrganizationByEnvironmentId).mockResolvedValue(mockOrganization);
|
vi.mocked(getOrganizationByEnvironmentId).mockResolvedValue(mockOrganization);
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import { validateInputs } from "@/lib/utils/validate";
|
|||||||
import { Prisma } from "@prisma/client";
|
import { Prisma } from "@prisma/client";
|
||||||
import { cache as reactCache } from "react";
|
import { cache as reactCache } from "react";
|
||||||
import { prisma } from "@formbricks/database";
|
import { prisma } from "@formbricks/database";
|
||||||
|
import { PrismaErrorType } from "@formbricks/database/types/error";
|
||||||
import { logger } from "@formbricks/logger";
|
import { logger } from "@formbricks/logger";
|
||||||
import { ZId, ZOptionalNumber } from "@formbricks/types/common";
|
import { ZId, ZOptionalNumber } from "@formbricks/types/common";
|
||||||
import { TContactAttributes } from "@formbricks/types/contact-attribute";
|
import { TContactAttributes } from "@formbricks/types/contact-attribute";
|
||||||
@@ -176,6 +177,9 @@ export const createResponse = async (responseInput: TResponseInput): Promise<TRe
|
|||||||
return response;
|
return response;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
||||||
|
if (error.code === PrismaErrorType.RelatedRecordDoesNotExist) {
|
||||||
|
throw new DatabaseError("Display ID does not exist");
|
||||||
|
}
|
||||||
throw new DatabaseError(error.message);
|
throw new DatabaseError(error.message);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -149,6 +149,10 @@ export const POST = withApiLogging(
|
|||||||
return {
|
return {
|
||||||
response: responses.badRequestResponse(error.message),
|
response: responses.badRequestResponse(error.message),
|
||||||
};
|
};
|
||||||
|
} else if (error instanceof DatabaseError) {
|
||||||
|
return {
|
||||||
|
response: responses.badRequestResponse(error.message),
|
||||||
|
};
|
||||||
}
|
}
|
||||||
logger.error({ error, url: request.url }, "Error in POST /api/v1/management/responses");
|
logger.error({ error, url: request.url }, "Error in POST /api/v1/management/responses");
|
||||||
return {
|
return {
|
||||||
@@ -158,7 +162,7 @@ export const POST = withApiLogging(
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
if (error instanceof DatabaseError) {
|
if (error instanceof DatabaseError) {
|
||||||
return {
|
return {
|
||||||
response: responses.badRequestResponse(error.message),
|
response: responses.badRequestResponse("An unexpected error occurred while creating the response"),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
throw error;
|
throw error;
|
||||||
|
|||||||
@@ -0,0 +1,212 @@
|
|||||||
|
import { authenticateRequest } from "@/app/api/v1/auth";
|
||||||
|
import { responses } from "@/app/lib/api/response";
|
||||||
|
import { hasUserEnvironmentAccess } from "@/lib/environment/auth";
|
||||||
|
import { hasPermission } from "@/modules/organization/settings/api-keys/lib/utils";
|
||||||
|
import { Session } from "next-auth";
|
||||||
|
import { NextRequest } from "next/server";
|
||||||
|
import { describe, expect, test } from "vitest";
|
||||||
|
import { vi } from "vitest";
|
||||||
|
import { TAuthenticationApiKey } from "@formbricks/types/auth";
|
||||||
|
import { checkForRequiredFields } from "./utils";
|
||||||
|
import { checkAuth } from "./utils";
|
||||||
|
|
||||||
|
// Create mock response objects
|
||||||
|
const mockBadRequestResponse = new Response("Bad Request", { status: 400 });
|
||||||
|
const mockNotAuthenticatedResponse = new Response("Not authenticated", { status: 401 });
|
||||||
|
const mockUnauthorizedResponse = new Response("Unauthorized", { status: 401 });
|
||||||
|
|
||||||
|
vi.mock("@/app/api/v1/auth", () => ({
|
||||||
|
authenticateRequest: vi.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/environment/auth", () => ({
|
||||||
|
hasUserEnvironmentAccess: vi.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/modules/organization/settings/api-keys/lib/utils", () => ({
|
||||||
|
hasPermission: vi.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/app/lib/api/response", () => ({
|
||||||
|
responses: {
|
||||||
|
badRequestResponse: vi.fn(() => mockBadRequestResponse),
|
||||||
|
notAuthenticatedResponse: vi.fn(() => mockNotAuthenticatedResponse),
|
||||||
|
unauthorizedResponse: vi.fn(() => mockUnauthorizedResponse),
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("checkForRequiredFields", () => {
|
||||||
|
test("should return undefined when all required fields are present", () => {
|
||||||
|
const result = checkForRequiredFields("env-123", "image/png", "test-file.png");
|
||||||
|
expect(result).toBeUndefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return bad request response when environmentId is missing", () => {
|
||||||
|
const result = checkForRequiredFields("", "image/png", "test-file.png");
|
||||||
|
expect(responses.badRequestResponse).toHaveBeenCalledWith("environmentId is required");
|
||||||
|
expect(result).toBe(mockBadRequestResponse);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return bad request response when fileType is missing", () => {
|
||||||
|
const result = checkForRequiredFields("env-123", "", "test-file.png");
|
||||||
|
expect(responses.badRequestResponse).toHaveBeenCalledWith("contentType is required");
|
||||||
|
expect(result).toBe(mockBadRequestResponse);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return bad request response when encodedFileName is missing", () => {
|
||||||
|
const result = checkForRequiredFields("env-123", "image/png", "");
|
||||||
|
expect(responses.badRequestResponse).toHaveBeenCalledWith("fileName is required");
|
||||||
|
expect(result).toBe(mockBadRequestResponse);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return bad request response when environmentId is undefined", () => {
|
||||||
|
const result = checkForRequiredFields(undefined as any, "image/png", "test-file.png");
|
||||||
|
expect(responses.badRequestResponse).toHaveBeenCalledWith("environmentId is required");
|
||||||
|
expect(result).toBe(mockBadRequestResponse);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return bad request response when fileType is undefined", () => {
|
||||||
|
const result = checkForRequiredFields("env-123", undefined as any, "test-file.png");
|
||||||
|
expect(responses.badRequestResponse).toHaveBeenCalledWith("contentType is required");
|
||||||
|
expect(result).toBe(mockBadRequestResponse);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return bad request response when encodedFileName is undefined", () => {
|
||||||
|
const result = checkForRequiredFields("env-123", "image/png", undefined as any);
|
||||||
|
expect(responses.badRequestResponse).toHaveBeenCalledWith("fileName is required");
|
||||||
|
expect(result).toBe(mockBadRequestResponse);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("checkAuth", () => {
|
||||||
|
const environmentId = "env-123";
|
||||||
|
const mockRequest = new NextRequest("http://localhost:3000/api/test");
|
||||||
|
|
||||||
|
test("returns notAuthenticatedResponse when no session and no authentication", async () => {
|
||||||
|
vi.mocked(authenticateRequest).mockResolvedValue(null);
|
||||||
|
|
||||||
|
const result = await checkAuth(null, environmentId, mockRequest);
|
||||||
|
|
||||||
|
expect(authenticateRequest).toHaveBeenCalledWith(mockRequest);
|
||||||
|
expect(responses.notAuthenticatedResponse).toHaveBeenCalled();
|
||||||
|
expect(result).toBe(mockNotAuthenticatedResponse);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("returns unauthorizedResponse when no session and authentication lacks POST permission", async () => {
|
||||||
|
const mockAuthentication: TAuthenticationApiKey = {
|
||||||
|
type: "apiKey",
|
||||||
|
environmentPermissions: [
|
||||||
|
{
|
||||||
|
environmentId: "env-123",
|
||||||
|
permission: "read",
|
||||||
|
environmentType: "development",
|
||||||
|
projectId: "project-1",
|
||||||
|
projectName: "Project 1",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
hashedApiKey: "hashed-key",
|
||||||
|
apiKeyId: "api-key-id",
|
||||||
|
organizationId: "org-id",
|
||||||
|
organizationAccess: {
|
||||||
|
accessControl: {},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
vi.mocked(authenticateRequest).mockResolvedValue(mockAuthentication);
|
||||||
|
vi.mocked(hasPermission).mockReturnValue(false);
|
||||||
|
|
||||||
|
const result = await checkAuth(null, environmentId, mockRequest);
|
||||||
|
|
||||||
|
expect(authenticateRequest).toHaveBeenCalledWith(mockRequest);
|
||||||
|
expect(hasPermission).toHaveBeenCalledWith(
|
||||||
|
mockAuthentication.environmentPermissions,
|
||||||
|
environmentId,
|
||||||
|
"POST"
|
||||||
|
);
|
||||||
|
expect(responses.unauthorizedResponse).toHaveBeenCalled();
|
||||||
|
expect(result).toBe(mockUnauthorizedResponse);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("returns undefined when no session and authentication has POST permission", async () => {
|
||||||
|
const mockAuthentication: TAuthenticationApiKey = {
|
||||||
|
type: "apiKey",
|
||||||
|
environmentPermissions: [
|
||||||
|
{
|
||||||
|
environmentId: "env-123",
|
||||||
|
permission: "write",
|
||||||
|
environmentType: "development",
|
||||||
|
projectId: "project-1",
|
||||||
|
projectName: "Project 1",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
hashedApiKey: "hashed-key",
|
||||||
|
apiKeyId: "api-key-id",
|
||||||
|
organizationId: "org-id",
|
||||||
|
organizationAccess: {
|
||||||
|
accessControl: {},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
vi.mocked(authenticateRequest).mockResolvedValue(mockAuthentication);
|
||||||
|
vi.mocked(hasPermission).mockReturnValue(true);
|
||||||
|
|
||||||
|
const result = await checkAuth(null, environmentId, mockRequest);
|
||||||
|
|
||||||
|
expect(authenticateRequest).toHaveBeenCalledWith(mockRequest);
|
||||||
|
expect(hasPermission).toHaveBeenCalledWith(
|
||||||
|
mockAuthentication.environmentPermissions,
|
||||||
|
environmentId,
|
||||||
|
"POST"
|
||||||
|
);
|
||||||
|
expect(result).toBeUndefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("returns unauthorizedResponse when session exists but user lacks environment access", async () => {
|
||||||
|
const mockSession: Session = {
|
||||||
|
user: {
|
||||||
|
id: "user-123",
|
||||||
|
},
|
||||||
|
expires: "2024-12-31T23:59:59.999Z",
|
||||||
|
};
|
||||||
|
|
||||||
|
vi.mocked(hasUserEnvironmentAccess).mockResolvedValue(false);
|
||||||
|
|
||||||
|
const result = await checkAuth(mockSession, environmentId, mockRequest);
|
||||||
|
|
||||||
|
expect(hasUserEnvironmentAccess).toHaveBeenCalledWith("user-123", environmentId);
|
||||||
|
expect(responses.unauthorizedResponse).toHaveBeenCalled();
|
||||||
|
expect(result).toBe(mockUnauthorizedResponse);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("returns undefined when session exists and user has environment access", async () => {
|
||||||
|
const mockSession: Session = {
|
||||||
|
user: {
|
||||||
|
id: "user-123",
|
||||||
|
},
|
||||||
|
expires: "2024-12-31T23:59:59.999Z",
|
||||||
|
};
|
||||||
|
|
||||||
|
vi.mocked(hasUserEnvironmentAccess).mockResolvedValue(true);
|
||||||
|
|
||||||
|
const result = await checkAuth(mockSession, environmentId, mockRequest);
|
||||||
|
|
||||||
|
expect(hasUserEnvironmentAccess).toHaveBeenCalledWith("user-123", environmentId);
|
||||||
|
expect(result).toBeUndefined();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("does not call authenticateRequest when session exists", async () => {
|
||||||
|
const mockSession: Session = {
|
||||||
|
user: {
|
||||||
|
id: "user-123",
|
||||||
|
},
|
||||||
|
expires: "2024-12-31T23:59:59.999Z",
|
||||||
|
};
|
||||||
|
|
||||||
|
vi.mocked(hasUserEnvironmentAccess).mockResolvedValue(true);
|
||||||
|
|
||||||
|
await checkAuth(mockSession, environmentId, mockRequest);
|
||||||
|
|
||||||
|
expect(authenticateRequest).not.toHaveBeenCalled();
|
||||||
|
expect(hasUserEnvironmentAccess).toHaveBeenCalledWith("user-123", environmentId);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
import { authenticateRequest } from "@/app/api/v1/auth";
|
||||||
|
import { responses } from "@/app/lib/api/response";
|
||||||
|
import { hasUserEnvironmentAccess } from "@/lib/environment/auth";
|
||||||
|
import { hasPermission } from "@/modules/organization/settings/api-keys/lib/utils";
|
||||||
|
import { Session } from "next-auth";
|
||||||
|
import { NextRequest } from "next/server";
|
||||||
|
|
||||||
|
export const checkForRequiredFields = (
|
||||||
|
environmentId: string,
|
||||||
|
fileType: string,
|
||||||
|
encodedFileName: string
|
||||||
|
): Response | undefined => {
|
||||||
|
if (!environmentId) {
|
||||||
|
return responses.badRequestResponse("environmentId is required");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!fileType) {
|
||||||
|
return responses.badRequestResponse("contentType is required");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!encodedFileName) {
|
||||||
|
return responses.badRequestResponse("fileName is required");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const checkAuth = async (session: Session | null, environmentId: string, request: NextRequest) => {
|
||||||
|
if (!session) {
|
||||||
|
//check whether its using API key
|
||||||
|
const authentication = await authenticateRequest(request);
|
||||||
|
if (!authentication) return responses.notAuthenticatedResponse();
|
||||||
|
|
||||||
|
if (!hasPermission(authentication.environmentPermissions, environmentId, "POST")) {
|
||||||
|
return responses.unauthorizedResponse();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const isUserAuthorized = await hasUserEnvironmentAccess(session.user.id, environmentId);
|
||||||
|
if (!isUserAuthorized) {
|
||||||
|
return responses.unauthorizedResponse();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
// headers -> "Content-Type" should be present and set to a valid MIME type
|
// headers -> "Content-Type" should be present and set to a valid MIME type
|
||||||
// body -> should be a valid file object (buffer)
|
// body -> should be a valid file object (buffer)
|
||||||
// method -> PUT (to be the same as the signedUrl method)
|
// method -> PUT (to be the same as the signedUrl method)
|
||||||
|
import { checkAuth, checkForRequiredFields } from "@/app/api/v1/management/storage/lib/utils";
|
||||||
import { responses } from "@/app/lib/api/response";
|
import { responses } from "@/app/lib/api/response";
|
||||||
import { ENCRYPTION_KEY, UPLOADS_DIR } from "@/lib/constants";
|
import { ENCRYPTION_KEY, UPLOADS_DIR } from "@/lib/constants";
|
||||||
import { validateLocalSignedUrl } from "@/lib/crypto";
|
import { validateLocalSignedUrl } from "@/lib/crypto";
|
||||||
import { hasUserEnvironmentAccess } from "@/lib/environment/auth";
|
|
||||||
import { validateFile } from "@/lib/fileValidation";
|
import { validateFile } from "@/lib/fileValidation";
|
||||||
import { putFileToLocalStorage } from "@/lib/storage/service";
|
import { putFileToLocalStorage } from "@/lib/storage/service";
|
||||||
import { authOptions } from "@/modules/auth/lib/authOptions";
|
import { authOptions } from "@/modules/auth/lib/authOptions";
|
||||||
@@ -27,41 +27,17 @@ export const POST = async (req: NextRequest): Promise<Response> => {
|
|||||||
const signedTimestamp = jsonInput.timestamp as string;
|
const signedTimestamp = jsonInput.timestamp as string;
|
||||||
const environmentId = jsonInput.environmentId as string;
|
const environmentId = jsonInput.environmentId as string;
|
||||||
|
|
||||||
if (!environmentId) {
|
const requiredFieldResponse = checkForRequiredFields(environmentId, fileType, encodedFileName);
|
||||||
return responses.badRequestResponse("environmentId is required");
|
if (requiredFieldResponse) return requiredFieldResponse;
|
||||||
}
|
|
||||||
|
|
||||||
if (!fileType) {
|
if (!signedSignature || !signedUuid || !signedTimestamp) {
|
||||||
return responses.badRequestResponse("contentType is required");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!encodedFileName) {
|
|
||||||
return responses.badRequestResponse("fileName is required");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!signedSignature) {
|
|
||||||
return responses.unauthorizedResponse();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!signedUuid) {
|
|
||||||
return responses.unauthorizedResponse();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!signedTimestamp) {
|
|
||||||
return responses.unauthorizedResponse();
|
return responses.unauthorizedResponse();
|
||||||
}
|
}
|
||||||
|
|
||||||
const session = await getServerSession(authOptions);
|
const session = await getServerSession(authOptions);
|
||||||
|
|
||||||
if (!session || !session.user) {
|
const authResponse = await checkAuth(session, environmentId, req);
|
||||||
return responses.notAuthenticatedResponse();
|
if (authResponse) return authResponse;
|
||||||
}
|
|
||||||
|
|
||||||
const isUserAuthorized = await hasUserEnvironmentAccess(session.user.id, environmentId);
|
|
||||||
|
|
||||||
if (!isUserAuthorized) {
|
|
||||||
return responses.unauthorizedResponse();
|
|
||||||
}
|
|
||||||
|
|
||||||
const fileName = decodeURIComponent(encodedFileName);
|
const fileName = decodeURIComponent(encodedFileName);
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
|
import { checkAuth, checkForRequiredFields } from "@/app/api/v1/management/storage/lib/utils";
|
||||||
import { responses } from "@/app/lib/api/response";
|
import { responses } from "@/app/lib/api/response";
|
||||||
import { hasUserEnvironmentAccess } from "@/lib/environment/auth";
|
|
||||||
import { validateFile } from "@/lib/fileValidation";
|
import { validateFile } from "@/lib/fileValidation";
|
||||||
import { authOptions } from "@/modules/auth/lib/authOptions";
|
import { authOptions } from "@/modules/auth/lib/authOptions";
|
||||||
import { getServerSession } from "next-auth";
|
import { getServerSession } from "next-auth";
|
||||||
@@ -13,29 +13,24 @@ import { getSignedUrlForPublicFile } from "./lib/getSignedUrl";
|
|||||||
// use this to upload files for a specific resource, e.g. a user profile picture or a survey
|
// use this to upload files for a specific resource, e.g. a user profile picture or a survey
|
||||||
// this api endpoint will return a signed url for uploading the file to s3 and another url for uploading file to the local storage
|
// this api endpoint will return a signed url for uploading the file to s3 and another url for uploading file to the local storage
|
||||||
|
|
||||||
export const POST = async (req: NextRequest): Promise<Response> => {
|
export const POST = async (request: NextRequest): Promise<Response> => {
|
||||||
let storageInput;
|
let storageInput;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
storageInput = await req.json();
|
storageInput = await request.json();
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error({ error, url: req.url }, "Error parsing JSON input");
|
logger.error({ error, url: request.url }, "Error parsing JSON input");
|
||||||
return responses.badRequestResponse("Malformed JSON input, please check your request body");
|
return responses.badRequestResponse("Malformed JSON input, please check your request body");
|
||||||
}
|
}
|
||||||
|
|
||||||
const { fileName, fileType, environmentId, allowedFileExtensions } = storageInput;
|
const { fileName, fileType, environmentId, allowedFileExtensions } = storageInput;
|
||||||
|
|
||||||
if (!fileName) {
|
const requiredFieldResponse = checkForRequiredFields(environmentId, fileType, fileName);
|
||||||
return responses.badRequestResponse("fileName is required");
|
if (requiredFieldResponse) return requiredFieldResponse;
|
||||||
}
|
const session = await getServerSession(authOptions);
|
||||||
|
|
||||||
if (!fileType) {
|
const authResponse = await checkAuth(session, environmentId, request);
|
||||||
return responses.badRequestResponse("fileType is required");
|
if (authResponse) return authResponse;
|
||||||
}
|
|
||||||
|
|
||||||
if (!environmentId) {
|
|
||||||
return responses.badRequestResponse("environmentId is required");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Perform server-side file validation first to block dangerous file types
|
// Perform server-side file validation first to block dangerous file types
|
||||||
const fileValidation = validateFile(fileName, fileType);
|
const fileValidation = validateFile(fileName, fileType);
|
||||||
@@ -53,18 +48,5 @@ export const POST = async (req: NextRequest): Promise<Response> => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// auth and upload private file
|
|
||||||
const session = await getServerSession(authOptions);
|
|
||||||
|
|
||||||
if (!session || !session.user) {
|
|
||||||
return responses.notAuthenticatedResponse();
|
|
||||||
}
|
|
||||||
|
|
||||||
const isUserAuthorized = await hasUserEnvironmentAccess(session.user.id, environmentId);
|
|
||||||
|
|
||||||
if (!isUserAuthorized) {
|
|
||||||
return responses.unauthorizedResponse();
|
|
||||||
}
|
|
||||||
|
|
||||||
return await getSignedUrlForPublicFile(fileName, environmentId, fileType);
|
return await getSignedUrlForPublicFile(fileName, environmentId, fileType);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { authenticateRequest, handleErrorResponse } from "@/app/api/v1/auth";
|
import { authenticateRequest, handleErrorResponse } from "@/app/api/v1/auth";
|
||||||
import { responses } from "@/app/lib/api/response";
|
import { responses } from "@/app/lib/api/response";
|
||||||
import { getSurveyDomain } from "@/lib/getSurveyUrl";
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getSurvey } from "@/lib/survey/service";
|
import { getSurvey } from "@/lib/survey/service";
|
||||||
import { generateSurveySingleUseIds } from "@/lib/utils/single-use-surveys";
|
import { generateSurveySingleUseIds } from "@/lib/utils/single-use-surveys";
|
||||||
import { hasPermission } from "@/modules/organization/settings/api-keys/lib/utils";
|
import { hasPermission } from "@/modules/organization/settings/api-keys/lib/utils";
|
||||||
@@ -42,10 +42,10 @@ export const GET = async (
|
|||||||
|
|
||||||
const singleUseIds = generateSurveySingleUseIds(limit, survey.singleUse.isEncrypted);
|
const singleUseIds = generateSurveySingleUseIds(limit, survey.singleUse.isEncrypted);
|
||||||
|
|
||||||
const surveyDomain = getSurveyDomain();
|
const publicDomain = getPublicDomain();
|
||||||
// map single use ids to survey links
|
// map single use ids to survey links
|
||||||
const surveyLinks = singleUseIds.map(
|
const surveyLinks = singleUseIds.map(
|
||||||
(singleUseId) => `${surveyDomain}/s/${survey.id}?suId=${singleUseId}`
|
(singleUseId) => `${publicDomain}/s/${survey.id}?suId=${singleUseId}`
|
||||||
);
|
);
|
||||||
|
|
||||||
return responses.successResponse(surveyLinks);
|
return responses.successResponse(surveyLinks);
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ describe("Survey Builder", () => {
|
|||||||
buttonLabel: { default: "common.next" },
|
buttonLabel: { default: "common.next" },
|
||||||
backButtonLabel: { default: "common.back" },
|
backButtonLabel: { default: "common.back" },
|
||||||
shuffleOption: "none",
|
shuffleOption: "none",
|
||||||
required: true,
|
required: false,
|
||||||
});
|
});
|
||||||
expect(question.choices.length).toBe(3);
|
expect(question.choices.length).toBe(3);
|
||||||
expect(question.id).toBeDefined();
|
expect(question.id).toBeDefined();
|
||||||
@@ -141,7 +141,7 @@ describe("Survey Builder", () => {
|
|||||||
inputType: "text",
|
inputType: "text",
|
||||||
buttonLabel: { default: "common.next" },
|
buttonLabel: { default: "common.next" },
|
||||||
backButtonLabel: { default: "common.back" },
|
backButtonLabel: { default: "common.back" },
|
||||||
required: true,
|
required: false,
|
||||||
charLimit: {
|
charLimit: {
|
||||||
enabled: false,
|
enabled: false,
|
||||||
},
|
},
|
||||||
@@ -204,7 +204,7 @@ describe("Survey Builder", () => {
|
|||||||
range: 5,
|
range: 5,
|
||||||
buttonLabel: { default: "common.next" },
|
buttonLabel: { default: "common.next" },
|
||||||
backButtonLabel: { default: "common.back" },
|
backButtonLabel: { default: "common.back" },
|
||||||
required: true,
|
required: false,
|
||||||
isColorCodingEnabled: false,
|
isColorCodingEnabled: false,
|
||||||
});
|
});
|
||||||
expect(question.id).toBeDefined();
|
expect(question.id).toBeDefined();
|
||||||
@@ -265,7 +265,7 @@ describe("Survey Builder", () => {
|
|||||||
headline: { default: "NPS Question" },
|
headline: { default: "NPS Question" },
|
||||||
buttonLabel: { default: "common.next" },
|
buttonLabel: { default: "common.next" },
|
||||||
backButtonLabel: { default: "common.back" },
|
backButtonLabel: { default: "common.back" },
|
||||||
required: true,
|
required: false,
|
||||||
isColorCodingEnabled: false,
|
isColorCodingEnabled: false,
|
||||||
});
|
});
|
||||||
expect(question.id).toBeDefined();
|
expect(question.id).toBeDefined();
|
||||||
@@ -324,7 +324,7 @@ describe("Survey Builder", () => {
|
|||||||
label: { default: "I agree to terms" },
|
label: { default: "I agree to terms" },
|
||||||
buttonLabel: { default: "common.next" },
|
buttonLabel: { default: "common.next" },
|
||||||
backButtonLabel: { default: "common.back" },
|
backButtonLabel: { default: "common.back" },
|
||||||
required: true,
|
required: false,
|
||||||
});
|
});
|
||||||
expect(question.id).toBeDefined();
|
expect(question.id).toBeDefined();
|
||||||
});
|
});
|
||||||
@@ -377,7 +377,7 @@ describe("Survey Builder", () => {
|
|||||||
headline: { default: "CTA Question" },
|
headline: { default: "CTA Question" },
|
||||||
buttonLabel: { default: "common.next" },
|
buttonLabel: { default: "common.next" },
|
||||||
backButtonLabel: { default: "common.back" },
|
backButtonLabel: { default: "common.back" },
|
||||||
required: true,
|
required: false,
|
||||||
buttonExternal: false,
|
buttonExternal: false,
|
||||||
});
|
});
|
||||||
expect(question.id).toBeDefined();
|
expect(question.id).toBeDefined();
|
||||||
|
|||||||
@@ -66,7 +66,7 @@ export const buildMultipleChoiceQuestion = ({
|
|||||||
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
||||||
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
||||||
shuffleOption: shuffleOption || "none",
|
shuffleOption: shuffleOption || "none",
|
||||||
required: required ?? true,
|
required: required ?? false,
|
||||||
logic,
|
logic,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
@@ -105,7 +105,7 @@ export const buildOpenTextQuestion = ({
|
|||||||
headline: createI18nString(headline, []),
|
headline: createI18nString(headline, []),
|
||||||
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
||||||
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
||||||
required: required ?? true,
|
required: required ?? false,
|
||||||
longAnswer,
|
longAnswer,
|
||||||
logic,
|
logic,
|
||||||
charLimit: {
|
charLimit: {
|
||||||
@@ -153,7 +153,7 @@ export const buildRatingQuestion = ({
|
|||||||
range,
|
range,
|
||||||
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
||||||
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
||||||
required: required ?? true,
|
required: required ?? false,
|
||||||
isColorCodingEnabled,
|
isColorCodingEnabled,
|
||||||
lowerLabel: lowerLabel ? createI18nString(lowerLabel, []) : undefined,
|
lowerLabel: lowerLabel ? createI18nString(lowerLabel, []) : undefined,
|
||||||
upperLabel: upperLabel ? createI18nString(upperLabel, []) : undefined,
|
upperLabel: upperLabel ? createI18nString(upperLabel, []) : undefined,
|
||||||
@@ -194,7 +194,7 @@ export const buildNPSQuestion = ({
|
|||||||
headline: createI18nString(headline, []),
|
headline: createI18nString(headline, []),
|
||||||
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
||||||
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
||||||
required: required ?? true,
|
required: required ?? false,
|
||||||
isColorCodingEnabled,
|
isColorCodingEnabled,
|
||||||
lowerLabel: lowerLabel ? createI18nString(lowerLabel, []) : undefined,
|
lowerLabel: lowerLabel ? createI18nString(lowerLabel, []) : undefined,
|
||||||
upperLabel: upperLabel ? createI18nString(upperLabel, []) : undefined,
|
upperLabel: upperLabel ? createI18nString(upperLabel, []) : undefined,
|
||||||
@@ -230,7 +230,7 @@ export const buildConsentQuestion = ({
|
|||||||
headline: createI18nString(headline, []),
|
headline: createI18nString(headline, []),
|
||||||
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
||||||
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
||||||
required: required ?? true,
|
required: required ?? false,
|
||||||
label: createI18nString(label, []),
|
label: createI18nString(label, []),
|
||||||
logic,
|
logic,
|
||||||
};
|
};
|
||||||
@@ -269,7 +269,7 @@ export const buildCTAQuestion = ({
|
|||||||
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
buttonLabel: createI18nString(buttonLabel || t(defaultButtonLabel), []),
|
||||||
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
backButtonLabel: createI18nString(backButtonLabel || t(defaultBackButtonLabel), []),
|
||||||
dismissButtonLabel: dismissButtonLabel ? createI18nString(dismissButtonLabel, []) : undefined,
|
dismissButtonLabel: dismissButtonLabel ? createI18nString(dismissButtonLabel, []) : undefined,
|
||||||
required: required ?? true,
|
required: required ?? false,
|
||||||
buttonExternal,
|
buttonExternal,
|
||||||
buttonUrl,
|
buttonUrl,
|
||||||
logic,
|
logic,
|
||||||
@@ -309,7 +309,7 @@ export const createJumpLogic = (
|
|||||||
// Helper function to create jump logic based on choice selection
|
// Helper function to create jump logic based on choice selection
|
||||||
export const createChoiceJumpLogic = (
|
export const createChoiceJumpLogic = (
|
||||||
sourceQuestionId: string,
|
sourceQuestionId: string,
|
||||||
choiceId: string,
|
choiceId: string | number,
|
||||||
targetId: string
|
targetId: string
|
||||||
): TSurveyLogic => ({
|
): TSurveyLogic => ({
|
||||||
id: createId(),
|
id: createId(),
|
||||||
|
|||||||
@@ -1210,6 +1210,7 @@ const feedbackBox = (t: TFnType): TTemplate => {
|
|||||||
t("templates.feedback_box_question_1_choice_1"),
|
t("templates.feedback_box_question_1_choice_1"),
|
||||||
t("templates.feedback_box_question_1_choice_2"),
|
t("templates.feedback_box_question_1_choice_2"),
|
||||||
],
|
],
|
||||||
|
choiceIds: [reusableOptionIds[0], reusableOptionIds[1]],
|
||||||
headline: t("templates.feedback_box_question_1_headline"),
|
headline: t("templates.feedback_box_question_1_headline"),
|
||||||
required: true,
|
required: true,
|
||||||
subheader: t("templates.feedback_box_question_1_subheader"),
|
subheader: t("templates.feedback_box_question_1_subheader"),
|
||||||
@@ -2054,7 +2055,7 @@ const professionalDevelopmentSurvey = (t: TFnType): TTemplate => {
|
|||||||
shuffleOption: "none",
|
shuffleOption: "none",
|
||||||
choices: [
|
choices: [
|
||||||
t("templates.professional_development_survey_question_1_choice_1"),
|
t("templates.professional_development_survey_question_1_choice_1"),
|
||||||
t("templates.professional_development_survey_question_1_choice_1"),
|
t("templates.professional_development_survey_question_1_choice_2"),
|
||||||
],
|
],
|
||||||
t,
|
t,
|
||||||
}),
|
}),
|
||||||
@@ -2381,6 +2382,7 @@ const measureTaskAccomplishment = (t: TFnType): TTemplate => {
|
|||||||
t("templates.measure_task_accomplishment_question_1_option_2_label"),
|
t("templates.measure_task_accomplishment_question_1_option_2_label"),
|
||||||
t("templates.measure_task_accomplishment_question_1_option_3_label"),
|
t("templates.measure_task_accomplishment_question_1_option_3_label"),
|
||||||
],
|
],
|
||||||
|
choiceIds: [reusableOptionIds[0], reusableOptionIds[1], reusableOptionIds[2]],
|
||||||
headline: t("templates.measure_task_accomplishment_question_1_headline"),
|
headline: t("templates.measure_task_accomplishment_question_1_headline"),
|
||||||
required: true,
|
required: true,
|
||||||
t,
|
t,
|
||||||
@@ -2739,10 +2741,10 @@ const understandPurchaseIntention = (t: TFnType): TTemplate => {
|
|||||||
buildRatingQuestion({
|
buildRatingQuestion({
|
||||||
id: reusableQuestionIds[0],
|
id: reusableQuestionIds[0],
|
||||||
logic: [
|
logic: [
|
||||||
createChoiceJumpLogic(reusableQuestionIds[0], "2", reusableQuestionIds[1]),
|
createChoiceJumpLogic(reusableQuestionIds[0], 2, reusableQuestionIds[1]),
|
||||||
createChoiceJumpLogic(reusableQuestionIds[0], "3", reusableQuestionIds[2]),
|
createChoiceJumpLogic(reusableQuestionIds[0], 3, reusableQuestionIds[2]),
|
||||||
createChoiceJumpLogic(reusableQuestionIds[0], "4", reusableQuestionIds[2]),
|
createChoiceJumpLogic(reusableQuestionIds[0], 4, reusableQuestionIds[2]),
|
||||||
createChoiceJumpLogic(reusableQuestionIds[0], "5", localSurvey.endings[0].id),
|
createChoiceJumpLogic(reusableQuestionIds[0], 5, localSurvey.endings[0].id),
|
||||||
],
|
],
|
||||||
range: 5,
|
range: 5,
|
||||||
scale: "number",
|
scale: "number",
|
||||||
@@ -2795,7 +2797,7 @@ const improveNewsletterContent = (t: TFnType): TTemplate => {
|
|||||||
buildRatingQuestion({
|
buildRatingQuestion({
|
||||||
id: reusableQuestionIds[0],
|
id: reusableQuestionIds[0],
|
||||||
logic: [
|
logic: [
|
||||||
createChoiceJumpLogic(reusableQuestionIds[0], "5", reusableQuestionIds[2]),
|
createChoiceJumpLogic(reusableQuestionIds[0], 5, reusableQuestionIds[2]),
|
||||||
{
|
{
|
||||||
id: createId(),
|
id: createId(),
|
||||||
conditions: {
|
conditions: {
|
||||||
@@ -2895,8 +2897,8 @@ const evaluateAProductIdea = (t: TFnType): TTemplate => {
|
|||||||
buildRatingQuestion({
|
buildRatingQuestion({
|
||||||
id: reusableQuestionIds[1],
|
id: reusableQuestionIds[1],
|
||||||
logic: [
|
logic: [
|
||||||
createChoiceJumpLogic(reusableQuestionIds[1], "3", reusableQuestionIds[2]),
|
createChoiceJumpLogic(reusableQuestionIds[1], 3, reusableQuestionIds[2]),
|
||||||
createChoiceJumpLogic(reusableQuestionIds[1], "4", reusableQuestionIds[3]),
|
createChoiceJumpLogic(reusableQuestionIds[1], 4, reusableQuestionIds[3]),
|
||||||
],
|
],
|
||||||
range: 5,
|
range: 5,
|
||||||
scale: "number",
|
scale: "number",
|
||||||
@@ -2928,8 +2930,8 @@ const evaluateAProductIdea = (t: TFnType): TTemplate => {
|
|||||||
buildRatingQuestion({
|
buildRatingQuestion({
|
||||||
id: reusableQuestionIds[4],
|
id: reusableQuestionIds[4],
|
||||||
logic: [
|
logic: [
|
||||||
createChoiceJumpLogic(reusableQuestionIds[4], "3", reusableQuestionIds[5]),
|
createChoiceJumpLogic(reusableQuestionIds[4], 3, reusableQuestionIds[5]),
|
||||||
createChoiceJumpLogic(reusableQuestionIds[4], "4", reusableQuestionIds[6]),
|
createChoiceJumpLogic(reusableQuestionIds[4], 4, reusableQuestionIds[6]),
|
||||||
],
|
],
|
||||||
range: 5,
|
range: 5,
|
||||||
scale: "number",
|
scale: "number",
|
||||||
@@ -3004,6 +3006,7 @@ const understandLowEngagement = (t: TFnType): TTemplate => {
|
|||||||
t("templates.understand_low_engagement_question_1_choice_4"),
|
t("templates.understand_low_engagement_question_1_choice_4"),
|
||||||
t("templates.understand_low_engagement_question_1_choice_5"),
|
t("templates.understand_low_engagement_question_1_choice_5"),
|
||||||
],
|
],
|
||||||
|
choiceIds: [reusableOptionIds[0], reusableOptionIds[1], reusableOptionIds[2], reusableOptionIds[3]],
|
||||||
headline: t("templates.understand_low_engagement_question_1_headline"),
|
headline: t("templates.understand_low_engagement_question_1_headline"),
|
||||||
required: true,
|
required: true,
|
||||||
containsOther: true,
|
containsOther: true,
|
||||||
|
|||||||
@@ -0,0 +1,85 @@
|
|||||||
|
import { NextRequest } from "next/server";
|
||||||
|
import { beforeEach, describe, expect, test, vi } from "vitest";
|
||||||
|
import { getPublicDomainHost, isPublicDomainConfigured, isRequestFromPublicDomain } from "./domain-utils";
|
||||||
|
|
||||||
|
// Mock the env module
|
||||||
|
vi.mock("@/lib/env", () => ({
|
||||||
|
env: {
|
||||||
|
get PUBLIC_URL() {
|
||||||
|
return process.env.PUBLIC_URL || "";
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("Domain Utils", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
process.env.PUBLIC_URL = "";
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getPublicDomain", () => {
|
||||||
|
test("should return null when PUBLIC_URL is empty", () => {
|
||||||
|
expect(getPublicDomainHost()).toBeNull();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return the host from a valid PUBLIC_URL", () => {
|
||||||
|
process.env.PUBLIC_URL = "https://example.com";
|
||||||
|
expect(getPublicDomainHost()).toBe("example.com");
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should handle URLs with paths", () => {
|
||||||
|
process.env.PUBLIC_URL = "https://example.com/path";
|
||||||
|
expect(getPublicDomainHost()).toBe("example.com");
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should handle URLs with ports", () => {
|
||||||
|
process.env.PUBLIC_URL = "https://example.com:3000";
|
||||||
|
expect(getPublicDomainHost()).toBe("example.com:3000");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("isPublicDomainConfigured", () => {
|
||||||
|
test("should return false when PUBLIC_URL is empty", () => {
|
||||||
|
process.env.PUBLIC_URL = "";
|
||||||
|
expect(isPublicDomainConfigured()).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return true when PUBLIC_URL is valid", () => {
|
||||||
|
process.env.PUBLIC_URL = "https://example.com";
|
||||||
|
expect(isPublicDomainConfigured()).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("isRequestFromPublicDomain", () => {
|
||||||
|
test("should return false when public domain is not configured", () => {
|
||||||
|
process.env.PUBLIC_URL = "";
|
||||||
|
const request = new NextRequest("https://example.com");
|
||||||
|
expect(isRequestFromPublicDomain(request)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return false when host doesn't match public domain", () => {
|
||||||
|
process.env.PUBLIC_URL = "https://example.com";
|
||||||
|
const request = new NextRequest("https://different-domain.com");
|
||||||
|
expect(isRequestFromPublicDomain(request)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return true when host matches public domain", () => {
|
||||||
|
process.env.PUBLIC_URL = "https://example.com";
|
||||||
|
const request = new NextRequest("https://example.com", {
|
||||||
|
headers: {
|
||||||
|
host: "example.com",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(isRequestFromPublicDomain(request)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should handle domains with ports", () => {
|
||||||
|
process.env.PUBLIC_URL = "https://example.com:3000";
|
||||||
|
const request = new NextRequest("https://example.com:3000", {
|
||||||
|
headers: {
|
||||||
|
host: "example.com:3000",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
expect(isRequestFromPublicDomain(request)).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
import { env } from "@/lib/env";
|
||||||
|
import { NextRequest } from "next/server";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the public domain from PUBLIC_URL environment variable
|
||||||
|
*/
|
||||||
|
export const getPublicDomainHost = (): string | null => {
|
||||||
|
const PUBLIC_URL = env.PUBLIC_URL;
|
||||||
|
if (!PUBLIC_URL) return null;
|
||||||
|
|
||||||
|
return new URL(PUBLIC_URL).host;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if PUBLIC_URL is configured (has a valid public domain)
|
||||||
|
*/
|
||||||
|
export const isPublicDomainConfigured = (): boolean => {
|
||||||
|
return getPublicDomainHost() !== null;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if the current request is coming from the public domain
|
||||||
|
*/
|
||||||
|
export const isRequestFromPublicDomain = (request: NextRequest): boolean => {
|
||||||
|
const host = request.headers.get("host");
|
||||||
|
const publicDomainHost = getPublicDomainHost();
|
||||||
|
|
||||||
|
if (!publicDomainHost) return false;
|
||||||
|
|
||||||
|
return host === publicDomainHost;
|
||||||
|
};
|
||||||
@@ -1,10 +1,13 @@
|
|||||||
import { describe, expect, test } from "vitest";
|
import { describe, expect, test } from "vitest";
|
||||||
import {
|
import {
|
||||||
|
isAdminDomainRoute,
|
||||||
isAuthProtectedRoute,
|
isAuthProtectedRoute,
|
||||||
isClientSideApiRoute,
|
isClientSideApiRoute,
|
||||||
isForgotPasswordRoute,
|
isForgotPasswordRoute,
|
||||||
isLoginRoute,
|
isLoginRoute,
|
||||||
isManagementApiRoute,
|
isManagementApiRoute,
|
||||||
|
isPublicDomainRoute,
|
||||||
|
isRouteAllowedForDomain,
|
||||||
isShareUrlRoute,
|
isShareUrlRoute,
|
||||||
isSignupRoute,
|
isSignupRoute,
|
||||||
isSyncWithUserIdentificationEndpoint,
|
isSyncWithUserIdentificationEndpoint,
|
||||||
@@ -69,6 +72,9 @@ describe("endpoint-validator", () => {
|
|||||||
expect(isClientSideApiRoute("/api/v1/management/something")).toBe(false);
|
expect(isClientSideApiRoute("/api/v1/management/something")).toBe(false);
|
||||||
expect(isClientSideApiRoute("/api/something")).toBe(false);
|
expect(isClientSideApiRoute("/api/something")).toBe(false);
|
||||||
expect(isClientSideApiRoute("/auth/login")).toBe(false);
|
expect(isClientSideApiRoute("/auth/login")).toBe(false);
|
||||||
|
|
||||||
|
// exception for open graph image generation route, it should not be rate limited
|
||||||
|
expect(isClientSideApiRoute("/api/v1/client/og")).toBe(false);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -136,4 +142,138 @@ describe("endpoint-validator", () => {
|
|||||||
expect(isSyncWithUserIdentificationEndpoint("/api/something")).toBe(false);
|
expect(isSyncWithUserIdentificationEndpoint("/api/something")).toBe(false);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("isPublicDomainRoute", () => {
|
||||||
|
test("should return true for health endpoint", () => {
|
||||||
|
expect(isPublicDomainRoute("/health")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Static assets are not handled by domain routing - middleware doesn't run on them
|
||||||
|
|
||||||
|
test("should return true for survey routes", () => {
|
||||||
|
expect(isPublicDomainRoute("/s/survey123")).toBe(true);
|
||||||
|
expect(isPublicDomainRoute("/s/survey-id-with-dashes")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return true for contact survey routes", () => {
|
||||||
|
expect(isPublicDomainRoute("/c/jwt-token")).toBe(true);
|
||||||
|
expect(isPublicDomainRoute("/c/very-long-jwt-token-123")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return true for client API routes", () => {
|
||||||
|
expect(isPublicDomainRoute("/api/v1/client/something")).toBe(true);
|
||||||
|
expect(isPublicDomainRoute("/api/v2/client/other")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return true for share routes", () => {
|
||||||
|
expect(isPublicDomainRoute("/share/abc123/summary")).toBe(true);
|
||||||
|
expect(isPublicDomainRoute("/share/xyz789/responses")).toBe(true);
|
||||||
|
expect(isPublicDomainRoute("/share/anything")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return false for admin-only routes", () => {
|
||||||
|
expect(isPublicDomainRoute("/")).toBe(false);
|
||||||
|
expect(isPublicDomainRoute("/environments/123")).toBe(false);
|
||||||
|
expect(isPublicDomainRoute("/auth/login")).toBe(false);
|
||||||
|
expect(isPublicDomainRoute("/setup/organization")).toBe(false);
|
||||||
|
expect(isPublicDomainRoute("/organizations/123")).toBe(false);
|
||||||
|
expect(isPublicDomainRoute("/product/settings")).toBe(false);
|
||||||
|
expect(isPublicDomainRoute("/api/v1/management/users")).toBe(false);
|
||||||
|
expect(isPublicDomainRoute("/api/v2/management/surveys")).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("isAdminDomainRoute", () => {
|
||||||
|
test("should return true for health endpoint (backward compatibility)", () => {
|
||||||
|
expect(isAdminDomainRoute("/health")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/health")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Static assets are not handled by domain routing - middleware doesn't run on them
|
||||||
|
|
||||||
|
test("should return true for admin routes", () => {
|
||||||
|
expect(isAdminDomainRoute("/")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/environments/123")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/environments/123/surveys")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/auth/login")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/auth/signup")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/setup/organization")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/setup/team")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/organizations/123")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/organizations/123/settings")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/product/settings")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/product/features")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/api/v1/management/users")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/api/v2/management/surveys")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/pipeline/jobs")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/cron/tasks")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/random/route")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("/s/survey123")).toBe(false);
|
||||||
|
expect(isAdminDomainRoute("/c/jwt-token")).toBe(false);
|
||||||
|
expect(isAdminDomainRoute("/api/v1/client/test")).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("isRouteAllowedForDomain", () => {
|
||||||
|
test("should allow public routes on public domain", () => {
|
||||||
|
expect(isRouteAllowedForDomain("/s/survey123", true)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/c/jwt-token", true)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/api/v1/client/test", true)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/share/abc/summary", true)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/health", true)).toBe(true);
|
||||||
|
// Static assets not tested - middleware doesn't run on them
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should block admin routes on public domain", () => {
|
||||||
|
expect(isRouteAllowedForDomain("/", true)).toBe(false);
|
||||||
|
expect(isRouteAllowedForDomain("/environments/123", true)).toBe(false);
|
||||||
|
expect(isRouteAllowedForDomain("/auth/login", true)).toBe(false);
|
||||||
|
expect(isRouteAllowedForDomain("/api/v1/management/users", true)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should block public routes on admin domain when PUBLIC_URL is configured", () => {
|
||||||
|
// Admin routes should be allowed
|
||||||
|
expect(isRouteAllowedForDomain("/", false)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/environments/123", false)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/auth/login", false)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/api/v1/management/users", false)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/health", false)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/pipeline/jobs", false)).toBe(true);
|
||||||
|
expect(isRouteAllowedForDomain("/cron/tasks", false)).toBe(true);
|
||||||
|
|
||||||
|
// Public routes should be blocked on admin domain
|
||||||
|
expect(isRouteAllowedForDomain("/s/survey123", false)).toBe(false);
|
||||||
|
expect(isRouteAllowedForDomain("/c/jwt-token", false)).toBe(false);
|
||||||
|
expect(isRouteAllowedForDomain("/api/v1/client/test", false)).toBe(false);
|
||||||
|
expect(isRouteAllowedForDomain("/share/abc/summary", false)).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("edge cases", () => {
|
||||||
|
test("should handle empty paths", () => {
|
||||||
|
expect(isPublicDomainRoute("")).toBe(false);
|
||||||
|
expect(isAdminDomainRoute("")).toBe(true);
|
||||||
|
expect(isAdminDomainRoute("")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should handle paths with query parameters", () => {
|
||||||
|
expect(isPublicDomainRoute("/s/survey123?param=value")).toBe(true);
|
||||||
|
expect(isPublicDomainRoute("/environments/123?tab=settings")).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should handle paths with fragments", () => {
|
||||||
|
expect(isPublicDomainRoute("/s/survey123#section")).toBe(true);
|
||||||
|
expect(isPublicDomainRoute("/environments/123#overview")).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should handle nested survey routes", () => {
|
||||||
|
expect(isPublicDomainRoute("/s/survey123/preview")).toBe(true);
|
||||||
|
expect(isPublicDomainRoute("/s/survey123/embed")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should handle nested client API routes", () => {
|
||||||
|
expect(isPublicDomainRoute("/api/v1/client/env123/actions")).toBe(true);
|
||||||
|
expect(isPublicDomainRoute("/api/v2/client/env456/responses")).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,3 +1,9 @@
|
|||||||
|
import {
|
||||||
|
getAllPubliclyAccessibleRoutePatterns,
|
||||||
|
getPublicDomainRoutePatterns,
|
||||||
|
matchesAnyPattern,
|
||||||
|
} from "./route-config";
|
||||||
|
|
||||||
export const isLoginRoute = (url: string) =>
|
export const isLoginRoute = (url: string) =>
|
||||||
url === "/api/auth/callback/credentials" || url === "/auth/login";
|
url === "/api/auth/callback/credentials" || url === "/auth/login";
|
||||||
|
|
||||||
@@ -8,6 +14,9 @@ export const isVerifyEmailRoute = (url: string) => url === "/auth/verify-email";
|
|||||||
export const isForgotPasswordRoute = (url: string) => url === "/auth/forgot-password";
|
export const isForgotPasswordRoute = (url: string) => url === "/auth/forgot-password";
|
||||||
|
|
||||||
export const isClientSideApiRoute = (url: string): boolean => {
|
export const isClientSideApiRoute = (url: string): boolean => {
|
||||||
|
// Open Graph image generation route is a client side API route but it should not be rate limited
|
||||||
|
if (url.includes("/api/v1/client/og")) return false;
|
||||||
|
|
||||||
if (url.includes("/api/v1/js/actions")) return true;
|
if (url.includes("/api/v1/js/actions")) return true;
|
||||||
if (url.includes("/api/v1/client/storage")) return true;
|
if (url.includes("/api/v1/client/storage")) return true;
|
||||||
const regex = /^\/api\/v\d+\/client\//;
|
const regex = /^\/api\/v\d+\/client\//;
|
||||||
@@ -38,3 +47,39 @@ export const isSyncWithUserIdentificationEndpoint = (
|
|||||||
const match = url.match(regex);
|
const match = url.match(regex);
|
||||||
return match ? { environmentId: match.groups!.environmentId, userId: match.groups!.userId } : false;
|
return match ? { environmentId: match.groups!.environmentId, userId: match.groups!.userId } : false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if the route should be accessible on the public domain (PUBLIC_URL)
|
||||||
|
* Uses whitelist approach - only explicitly allowed routes are accessible
|
||||||
|
*/
|
||||||
|
export const isPublicDomainRoute = (url: string): boolean => {
|
||||||
|
const publicRoutePatterns = getAllPubliclyAccessibleRoutePatterns();
|
||||||
|
return matchesAnyPattern(url, publicRoutePatterns);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if the route should be accessible on the admin domain (WEBAPP_URL)
|
||||||
|
* When PUBLIC_URL is configured, admin domain should only allow admin-specific routes + health
|
||||||
|
*/
|
||||||
|
export const isAdminDomainRoute = (url: string): boolean => {
|
||||||
|
const publicOnlyRoutePatterns = getPublicDomainRoutePatterns();
|
||||||
|
const isPublicRoute = matchesAnyPattern(url, publicOnlyRoutePatterns);
|
||||||
|
|
||||||
|
if (isPublicRoute) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// For non-public routes, allow them (includes known admin routes and unknown routes like pipeline, cron)
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if a request should be allowed based on domain and route
|
||||||
|
*/
|
||||||
|
export const isRouteAllowedForDomain = (url: string, isPublicDomain: boolean): boolean => {
|
||||||
|
if (isPublicDomain) {
|
||||||
|
return isPublicDomainRoute(url);
|
||||||
|
}
|
||||||
|
|
||||||
|
return isAdminDomainRoute(url);
|
||||||
|
};
|
||||||
|
|||||||
@@ -0,0 +1,54 @@
|
|||||||
|
/**
|
||||||
|
* Routes that should be accessible on the public domain (PUBLIC_URL)
|
||||||
|
* Uses whitelist approach - only these routes are allowed on public domain
|
||||||
|
*/
|
||||||
|
const PUBLIC_ROUTES = {
|
||||||
|
// Survey routes
|
||||||
|
SURVEY_ROUTES: [
|
||||||
|
/^\/s\/[^/]+/, // /s/[surveyId] - survey pages
|
||||||
|
/^\/c\/[^/]+/, // /c/[jwt] - contact survey pages
|
||||||
|
],
|
||||||
|
|
||||||
|
// API routes accessible from public domain
|
||||||
|
API_ROUTES: [
|
||||||
|
/^\/api\/v[12]\/client\//, // /api/v1/client/** and /api/v2/client/**
|
||||||
|
],
|
||||||
|
|
||||||
|
// Share routes
|
||||||
|
SHARE_ROUTES: [
|
||||||
|
/^\/share\//, // /share/** - shared survey results
|
||||||
|
],
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
const COMMON_ROUTES = {
|
||||||
|
HEALTH_ROUTES: [/^\/health$/], // /health endpoint
|
||||||
|
PUBLIC_STORAGE_ROUTES: [
|
||||||
|
/^\/storage\/[^/]+\/public\//, // /storage/[environmentId]/public/** - public storage
|
||||||
|
],
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get public only route patterns as a flat array
|
||||||
|
*/
|
||||||
|
export const getPublicDomainRoutePatterns = (): RegExp[] => {
|
||||||
|
return Object.values(PUBLIC_ROUTES).flat();
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all public route patterns as a flat array
|
||||||
|
*/
|
||||||
|
export const getAllPubliclyAccessibleRoutePatterns = (): RegExp[] => {
|
||||||
|
const routes = {
|
||||||
|
...PUBLIC_ROUTES,
|
||||||
|
...COMMON_ROUTES,
|
||||||
|
};
|
||||||
|
|
||||||
|
return Object.values(routes).flat();
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a URL matches any of the given route patterns
|
||||||
|
*/
|
||||||
|
export const matchesAnyPattern = (url: string, patterns: RegExp[]): boolean => {
|
||||||
|
return patterns.some((pattern) => pattern.test(url));
|
||||||
|
};
|
||||||
+72
-13
@@ -3,12 +3,12 @@ import { cleanup } from "@testing-library/react";
|
|||||||
import { beforeEach, describe, expect, test, vi } from "vitest";
|
import { beforeEach, describe, expect, test, vi } from "vitest";
|
||||||
import { TMembership } from "@formbricks/types/memberships";
|
import { TMembership } from "@formbricks/types/memberships";
|
||||||
import { TOrganization } from "@formbricks/types/organizations";
|
import { TOrganization } from "@formbricks/types/organizations";
|
||||||
|
import { TProject } from "@formbricks/types/project";
|
||||||
import { TUser } from "@formbricks/types/user";
|
import { TUser } from "@formbricks/types/user";
|
||||||
import Page from "./page";
|
import Page from "./page";
|
||||||
|
|
||||||
// Mock dependencies
|
vi.mock("@/lib/project/service", () => ({
|
||||||
vi.mock("@/lib/environment/service", () => ({
|
getUserProjectEnvironmentsByOrganizationIds: vi.fn(),
|
||||||
getFirstEnvironmentIdByUserId: vi.fn(),
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
vi.mock("@/lib/instance/service", () => ({
|
vi.mock("@/lib/instance/service", () => ({
|
||||||
@@ -48,8 +48,11 @@ vi.mock("@/modules/ui/components/client-logout", () => ({
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
vi.mock("@/app/ClientEnvironmentRedirect", () => ({
|
vi.mock("@/app/ClientEnvironmentRedirect", () => ({
|
||||||
default: ({ environmentId }: { environmentId: string }) => (
|
default: ({ environmentId, userEnvironments }: { environmentId: string; userEnvironments?: string[] }) => (
|
||||||
<div data-testid="client-environment-redirect">Environment ID: {environmentId}</div>
|
<div data-testid="client-environment-redirect">
|
||||||
|
Environment ID: {environmentId}
|
||||||
|
{userEnvironments && ` | User Environments: ${userEnvironments.join(", ")}`}
|
||||||
|
</div>
|
||||||
),
|
),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
@@ -149,7 +152,7 @@ describe("Page", () => {
|
|||||||
const { getIsFreshInstance } = await import("@/lib/instance/service");
|
const { getIsFreshInstance } = await import("@/lib/instance/service");
|
||||||
const { getUser } = await import("@/lib/user/service");
|
const { getUser } = await import("@/lib/user/service");
|
||||||
const { getOrganizationsByUserId } = await import("@/lib/organization/service");
|
const { getOrganizationsByUserId } = await import("@/lib/organization/service");
|
||||||
const { getFirstEnvironmentIdByUserId } = await import("@/lib/environment/service");
|
const { getUserProjectEnvironmentsByOrganizationIds } = await import("@/lib/project/service");
|
||||||
const { getMembershipByUserIdOrganizationId } = await import("@/lib/membership/service");
|
const { getMembershipByUserIdOrganizationId } = await import("@/lib/membership/service");
|
||||||
const { getAccessFlags } = await import("@/lib/membership/utils");
|
const { getAccessFlags } = await import("@/lib/membership/utils");
|
||||||
const { redirect } = await import("next/navigation");
|
const { redirect } = await import("next/navigation");
|
||||||
@@ -204,13 +207,23 @@ describe("Page", () => {
|
|||||||
role: "owner",
|
role: "owner",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const mockUserProjects = [
|
||||||
|
{
|
||||||
|
id: "test-project-id",
|
||||||
|
name: "Test Project",
|
||||||
|
environments: [],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
vi.mocked(getServerSession).mockResolvedValue({
|
vi.mocked(getServerSession).mockResolvedValue({
|
||||||
user: { id: "test-user-id" },
|
user: { id: "test-user-id" },
|
||||||
} as any);
|
} as any);
|
||||||
vi.mocked(getIsFreshInstance).mockResolvedValue(false);
|
vi.mocked(getIsFreshInstance).mockResolvedValue(false);
|
||||||
vi.mocked(getUser).mockResolvedValue(mockUser);
|
vi.mocked(getUser).mockResolvedValue(mockUser);
|
||||||
|
vi.mocked(getUserProjectEnvironmentsByOrganizationIds).mockResolvedValue(
|
||||||
|
mockUserProjects as unknown as TProject[]
|
||||||
|
);
|
||||||
vi.mocked(getOrganizationsByUserId).mockResolvedValue([mockOrganization]);
|
vi.mocked(getOrganizationsByUserId).mockResolvedValue([mockOrganization]);
|
||||||
vi.mocked(getFirstEnvironmentIdByUserId).mockResolvedValue(null);
|
|
||||||
vi.mocked(getMembershipByUserIdOrganizationId).mockResolvedValue(mockMembership);
|
vi.mocked(getMembershipByUserIdOrganizationId).mockResolvedValue(mockMembership);
|
||||||
vi.mocked(getAccessFlags).mockReturnValue({
|
vi.mocked(getAccessFlags).mockReturnValue({
|
||||||
isManager: false,
|
isManager: false,
|
||||||
@@ -228,8 +241,8 @@ describe("Page", () => {
|
|||||||
const { getServerSession } = await import("next-auth");
|
const { getServerSession } = await import("next-auth");
|
||||||
const { getIsFreshInstance } = await import("@/lib/instance/service");
|
const { getIsFreshInstance } = await import("@/lib/instance/service");
|
||||||
const { getUser } = await import("@/lib/user/service");
|
const { getUser } = await import("@/lib/user/service");
|
||||||
|
const { getUserProjectEnvironmentsByOrganizationIds } = await import("@/lib/project/service");
|
||||||
const { getOrganizationsByUserId } = await import("@/lib/organization/service");
|
const { getOrganizationsByUserId } = await import("@/lib/organization/service");
|
||||||
const { getFirstEnvironmentIdByUserId } = await import("@/lib/environment/service");
|
|
||||||
const { getMembershipByUserIdOrganizationId } = await import("@/lib/membership/service");
|
const { getMembershipByUserIdOrganizationId } = await import("@/lib/membership/service");
|
||||||
const { getAccessFlags } = await import("@/lib/membership/utils");
|
const { getAccessFlags } = await import("@/lib/membership/utils");
|
||||||
const { redirect } = await import("next/navigation");
|
const { redirect } = await import("next/navigation");
|
||||||
@@ -284,13 +297,23 @@ describe("Page", () => {
|
|||||||
role: "member",
|
role: "member",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const mockUserProjects = [
|
||||||
|
{
|
||||||
|
id: "test-project-id",
|
||||||
|
name: "Test Project",
|
||||||
|
environments: [],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
vi.mocked(getServerSession).mockResolvedValue({
|
vi.mocked(getServerSession).mockResolvedValue({
|
||||||
user: { id: "test-user-id" },
|
user: { id: "test-user-id" },
|
||||||
} as any);
|
} as any);
|
||||||
vi.mocked(getIsFreshInstance).mockResolvedValue(false);
|
vi.mocked(getIsFreshInstance).mockResolvedValue(false);
|
||||||
vi.mocked(getUser).mockResolvedValue(mockUser);
|
vi.mocked(getUser).mockResolvedValue(mockUser);
|
||||||
|
vi.mocked(getUserProjectEnvironmentsByOrganizationIds).mockResolvedValue(
|
||||||
|
mockUserProjects as unknown as TProject[]
|
||||||
|
);
|
||||||
vi.mocked(getOrganizationsByUserId).mockResolvedValue([mockOrganization]);
|
vi.mocked(getOrganizationsByUserId).mockResolvedValue([mockOrganization]);
|
||||||
vi.mocked(getFirstEnvironmentIdByUserId).mockResolvedValue(null);
|
|
||||||
vi.mocked(getMembershipByUserIdOrganizationId).mockResolvedValue(mockMembership);
|
vi.mocked(getMembershipByUserIdOrganizationId).mockResolvedValue(mockMembership);
|
||||||
vi.mocked(getAccessFlags).mockReturnValue({
|
vi.mocked(getAccessFlags).mockReturnValue({
|
||||||
isManager: false,
|
isManager: false,
|
||||||
@@ -309,9 +332,9 @@ describe("Page", () => {
|
|||||||
const { getIsFreshInstance } = await import("@/lib/instance/service");
|
const { getIsFreshInstance } = await import("@/lib/instance/service");
|
||||||
const { getUser } = await import("@/lib/user/service");
|
const { getUser } = await import("@/lib/user/service");
|
||||||
const { getOrganizationsByUserId } = await import("@/lib/organization/service");
|
const { getOrganizationsByUserId } = await import("@/lib/organization/service");
|
||||||
const { getFirstEnvironmentIdByUserId } = await import("@/lib/environment/service");
|
|
||||||
const { getMembershipByUserIdOrganizationId } = await import("@/lib/membership/service");
|
const { getMembershipByUserIdOrganizationId } = await import("@/lib/membership/service");
|
||||||
const { getAccessFlags } = await import("@/lib/membership/utils");
|
const { getAccessFlags } = await import("@/lib/membership/utils");
|
||||||
|
const { getUserProjectEnvironmentsByOrganizationIds } = await import("@/lib/project/service");
|
||||||
const { render } = await import("@testing-library/react");
|
const { render } = await import("@testing-library/react");
|
||||||
|
|
||||||
const mockUser: TUser = {
|
const mockUser: TUser = {
|
||||||
@@ -364,7 +387,43 @@ describe("Page", () => {
|
|||||||
role: "member",
|
role: "member",
|
||||||
};
|
};
|
||||||
|
|
||||||
const mockEnvironmentId = "test-env-id";
|
const mockUserProjects = [
|
||||||
|
{
|
||||||
|
id: "project-1",
|
||||||
|
name: "Test Project",
|
||||||
|
createdAt: new Date(),
|
||||||
|
updatedAt: new Date(),
|
||||||
|
organizationId: "test-org-id",
|
||||||
|
styling: { allowStyleOverwrite: true },
|
||||||
|
recontactDays: 0,
|
||||||
|
inAppSurveyBranding: false,
|
||||||
|
linkSurveyBranding: false,
|
||||||
|
config: { channel: "link" as const, industry: "saas" as const },
|
||||||
|
placement: "bottomRight" as const,
|
||||||
|
clickOutsideClose: false,
|
||||||
|
darkOverlay: false,
|
||||||
|
languages: [],
|
||||||
|
logo: null,
|
||||||
|
environments: [
|
||||||
|
{
|
||||||
|
id: "test-env-id",
|
||||||
|
type: "production" as const,
|
||||||
|
createdAt: new Date(),
|
||||||
|
updatedAt: new Date(),
|
||||||
|
projectId: "project-1",
|
||||||
|
appSetupCompleted: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "test-env-dev",
|
||||||
|
type: "development" as const,
|
||||||
|
createdAt: new Date(),
|
||||||
|
updatedAt: new Date(),
|
||||||
|
projectId: "project-1",
|
||||||
|
appSetupCompleted: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
] as any;
|
||||||
|
|
||||||
vi.mocked(getServerSession).mockResolvedValue({
|
vi.mocked(getServerSession).mockResolvedValue({
|
||||||
user: { id: "test-user-id" },
|
user: { id: "test-user-id" },
|
||||||
@@ -372,8 +431,8 @@ describe("Page", () => {
|
|||||||
vi.mocked(getIsFreshInstance).mockResolvedValue(false);
|
vi.mocked(getIsFreshInstance).mockResolvedValue(false);
|
||||||
vi.mocked(getUser).mockResolvedValue(mockUser);
|
vi.mocked(getUser).mockResolvedValue(mockUser);
|
||||||
vi.mocked(getOrganizationsByUserId).mockResolvedValue([mockOrganization]);
|
vi.mocked(getOrganizationsByUserId).mockResolvedValue([mockOrganization]);
|
||||||
vi.mocked(getFirstEnvironmentIdByUserId).mockResolvedValue(mockEnvironmentId);
|
|
||||||
vi.mocked(getMembershipByUserIdOrganizationId).mockResolvedValue(mockMembership);
|
vi.mocked(getMembershipByUserIdOrganizationId).mockResolvedValue(mockMembership);
|
||||||
|
vi.mocked(getUserProjectEnvironmentsByOrganizationIds).mockResolvedValue(mockUserProjects);
|
||||||
vi.mocked(getAccessFlags).mockReturnValue({
|
vi.mocked(getAccessFlags).mockReturnValue({
|
||||||
isManager: false,
|
isManager: false,
|
||||||
isOwner: false,
|
isOwner: false,
|
||||||
@@ -385,7 +444,7 @@ describe("Page", () => {
|
|||||||
const { container } = render(result);
|
const { container } = render(result);
|
||||||
|
|
||||||
expect(container.querySelector('[data-testid="client-environment-redirect"]')).toHaveTextContent(
|
expect(container.querySelector('[data-testid="client-environment-redirect"]')).toHaveTextContent(
|
||||||
`Environment ID: ${mockEnvironmentId}`
|
`User Environments: test-env-id, test-env-dev`
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
+29
-5
@@ -1,9 +1,9 @@
|
|||||||
import ClientEnvironmentRedirect from "@/app/ClientEnvironmentRedirect";
|
import ClientEnvironmentRedirect from "@/app/ClientEnvironmentRedirect";
|
||||||
import { getFirstEnvironmentIdByUserId } from "@/lib/environment/service";
|
|
||||||
import { getIsFreshInstance } from "@/lib/instance/service";
|
import { getIsFreshInstance } from "@/lib/instance/service";
|
||||||
import { getMembershipByUserIdOrganizationId } from "@/lib/membership/service";
|
import { getMembershipByUserIdOrganizationId } from "@/lib/membership/service";
|
||||||
import { getAccessFlags } from "@/lib/membership/utils";
|
import { getAccessFlags } from "@/lib/membership/utils";
|
||||||
import { getOrganizationsByUserId } from "@/lib/organization/service";
|
import { getOrganizationsByUserId } from "@/lib/organization/service";
|
||||||
|
import { getUserProjectEnvironmentsByOrganizationIds } from "@/lib/project/service";
|
||||||
import { getUser } from "@/lib/user/service";
|
import { getUser } from "@/lib/user/service";
|
||||||
import { authOptions } from "@/modules/auth/lib/authOptions";
|
import { authOptions } from "@/modules/auth/lib/authOptions";
|
||||||
import { ClientLogout } from "@/modules/ui/components/client-logout";
|
import { ClientLogout } from "@/modules/ui/components/client-logout";
|
||||||
@@ -34,16 +34,37 @@ const Page = async () => {
|
|||||||
return redirect("/setup/organization/create");
|
return redirect("/setup/organization/create");
|
||||||
}
|
}
|
||||||
|
|
||||||
let environmentId: string | null = null;
|
const projectsByOrg = await getUserProjectEnvironmentsByOrganizationIds(
|
||||||
environmentId = await getFirstEnvironmentIdByUserId(session.user.id);
|
userOrganizations.map((org) => org.id),
|
||||||
|
user.id
|
||||||
|
);
|
||||||
|
|
||||||
|
// Flatten all environments from all projects across all organizations
|
||||||
|
const allEnvironments = projectsByOrg.flatMap((project) => project.environments);
|
||||||
|
|
||||||
|
// Find first production environment and collect all other environment IDs in one pass
|
||||||
|
const { firstProductionEnvironmentId, otherEnvironmentIds } = allEnvironments.reduce(
|
||||||
|
(acc, env) => {
|
||||||
|
if (env.type === "production" && !acc.firstProductionEnvironmentId) {
|
||||||
|
acc.firstProductionEnvironmentId = env.id;
|
||||||
|
} else {
|
||||||
|
acc.otherEnvironmentIds.add(env.id);
|
||||||
|
}
|
||||||
|
return acc;
|
||||||
|
},
|
||||||
|
{ firstProductionEnvironmentId: null as string | null, otherEnvironmentIds: new Set<string>() }
|
||||||
|
);
|
||||||
|
|
||||||
|
const userEnvironments = [...otherEnvironmentIds];
|
||||||
|
|
||||||
const currentUserMembership = await getMembershipByUserIdOrganizationId(
|
const currentUserMembership = await getMembershipByUserIdOrganizationId(
|
||||||
session.user.id,
|
session.user.id,
|
||||||
userOrganizations[0].id
|
userOrganizations[0].id
|
||||||
);
|
);
|
||||||
|
|
||||||
const { isManager, isOwner } = getAccessFlags(currentUserMembership?.role);
|
const { isManager, isOwner } = getAccessFlags(currentUserMembership?.role);
|
||||||
|
|
||||||
if (!environmentId) {
|
if (!firstProductionEnvironmentId) {
|
||||||
if (isOwner || isManager) {
|
if (isOwner || isManager) {
|
||||||
return redirect(`/organizations/${userOrganizations[0].id}/projects/new/mode`);
|
return redirect(`/organizations/${userOrganizations[0].id}/projects/new/mode`);
|
||||||
} else {
|
} else {
|
||||||
@@ -51,7 +72,10 @@ const Page = async () => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return <ClientEnvironmentRedirect environmentId={environmentId} />;
|
// Put the first production environment at the front of the array
|
||||||
|
const sortedUserEnvironments = [firstProductionEnvironmentId, ...userEnvironments];
|
||||||
|
|
||||||
|
return <ClientEnvironmentRedirect userEnvironments={sortedUserEnvironments} />;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Page;
|
export default Page;
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
import { LinkSurveyNotFound } from "@/modules/survey/link/not-found";
|
import { LinkSurveyNotFound } from "@/modules/survey/link/not-found";
|
||||||
|
|
||||||
export default LinkSurveyNotFound;
|
export default function NotFound() {
|
||||||
|
return <LinkSurveyNotFound />;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
import { SurveyAnalysisNavigation } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/components/SurveyAnalysisNavigation";
|
import { SurveyAnalysisNavigation } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/components/SurveyAnalysisNavigation";
|
||||||
import { ResponsePage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/responses/components/ResponsePage";
|
import { ResponsePage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/responses/components/ResponsePage";
|
||||||
import { RESPONSES_PER_PAGE, WEBAPP_URL } from "@/lib/constants";
|
import { RESPONSES_PER_PAGE } from "@/lib/constants";
|
||||||
import { getEnvironment } from "@/lib/environment/service";
|
import { getEnvironment } from "@/lib/environment/service";
|
||||||
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
||||||
import { getSurvey, getSurveyIdByResultShareKey } from "@/lib/survey/service";
|
import { getSurvey, getSurveyIdByResultShareKey } from "@/lib/survey/service";
|
||||||
import { getTagsByEnvironmentId } from "@/lib/tag/service";
|
import { getTagsByEnvironmentId } from "@/lib/tag/service";
|
||||||
@@ -46,6 +47,7 @@ const Page = async (props: ResponsesPageProps) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const locale = await findMatchingLocale();
|
const locale = await findMatchingLocale();
|
||||||
|
const publicDomain = getPublicDomain();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex w-full justify-center">
|
<div className="flex w-full justify-center">
|
||||||
@@ -57,7 +59,7 @@ const Page = async (props: ResponsesPageProps) => {
|
|||||||
environment={environment}
|
environment={environment}
|
||||||
survey={survey}
|
survey={survey}
|
||||||
surveyId={surveyId}
|
surveyId={surveyId}
|
||||||
webAppUrl={WEBAPP_URL}
|
publicDomain={publicDomain}
|
||||||
environmentTags={tags}
|
environmentTags={tags}
|
||||||
responsesPerPage={RESPONSES_PER_PAGE}
|
responsesPerPage={RESPONSES_PER_PAGE}
|
||||||
locale={locale}
|
locale={locale}
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
import { SurveyAnalysisNavigation } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/components/SurveyAnalysisNavigation";
|
import { SurveyAnalysisNavigation } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/components/SurveyAnalysisNavigation";
|
||||||
import { SummaryPage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SummaryPage";
|
import { SummaryPage } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/components/SummaryPage";
|
||||||
import { getSurveySummary } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/lib/surveySummary";
|
import { getSurveySummary } from "@/app/(app)/environments/[environmentId]/surveys/[surveyId]/(analysis)/summary/lib/surveySummary";
|
||||||
import { DEFAULT_LOCALE, WEBAPP_URL } from "@/lib/constants";
|
import { DEFAULT_LOCALE } from "@/lib/constants";
|
||||||
import { getEnvironment } from "@/lib/environment/service";
|
import { getEnvironment } from "@/lib/environment/service";
|
||||||
|
import { getPublicDomain } from "@/lib/getPublicUrl";
|
||||||
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
import { getProjectByEnvironmentId } from "@/lib/project/service";
|
||||||
import { getSurvey, getSurveyIdByResultShareKey } from "@/lib/survey/service";
|
import { getSurvey, getSurveyIdByResultShareKey } from "@/lib/survey/service";
|
||||||
import { PageContentWrapper } from "@/modules/ui/components/page-content-wrapper";
|
import { PageContentWrapper } from "@/modules/ui/components/page-content-wrapper";
|
||||||
@@ -50,6 +51,8 @@ const Page = async (props: SummaryPageProps) => {
|
|||||||
// Fetch initial survey summary data on the server to prevent duplicate API calls during hydration
|
// Fetch initial survey summary data on the server to prevent duplicate API calls during hydration
|
||||||
const initialSurveySummary = await getSurveySummary(surveyId);
|
const initialSurveySummary = await getSurveySummary(surveyId);
|
||||||
|
|
||||||
|
const publicDomain = getPublicDomain();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex w-full justify-center">
|
<div className="flex w-full justify-center">
|
||||||
<PageContentWrapper className="w-full">
|
<PageContentWrapper className="w-full">
|
||||||
@@ -60,7 +63,7 @@ const Page = async (props: SummaryPageProps) => {
|
|||||||
environment={environment}
|
environment={environment}
|
||||||
survey={survey}
|
survey={survey}
|
||||||
surveyId={survey.id}
|
surveyId={survey.id}
|
||||||
webAppUrl={WEBAPP_URL}
|
publicDomain={publicDomain}
|
||||||
isReadOnly={true}
|
isReadOnly={true}
|
||||||
locale={DEFAULT_LOCALE}
|
locale={DEFAULT_LOCALE}
|
||||||
initialSurveySummary={initialSurveySummary}
|
initialSurveySummary={initialSurveySummary}
|
||||||
|
|||||||
@@ -13,7 +13,8 @@ import {
|
|||||||
ZIntegrationAirtableTokenSchema,
|
ZIntegrationAirtableTokenSchema,
|
||||||
} from "@formbricks/types/integration/airtable";
|
} from "@formbricks/types/integration/airtable";
|
||||||
import { AIRTABLE_CLIENT_ID, AIRTABLE_MESSAGE_LIMIT } from "../constants";
|
import { AIRTABLE_CLIENT_ID, AIRTABLE_MESSAGE_LIMIT } from "../constants";
|
||||||
import { createOrUpdateIntegration, deleteIntegration, getIntegrationByType } from "../integration/service";
|
import { createOrUpdateIntegration, getIntegrationByType } from "../integration/service";
|
||||||
|
import { delay } from "../utils/promises";
|
||||||
import { truncateText } from "../utils/strings";
|
import { truncateText } from "../utils/strings";
|
||||||
|
|
||||||
export const getBases = async (key: string) => {
|
export const getBases = async (key: string) => {
|
||||||
@@ -99,7 +100,11 @@ export const getAirtableToken = async (environmentId: string) => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (!newToken) {
|
if (!newToken) {
|
||||||
throw new Error("Failed to create new token");
|
logger.error("Failed to fetch new Airtable token", {
|
||||||
|
environmentId,
|
||||||
|
airtableIntegration,
|
||||||
|
});
|
||||||
|
throw new Error("Failed to fetch new Airtable token");
|
||||||
}
|
}
|
||||||
|
|
||||||
await createOrUpdateIntegration(environmentId, {
|
await createOrUpdateIntegration(environmentId, {
|
||||||
@@ -116,9 +121,11 @@ export const getAirtableToken = async (environmentId: string) => {
|
|||||||
|
|
||||||
return access_token;
|
return access_token;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
await deleteIntegration(environmentId);
|
logger.error("Failed to get Airtable token", {
|
||||||
|
environmentId,
|
||||||
throw new Error("invalid token");
|
error,
|
||||||
|
});
|
||||||
|
throw new Error("Failed to get Airtable token");
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -178,6 +185,18 @@ const addField = async (
|
|||||||
return await req.json();
|
return await req.json();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getExistingFields = async (key: TIntegrationAirtableCredential, baseId: string, tableId: string) => {
|
||||||
|
const req = await tableFetcher(key, baseId);
|
||||||
|
const tables = ZIntegrationAirtableTablesWithFields.parse(req).tables;
|
||||||
|
const currentTable = tables.find((t) => t.id === tableId);
|
||||||
|
|
||||||
|
if (!currentTable) {
|
||||||
|
throw new Error(`Table with ID ${tableId} not found`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Set(currentTable.fields.map((f) => f.name));
|
||||||
|
};
|
||||||
|
|
||||||
export const writeData = async (
|
export const writeData = async (
|
||||||
key: TIntegrationAirtableCredential,
|
key: TIntegrationAirtableCredential,
|
||||||
configData: TIntegrationAirtableConfigData,
|
configData: TIntegrationAirtableConfigData,
|
||||||
@@ -186,6 +205,7 @@ export const writeData = async (
|
|||||||
const responses = values[0];
|
const responses = values[0];
|
||||||
const questions = values[1];
|
const questions = values[1];
|
||||||
|
|
||||||
|
// 1) Build the record payload
|
||||||
const data: Record<string, string> = {};
|
const data: Record<string, string> = {};
|
||||||
for (let i = 0; i < questions.length; i++) {
|
for (let i = 0; i < questions.length; i++) {
|
||||||
data[questions[i]] =
|
data[questions[i]] =
|
||||||
@@ -194,34 +214,73 @@ export const writeData = async (
|
|||||||
: responses[i];
|
: responses[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
const req = await tableFetcher(key, configData.baseId);
|
// 2) Figure out which fields need creating
|
||||||
const tables = ZIntegrationAirtableTablesWithFields.parse(req).tables;
|
const existingFields = await getExistingFields(key, configData.baseId, configData.tableId);
|
||||||
|
const fieldsToCreate = questions.filter((q) => !existingFields.has(q));
|
||||||
|
|
||||||
const currentTable = tables.find((table) => table.id === configData.tableId);
|
// 3) Create any missing fields with throttling to respect Airtable's 5 req/sec per base limit
|
||||||
if (currentTable) {
|
if (fieldsToCreate.length > 0) {
|
||||||
const currentFields = new Set(currentTable.fields.map((field) => field.name));
|
// Sequential processing with delays
|
||||||
const fieldsToCreate = new Set<string>();
|
const DELAY_BETWEEN_REQUESTS = 250; // 250ms = 4 requests per second (staying under 5/sec limit)
|
||||||
for (const field of questions) {
|
|
||||||
const hasField = currentFields.has(field);
|
for (let i = 0; i < fieldsToCreate.length; i++) {
|
||||||
if (!hasField) {
|
const fieldName = fieldsToCreate[i];
|
||||||
fieldsToCreate.add(field);
|
|
||||||
|
const createRes = await addField(key, configData.baseId, configData.tableId, {
|
||||||
|
name: fieldName,
|
||||||
|
type: "singleLineText",
|
||||||
|
});
|
||||||
|
|
||||||
|
if (createRes?.error) {
|
||||||
|
throw new Error(`Failed to create field "${fieldName}": ${JSON.stringify(createRes)}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add delay between requests (except for the last one)
|
||||||
|
if (i < fieldsToCreate.length - 1) {
|
||||||
|
await delay(DELAY_BETWEEN_REQUESTS);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fieldsToCreate.size > 0) {
|
// 4) Wait for the new fields to show up
|
||||||
const createFieldPromise: Promise<any>[] = [];
|
await waitForFieldsToExist(key, configData, fieldsToCreate);
|
||||||
fieldsToCreate.forEach((fieldName) => {
|
}
|
||||||
createFieldPromise.push(
|
|
||||||
addField(key, configData.baseId, configData.tableId, {
|
|
||||||
name: fieldName,
|
|
||||||
type: "singleLineText",
|
|
||||||
})
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
await Promise.all(createFieldPromise);
|
// 5) Finally, add the records
|
||||||
|
await addRecords(key, configData.baseId, configData.tableId, data);
|
||||||
|
};
|
||||||
|
|
||||||
|
async function waitForFieldsToExist(
|
||||||
|
key: TIntegrationAirtableCredential,
|
||||||
|
configData: TIntegrationAirtableConfigData,
|
||||||
|
fieldNames: string[],
|
||||||
|
maxRetries = 5,
|
||||||
|
intervalMs = 2000
|
||||||
|
) {
|
||||||
|
let existingFields: Set<string> = new Set(),
|
||||||
|
missingFields: string[] = [];
|
||||||
|
|
||||||
|
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
||||||
|
existingFields = await getExistingFields(key, configData.baseId, configData.tableId);
|
||||||
|
missingFields = fieldNames.filter((f) => !existingFields.has(f));
|
||||||
|
|
||||||
|
if (missingFields.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (attempt < maxRetries) {
|
||||||
|
logger.error(
|
||||||
|
`Attempt ${attempt}/${maxRetries}: ${missingFields.length} field(s) still missing [${missingFields.join(
|
||||||
|
", "
|
||||||
|
)}], retrying in ${intervalMs / 1000}s…`
|
||||||
|
);
|
||||||
|
|
||||||
|
await new Promise((r) => setTimeout(r, intervalMs));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await addRecords(key, configData.baseId, configData.tableId, data);
|
throw new Error(
|
||||||
};
|
`Timed out waiting for ${missingFields.length} field(s) [${missingFields.join(
|
||||||
|
", "
|
||||||
|
)}] to become available. Available fields: [${Array.from(existingFields).join(", ")}]`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|||||||
@@ -13,8 +13,6 @@ export const E2E_TESTING = env.E2E_TESTING === "1";
|
|||||||
export const WEBAPP_URL =
|
export const WEBAPP_URL =
|
||||||
env.WEBAPP_URL || (env.VERCEL_URL ? `https://${env.VERCEL_URL}` : false) || "http://localhost:3000";
|
env.WEBAPP_URL || (env.VERCEL_URL ? `https://${env.VERCEL_URL}` : false) || "http://localhost:3000";
|
||||||
|
|
||||||
export const SURVEY_URL = env.SURVEY_URL;
|
|
||||||
|
|
||||||
// encryption keys
|
// encryption keys
|
||||||
export const ENCRYPTION_KEY = env.ENCRYPTION_KEY;
|
export const ENCRYPTION_KEY = env.ENCRYPTION_KEY;
|
||||||
|
|
||||||
|
|||||||
+18
-2
@@ -85,7 +85,23 @@ export const env = createEnv({
|
|||||||
SMTP_REJECT_UNAUTHORIZED_TLS: z.enum(["1", "0"]).optional(),
|
SMTP_REJECT_UNAUTHORIZED_TLS: z.enum(["1", "0"]).optional(),
|
||||||
STRIPE_SECRET_KEY: z.string().optional(),
|
STRIPE_SECRET_KEY: z.string().optional(),
|
||||||
STRIPE_WEBHOOK_SECRET: z.string().optional(),
|
STRIPE_WEBHOOK_SECRET: z.string().optional(),
|
||||||
SURVEY_URL: z.string().optional(),
|
PUBLIC_URL: z
|
||||||
|
.string()
|
||||||
|
.url()
|
||||||
|
.refine(
|
||||||
|
(url) => {
|
||||||
|
try {
|
||||||
|
const parsed = new URL(url);
|
||||||
|
return parsed.host && parsed.host.length > 0;
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "PUBLIC_URL must be a valid URL with a proper host (e.g., https://example.com)",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.optional(),
|
||||||
TELEMETRY_DISABLED: z.enum(["1", "0"]).optional(),
|
TELEMETRY_DISABLED: z.enum(["1", "0"]).optional(),
|
||||||
TERMS_URL: z
|
TERMS_URL: z
|
||||||
.string()
|
.string()
|
||||||
@@ -190,7 +206,7 @@ export const env = createEnv({
|
|||||||
SMTP_AUTHENTICATED: process.env.SMTP_AUTHENTICATED,
|
SMTP_AUTHENTICATED: process.env.SMTP_AUTHENTICATED,
|
||||||
STRIPE_SECRET_KEY: process.env.STRIPE_SECRET_KEY,
|
STRIPE_SECRET_KEY: process.env.STRIPE_SECRET_KEY,
|
||||||
STRIPE_WEBHOOK_SECRET: process.env.STRIPE_WEBHOOK_SECRET,
|
STRIPE_WEBHOOK_SECRET: process.env.STRIPE_WEBHOOK_SECRET,
|
||||||
SURVEY_URL: process.env.SURVEY_URL,
|
PUBLIC_URL: process.env.PUBLIC_URL,
|
||||||
TELEMETRY_DISABLED: process.env.TELEMETRY_DISABLED,
|
TELEMETRY_DISABLED: process.env.TELEMETRY_DISABLED,
|
||||||
TURNSTILE_SECRET_KEY: process.env.TURNSTILE_SECRET_KEY,
|
TURNSTILE_SECRET_KEY: process.env.TURNSTILE_SECRET_KEY,
|
||||||
TURNSTILE_SITE_KEY: process.env.TURNSTILE_SITE_KEY,
|
TURNSTILE_SITE_KEY: process.env.TURNSTILE_SITE_KEY,
|
||||||
|
|||||||
@@ -65,7 +65,8 @@ export const validateSingleFile = (
|
|||||||
return !allowedFileExtensions || allowedFileExtensions.includes(extension as TAllowedFileExtension);
|
return !allowedFileExtensions || allowedFileExtensions.includes(extension as TAllowedFileExtension);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const validateFileUploads = (data: TResponseData, questions?: TSurveyQuestion[]): boolean => {
|
export const validateFileUploads = (data?: TResponseData, questions?: TSurveyQuestion[]): boolean => {
|
||||||
|
if (!data) return true;
|
||||||
for (const key of Object.keys(data)) {
|
for (const key of Object.keys(data)) {
|
||||||
const question = questions?.find((q) => q.id === key);
|
const question = questions?.find((q) => q.id === key);
|
||||||
if (!question || question.type !== TSurveyQuestionTypeEnum.FileUpload) continue;
|
if (!question || question.type !== TSurveyQuestionTypeEnum.FileUpload) continue;
|
||||||
|
|||||||
@@ -0,0 +1,44 @@
|
|||||||
|
import { beforeEach, describe, expect, test, vi } from "vitest";
|
||||||
|
|
||||||
|
// Mock constants module
|
||||||
|
const envMock = {
|
||||||
|
env: {
|
||||||
|
WEBAPP_URL: "http://localhost:3000",
|
||||||
|
PUBLIC_URL: undefined as string | undefined,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
vi.mock("@/lib/env", () => envMock);
|
||||||
|
|
||||||
|
describe("getPublicDomain", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
vi.resetModules();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return WEBAPP_URL when PUBLIC_URL is not set", async () => {
|
||||||
|
const { getPublicDomain } = await import("./getPublicUrl");
|
||||||
|
const domain = getPublicDomain();
|
||||||
|
expect(domain).toBe("http://localhost:3000");
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should return PUBLIC_URL when it is set", async () => {
|
||||||
|
envMock.env.PUBLIC_URL = "https://surveys.example.com";
|
||||||
|
const { getPublicDomain } = await import("./getPublicUrl");
|
||||||
|
const domain = getPublicDomain();
|
||||||
|
expect(domain).toBe("https://surveys.example.com");
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should handle empty string PUBLIC_URL by returning WEBAPP_URL", async () => {
|
||||||
|
envMock.env.PUBLIC_URL = "";
|
||||||
|
const { getPublicDomain } = await import("./getPublicUrl");
|
||||||
|
const domain = getPublicDomain();
|
||||||
|
expect(domain).toBe("http://localhost:3000");
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should handle undefined PUBLIC_URL by returning WEBAPP_URL", async () => {
|
||||||
|
envMock.env.PUBLIC_URL = undefined;
|
||||||
|
const { getPublicDomain } = await import("./getPublicUrl");
|
||||||
|
const domain = getPublicDomain();
|
||||||
|
expect(domain).toBe("http://localhost:3000");
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
import "server-only";
|
||||||
|
import { env } from "./env";
|
||||||
|
|
||||||
|
const WEBAPP_URL =
|
||||||
|
env.WEBAPP_URL ?? (env.VERCEL_URL ? `https://${env.VERCEL_URL}` : "") ?? "http://localhost:3000";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the public domain URL
|
||||||
|
* Uses PUBLIC_URL if set, otherwise falls back to WEBAPP_URL
|
||||||
|
*/
|
||||||
|
export const getPublicDomain = (): string => {
|
||||||
|
return env.PUBLIC_URL && env.PUBLIC_URL.trim() !== "" ? env.PUBLIC_URL : WEBAPP_URL;
|
||||||
|
};
|
||||||
@@ -1,46 +0,0 @@
|
|||||||
import { beforeEach, describe, expect, test, vi } from "vitest";
|
|
||||||
|
|
||||||
// Create a mock module for constants with proper types
|
|
||||||
const constantsMock = {
|
|
||||||
SURVEY_URL: undefined as string | undefined,
|
|
||||||
WEBAPP_URL: "http://localhost:3000" as string,
|
|
||||||
};
|
|
||||||
|
|
||||||
// Mock the constants module
|
|
||||||
vi.mock("./constants", () => constantsMock);
|
|
||||||
|
|
||||||
describe("getSurveyDomain", () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
// Reset the mock values before each test
|
|
||||||
constantsMock.SURVEY_URL = undefined;
|
|
||||||
constantsMock.WEBAPP_URL = "http://localhost:3000";
|
|
||||||
vi.resetModules();
|
|
||||||
});
|
|
||||||
|
|
||||||
test("should return WEBAPP_URL when SURVEY_URL is not set", async () => {
|
|
||||||
const { getSurveyDomain } = await import("./getSurveyUrl");
|
|
||||||
const domain = getSurveyDomain();
|
|
||||||
expect(domain).toBe("http://localhost:3000");
|
|
||||||
});
|
|
||||||
|
|
||||||
test("should return SURVEY_URL when it is set", async () => {
|
|
||||||
constantsMock.SURVEY_URL = "https://surveys.example.com";
|
|
||||||
const { getSurveyDomain } = await import("./getSurveyUrl");
|
|
||||||
const domain = getSurveyDomain();
|
|
||||||
expect(domain).toBe("https://surveys.example.com");
|
|
||||||
});
|
|
||||||
|
|
||||||
test("should handle empty string SURVEY_URL by returning WEBAPP_URL", async () => {
|
|
||||||
constantsMock.SURVEY_URL = "";
|
|
||||||
const { getSurveyDomain } = await import("./getSurveyUrl");
|
|
||||||
const domain = getSurveyDomain();
|
|
||||||
expect(domain).toBe("http://localhost:3000");
|
|
||||||
});
|
|
||||||
|
|
||||||
test("should handle undefined SURVEY_URL by returning WEBAPP_URL", async () => {
|
|
||||||
constantsMock.SURVEY_URL = undefined;
|
|
||||||
const { getSurveyDomain } = await import("./getSurveyUrl");
|
|
||||||
const domain = getSurveyDomain();
|
|
||||||
expect(domain).toBe("http://localhost:3000");
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
import "server-only";
|
|
||||||
import { SURVEY_URL, WEBAPP_URL } from "./constants";
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the base URL for public surveys
|
|
||||||
* Uses SURVEY_URL if set, otherwise falls back to WEBAPP_URL
|
|
||||||
*/
|
|
||||||
export const getSurveyDomain = (): string => {
|
|
||||||
return SURVEY_URL || WEBAPP_URL;
|
|
||||||
};
|
|
||||||
@@ -1,9 +1,16 @@
|
|||||||
import { BILLING_LIMITS, PROJECT_FEATURE_KEYS } from "@/lib/constants";
|
import { BILLING_LIMITS, PROJECT_FEATURE_KEYS } from "@/lib/constants";
|
||||||
|
import { updateUser } from "@/lib/user/service";
|
||||||
import { Prisma } from "@prisma/client";
|
import { Prisma } from "@prisma/client";
|
||||||
import { afterEach, describe, expect, test, vi } from "vitest";
|
import { afterEach, describe, expect, test, vi } from "vitest";
|
||||||
import { prisma } from "@formbricks/database";
|
import { prisma } from "@formbricks/database";
|
||||||
import { DatabaseError } from "@formbricks/types/errors";
|
import { DatabaseError } from "@formbricks/types/errors";
|
||||||
import { createOrganization, getOrganization, getOrganizationsByUserId, updateOrganization } from "./service";
|
import {
|
||||||
|
createOrganization,
|
||||||
|
getOrganization,
|
||||||
|
getOrganizationsByUserId,
|
||||||
|
subscribeOrganizationMembersToSurveyResponses,
|
||||||
|
updateOrganization,
|
||||||
|
} from "./service";
|
||||||
|
|
||||||
vi.mock("@formbricks/database", () => ({
|
vi.mock("@formbricks/database", () => ({
|
||||||
prisma: {
|
prisma: {
|
||||||
@@ -13,9 +20,16 @@ vi.mock("@formbricks/database", () => ({
|
|||||||
create: vi.fn(),
|
create: vi.fn(),
|
||||||
update: vi.fn(),
|
update: vi.fn(),
|
||||||
},
|
},
|
||||||
|
user: {
|
||||||
|
findUnique: vi.fn(),
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@/lib/user/service", () => ({
|
||||||
|
updateUser: vi.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
describe("Organization Service", () => {
|
describe("Organization Service", () => {
|
||||||
afterEach(() => {
|
afterEach(() => {
|
||||||
vi.clearAllMocks();
|
vi.clearAllMocks();
|
||||||
@@ -252,4 +266,62 @@ describe("Organization Service", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("subscribeOrganizationMembersToSurveyResponses", () => {
|
||||||
|
test("should subscribe user to survey responses when not unsubscribed", async () => {
|
||||||
|
const mockUser = {
|
||||||
|
id: "user-123",
|
||||||
|
notificationSettings: {
|
||||||
|
alert: { "existing-survey-id": true },
|
||||||
|
weeklySummary: {},
|
||||||
|
unsubscribedOrganizationIds: [], // User is subscribed to all organizations
|
||||||
|
},
|
||||||
|
} as any;
|
||||||
|
|
||||||
|
const surveyId = "survey-123";
|
||||||
|
const userId = "user-123";
|
||||||
|
const organizationId = "org-123";
|
||||||
|
|
||||||
|
vi.mocked(prisma.user.findUnique).mockResolvedValueOnce(mockUser);
|
||||||
|
vi.mocked(updateUser).mockResolvedValueOnce({} as any);
|
||||||
|
|
||||||
|
await subscribeOrganizationMembersToSurveyResponses(surveyId, userId, organizationId);
|
||||||
|
|
||||||
|
expect(prisma.user.findUnique).toHaveBeenCalledWith({
|
||||||
|
where: { id: userId },
|
||||||
|
});
|
||||||
|
expect(updateUser).toHaveBeenCalledWith(userId, {
|
||||||
|
notificationSettings: {
|
||||||
|
alert: {
|
||||||
|
"existing-survey-id": true,
|
||||||
|
"survey-123": true,
|
||||||
|
},
|
||||||
|
weeklySummary: {},
|
||||||
|
unsubscribedOrganizationIds: [],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should not subscribe user when unsubscribed from organization", async () => {
|
||||||
|
const mockUser = {
|
||||||
|
id: "user-123",
|
||||||
|
notificationSettings: {
|
||||||
|
alert: { "existing-survey-id": true },
|
||||||
|
weeklySummary: {},
|
||||||
|
unsubscribedOrganizationIds: ["org-123"], // User has unsubscribed from this organization
|
||||||
|
},
|
||||||
|
} as any;
|
||||||
|
|
||||||
|
const surveyId = "survey-123";
|
||||||
|
const userId = "user-123";
|
||||||
|
const organizationId = "org-123";
|
||||||
|
|
||||||
|
vi.mocked(prisma.user.findUnique).mockResolvedValueOnce(mockUser);
|
||||||
|
|
||||||
|
await subscribeOrganizationMembersToSurveyResponses(surveyId, userId, organizationId);
|
||||||
|
|
||||||
|
// Should not call updateUser because user is unsubscribed from this organization
|
||||||
|
expect(updateUser).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,10 +1,16 @@
|
|||||||
import { createId } from "@paralleldrive/cuid2";
|
import { createId } from "@paralleldrive/cuid2";
|
||||||
import { Prisma } from "@prisma/client";
|
import { OrganizationRole, Prisma, WidgetPlacement } from "@prisma/client";
|
||||||
import { afterEach, describe, expect, test, vi } from "vitest";
|
import { afterEach, describe, expect, test, vi } from "vitest";
|
||||||
import { prisma } from "@formbricks/database";
|
import { prisma } from "@formbricks/database";
|
||||||
import { DatabaseError, ValidationError } from "@formbricks/types/errors";
|
import { DatabaseError, ValidationError } from "@formbricks/types/errors";
|
||||||
import { ITEMS_PER_PAGE } from "../constants";
|
import { ITEMS_PER_PAGE } from "../constants";
|
||||||
import { getProject, getProjectByEnvironmentId, getProjects, getUserProjects } from "./service";
|
import {
|
||||||
|
getProject,
|
||||||
|
getProjectByEnvironmentId,
|
||||||
|
getProjects,
|
||||||
|
getUserProjectEnvironmentsByOrganizationIds,
|
||||||
|
getUserProjects,
|
||||||
|
} from "./service";
|
||||||
|
|
||||||
vi.mock("@formbricks/database", () => ({
|
vi.mock("@formbricks/database", () => ({
|
||||||
prisma: {
|
prisma: {
|
||||||
@@ -15,6 +21,7 @@ vi.mock("@formbricks/database", () => ({
|
|||||||
},
|
},
|
||||||
membership: {
|
membership: {
|
||||||
findFirst: vi.fn(),
|
findFirst: vi.fn(),
|
||||||
|
findMany: vi.fn(),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
}));
|
}));
|
||||||
@@ -35,13 +42,20 @@ describe("Project Service", () => {
|
|||||||
recontactDays: 0,
|
recontactDays: 0,
|
||||||
linkSurveyBranding: true,
|
linkSurveyBranding: true,
|
||||||
inAppSurveyBranding: true,
|
inAppSurveyBranding: true,
|
||||||
config: {},
|
config: {
|
||||||
placement: "bottomRight",
|
channel: null,
|
||||||
|
industry: null,
|
||||||
|
},
|
||||||
|
placement: WidgetPlacement.bottomRight,
|
||||||
clickOutsideClose: true,
|
clickOutsideClose: true,
|
||||||
darkOverlay: false,
|
darkOverlay: false,
|
||||||
environments: [],
|
environments: [],
|
||||||
styling: {},
|
styling: {
|
||||||
|
allowStyleOverwrite: true,
|
||||||
|
},
|
||||||
logo: null,
|
logo: null,
|
||||||
|
brandColor: null,
|
||||||
|
highlightBorderColor: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
vi.mocked(prisma.project.findUnique).mockResolvedValue(mockProject);
|
vi.mocked(prisma.project.findUnique).mockResolvedValue(mockProject);
|
||||||
@@ -86,13 +100,20 @@ describe("Project Service", () => {
|
|||||||
recontactDays: 0,
|
recontactDays: 0,
|
||||||
linkSurveyBranding: true,
|
linkSurveyBranding: true,
|
||||||
inAppSurveyBranding: true,
|
inAppSurveyBranding: true,
|
||||||
config: {},
|
config: {
|
||||||
placement: "bottomRight",
|
channel: null,
|
||||||
|
industry: null,
|
||||||
|
},
|
||||||
|
placement: WidgetPlacement.bottomRight,
|
||||||
clickOutsideClose: true,
|
clickOutsideClose: true,
|
||||||
darkOverlay: false,
|
darkOverlay: false,
|
||||||
environments: [],
|
environments: [],
|
||||||
styling: {},
|
styling: {
|
||||||
|
allowStyleOverwrite: true,
|
||||||
|
},
|
||||||
logo: null,
|
logo: null,
|
||||||
|
brandColor: null,
|
||||||
|
highlightBorderColor: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
vi.mocked(prisma.project.findFirst).mockResolvedValue(mockProject);
|
vi.mocked(prisma.project.findFirst).mockResolvedValue(mockProject);
|
||||||
@@ -144,13 +165,20 @@ describe("Project Service", () => {
|
|||||||
recontactDays: 0,
|
recontactDays: 0,
|
||||||
linkSurveyBranding: true,
|
linkSurveyBranding: true,
|
||||||
inAppSurveyBranding: true,
|
inAppSurveyBranding: true,
|
||||||
config: {},
|
config: {
|
||||||
placement: "bottomRight",
|
channel: null,
|
||||||
|
industry: null,
|
||||||
|
},
|
||||||
|
placement: WidgetPlacement.bottomRight,
|
||||||
clickOutsideClose: true,
|
clickOutsideClose: true,
|
||||||
darkOverlay: false,
|
darkOverlay: false,
|
||||||
environments: [],
|
environments: [],
|
||||||
styling: {},
|
styling: {
|
||||||
|
allowStyleOverwrite: true,
|
||||||
|
},
|
||||||
logo: null,
|
logo: null,
|
||||||
|
brandColor: null,
|
||||||
|
highlightBorderColor: null,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: createId(),
|
id: createId(),
|
||||||
@@ -162,23 +190,29 @@ describe("Project Service", () => {
|
|||||||
recontactDays: 0,
|
recontactDays: 0,
|
||||||
linkSurveyBranding: true,
|
linkSurveyBranding: true,
|
||||||
inAppSurveyBranding: true,
|
inAppSurveyBranding: true,
|
||||||
config: {},
|
config: {
|
||||||
placement: "bottomRight",
|
channel: null,
|
||||||
|
industry: null,
|
||||||
|
},
|
||||||
|
placement: WidgetPlacement.bottomRight,
|
||||||
clickOutsideClose: true,
|
clickOutsideClose: true,
|
||||||
darkOverlay: false,
|
darkOverlay: false,
|
||||||
environments: [],
|
environments: [],
|
||||||
styling: {},
|
styling: {
|
||||||
|
allowStyleOverwrite: true,
|
||||||
|
},
|
||||||
logo: null,
|
logo: null,
|
||||||
|
brandColor: null,
|
||||||
|
highlightBorderColor: null,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
vi.mocked(prisma.membership.findFirst).mockResolvedValue({
|
vi.mocked(prisma.membership.findFirst).mockResolvedValue({
|
||||||
id: createId(),
|
|
||||||
userId,
|
userId,
|
||||||
organizationId,
|
organizationId,
|
||||||
role: "admin",
|
role: OrganizationRole.owner,
|
||||||
createdAt: new Date(),
|
accepted: true,
|
||||||
updatedAt: new Date(),
|
deprecatedRole: null,
|
||||||
});
|
});
|
||||||
|
|
||||||
vi.mocked(prisma.project.findMany).mockResolvedValue(mockProjects);
|
vi.mocked(prisma.project.findMany).mockResolvedValue(mockProjects);
|
||||||
@@ -210,23 +244,29 @@ describe("Project Service", () => {
|
|||||||
recontactDays: 0,
|
recontactDays: 0,
|
||||||
linkSurveyBranding: true,
|
linkSurveyBranding: true,
|
||||||
inAppSurveyBranding: true,
|
inAppSurveyBranding: true,
|
||||||
config: {},
|
config: {
|
||||||
placement: "bottomRight",
|
channel: null,
|
||||||
|
industry: null,
|
||||||
|
},
|
||||||
|
placement: WidgetPlacement.bottomRight,
|
||||||
clickOutsideClose: true,
|
clickOutsideClose: true,
|
||||||
darkOverlay: false,
|
darkOverlay: false,
|
||||||
environments: [],
|
environments: [],
|
||||||
styling: {},
|
styling: {
|
||||||
|
allowStyleOverwrite: true,
|
||||||
|
},
|
||||||
logo: null,
|
logo: null,
|
||||||
|
brandColor: null,
|
||||||
|
highlightBorderColor: null,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
vi.mocked(prisma.membership.findFirst).mockResolvedValue({
|
vi.mocked(prisma.membership.findFirst).mockResolvedValue({
|
||||||
id: createId(),
|
|
||||||
userId,
|
userId,
|
||||||
organizationId,
|
organizationId,
|
||||||
role: "member",
|
role: OrganizationRole.member,
|
||||||
createdAt: new Date(),
|
accepted: true,
|
||||||
updatedAt: new Date(),
|
deprecatedRole: null,
|
||||||
});
|
});
|
||||||
|
|
||||||
vi.mocked(prisma.project.findMany).mockResolvedValue(mockProjects);
|
vi.mocked(prisma.project.findMany).mockResolvedValue(mockProjects);
|
||||||
@@ -278,23 +318,29 @@ describe("Project Service", () => {
|
|||||||
recontactDays: 0,
|
recontactDays: 0,
|
||||||
linkSurveyBranding: true,
|
linkSurveyBranding: true,
|
||||||
inAppSurveyBranding: true,
|
inAppSurveyBranding: true,
|
||||||
config: {},
|
config: {
|
||||||
placement: "bottomRight",
|
channel: null,
|
||||||
|
industry: null,
|
||||||
|
},
|
||||||
|
placement: WidgetPlacement.bottomRight,
|
||||||
clickOutsideClose: true,
|
clickOutsideClose: true,
|
||||||
darkOverlay: false,
|
darkOverlay: false,
|
||||||
environments: [],
|
environments: [],
|
||||||
styling: {},
|
styling: {
|
||||||
|
allowStyleOverwrite: true,
|
||||||
|
},
|
||||||
logo: null,
|
logo: null,
|
||||||
|
brandColor: null,
|
||||||
|
highlightBorderColor: null,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
vi.mocked(prisma.membership.findFirst).mockResolvedValue({
|
vi.mocked(prisma.membership.findFirst).mockResolvedValue({
|
||||||
id: createId(),
|
|
||||||
userId,
|
userId,
|
||||||
organizationId,
|
organizationId,
|
||||||
role: "admin",
|
role: OrganizationRole.owner,
|
||||||
createdAt: new Date(),
|
accepted: true,
|
||||||
updatedAt: new Date(),
|
deprecatedRole: null,
|
||||||
});
|
});
|
||||||
|
|
||||||
vi.mocked(prisma.project.findMany).mockResolvedValue(mockProjects);
|
vi.mocked(prisma.project.findMany).mockResolvedValue(mockProjects);
|
||||||
@@ -326,13 +372,20 @@ describe("Project Service", () => {
|
|||||||
recontactDays: 0,
|
recontactDays: 0,
|
||||||
linkSurveyBranding: true,
|
linkSurveyBranding: true,
|
||||||
inAppSurveyBranding: true,
|
inAppSurveyBranding: true,
|
||||||
config: {},
|
config: {
|
||||||
placement: "bottomRight",
|
channel: null,
|
||||||
|
industry: null,
|
||||||
|
},
|
||||||
|
placement: WidgetPlacement.bottomRight,
|
||||||
clickOutsideClose: true,
|
clickOutsideClose: true,
|
||||||
darkOverlay: false,
|
darkOverlay: false,
|
||||||
environments: [],
|
environments: [],
|
||||||
styling: {},
|
styling: {
|
||||||
|
allowStyleOverwrite: true,
|
||||||
|
},
|
||||||
logo: null,
|
logo: null,
|
||||||
|
brandColor: null,
|
||||||
|
highlightBorderColor: null,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: createId(),
|
id: createId(),
|
||||||
@@ -344,13 +397,20 @@ describe("Project Service", () => {
|
|||||||
recontactDays: 0,
|
recontactDays: 0,
|
||||||
linkSurveyBranding: true,
|
linkSurveyBranding: true,
|
||||||
inAppSurveyBranding: true,
|
inAppSurveyBranding: true,
|
||||||
config: {},
|
config: {
|
||||||
placement: "bottomRight",
|
channel: null,
|
||||||
|
industry: null,
|
||||||
|
},
|
||||||
|
placement: WidgetPlacement.bottomRight,
|
||||||
clickOutsideClose: true,
|
clickOutsideClose: true,
|
||||||
darkOverlay: false,
|
darkOverlay: false,
|
||||||
environments: [],
|
environments: [],
|
||||||
styling: {},
|
styling: {
|
||||||
|
allowStyleOverwrite: true,
|
||||||
|
},
|
||||||
logo: null,
|
logo: null,
|
||||||
|
brandColor: null,
|
||||||
|
highlightBorderColor: null,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -382,13 +442,20 @@ describe("Project Service", () => {
|
|||||||
recontactDays: 0,
|
recontactDays: 0,
|
||||||
linkSurveyBranding: true,
|
linkSurveyBranding: true,
|
||||||
inAppSurveyBranding: true,
|
inAppSurveyBranding: true,
|
||||||
config: {},
|
config: {
|
||||||
placement: "bottomRight",
|
channel: null,
|
||||||
|
industry: null,
|
||||||
|
},
|
||||||
|
placement: WidgetPlacement.bottomRight,
|
||||||
clickOutsideClose: true,
|
clickOutsideClose: true,
|
||||||
darkOverlay: false,
|
darkOverlay: false,
|
||||||
environments: [],
|
environments: [],
|
||||||
styling: {},
|
styling: {
|
||||||
|
allowStyleOverwrite: true,
|
||||||
|
},
|
||||||
logo: null,
|
logo: null,
|
||||||
|
brandColor: null,
|
||||||
|
highlightBorderColor: null,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -418,4 +485,199 @@ describe("Project Service", () => {
|
|||||||
|
|
||||||
await expect(getProjects(organizationId)).rejects.toThrow(DatabaseError);
|
await expect(getProjects(organizationId)).rejects.toThrow(DatabaseError);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("getProjectsByOrganizationIds should return projects for given organization IDs", async () => {
|
||||||
|
const organizationId1 = createId();
|
||||||
|
const organizationId2 = createId();
|
||||||
|
const userId = createId();
|
||||||
|
const mockProjects = [
|
||||||
|
{
|
||||||
|
environments: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
environments: [],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
vi.mocked(prisma.membership.findMany).mockResolvedValue([
|
||||||
|
{
|
||||||
|
userId,
|
||||||
|
organizationId: organizationId1,
|
||||||
|
role: "owner" as any,
|
||||||
|
accepted: true,
|
||||||
|
deprecatedRole: null,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
userId,
|
||||||
|
organizationId: organizationId2,
|
||||||
|
role: "owner" as any,
|
||||||
|
accepted: true,
|
||||||
|
deprecatedRole: null,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
vi.mocked(prisma.project.findMany).mockResolvedValue(mockProjects as any);
|
||||||
|
|
||||||
|
const result = await getUserProjectEnvironmentsByOrganizationIds(
|
||||||
|
[organizationId1, organizationId2],
|
||||||
|
userId
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(result).toEqual(mockProjects);
|
||||||
|
expect(prisma.project.findMany).toHaveBeenCalledWith({
|
||||||
|
where: {
|
||||||
|
OR: [{ organizationId: organizationId1 }, { organizationId: organizationId2 }],
|
||||||
|
},
|
||||||
|
select: { environments: true },
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("getProjectsByOrganizationIds should return empty array when no projects are found", async () => {
|
||||||
|
const organizationId1 = createId();
|
||||||
|
const organizationId2 = createId();
|
||||||
|
const userId = createId();
|
||||||
|
|
||||||
|
vi.mocked(prisma.membership.findMany).mockResolvedValue([
|
||||||
|
{
|
||||||
|
userId,
|
||||||
|
organizationId: organizationId1,
|
||||||
|
role: "owner" as any,
|
||||||
|
accepted: true,
|
||||||
|
deprecatedRole: null,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
userId,
|
||||||
|
organizationId: organizationId2,
|
||||||
|
role: "owner" as any,
|
||||||
|
accepted: true,
|
||||||
|
deprecatedRole: null,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
vi.mocked(prisma.project.findMany).mockResolvedValue([]);
|
||||||
|
|
||||||
|
const result = await getUserProjectEnvironmentsByOrganizationIds(
|
||||||
|
[organizationId1, organizationId2],
|
||||||
|
userId
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(result).toEqual([]);
|
||||||
|
expect(prisma.project.findMany).toHaveBeenCalledWith({
|
||||||
|
where: {
|
||||||
|
OR: [{ organizationId: organizationId1 }, { organizationId: organizationId2 }],
|
||||||
|
},
|
||||||
|
select: { environments: true },
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("getProjectsByOrganizationIds should throw DatabaseError when prisma throws", async () => {
|
||||||
|
const organizationId1 = createId();
|
||||||
|
const organizationId2 = createId();
|
||||||
|
const userId = createId();
|
||||||
|
const prismaError = new Prisma.PrismaClientKnownRequestError("Database error", {
|
||||||
|
code: "P2002",
|
||||||
|
clientVersion: "5.0.0",
|
||||||
|
});
|
||||||
|
|
||||||
|
vi.mocked(prisma.membership.findMany).mockResolvedValue([
|
||||||
|
{
|
||||||
|
userId,
|
||||||
|
organizationId: organizationId1,
|
||||||
|
role: "owner" as any,
|
||||||
|
accepted: true,
|
||||||
|
deprecatedRole: null,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
vi.mocked(prisma.project.findMany).mockRejectedValue(prismaError);
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
getUserProjectEnvironmentsByOrganizationIds([organizationId1, organizationId2], userId)
|
||||||
|
).rejects.toThrow(DatabaseError);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("getProjectsByOrganizationIds should throw ValidationError with wrong input", async () => {
|
||||||
|
const userId = createId();
|
||||||
|
await expect(getUserProjectEnvironmentsByOrganizationIds(["wrong-id"], userId)).rejects.toThrow(
|
||||||
|
ValidationError
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("getProjectsByOrganizationIds should return empty array when user has no memberships", async () => {
|
||||||
|
const organizationId1 = createId();
|
||||||
|
const organizationId2 = createId();
|
||||||
|
const userId = createId();
|
||||||
|
|
||||||
|
// Mock no memberships found
|
||||||
|
vi.mocked(prisma.membership.findMany).mockResolvedValue([]);
|
||||||
|
|
||||||
|
const result = await getUserProjectEnvironmentsByOrganizationIds(
|
||||||
|
[organizationId1, organizationId2],
|
||||||
|
userId
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(result).toEqual([]);
|
||||||
|
expect(prisma.membership.findMany).toHaveBeenCalledWith({
|
||||||
|
where: {
|
||||||
|
userId,
|
||||||
|
organizationId: {
|
||||||
|
in: [organizationId1, organizationId2],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
// Should not call project.findMany when no memberships
|
||||||
|
expect(prisma.project.findMany).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("getProjectsByOrganizationIds should handle member role with team access", async () => {
|
||||||
|
const organizationId1 = createId();
|
||||||
|
const organizationId2 = createId();
|
||||||
|
const userId = createId();
|
||||||
|
const mockProjects = [
|
||||||
|
{
|
||||||
|
environments: [],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
// Mock membership where user is a member
|
||||||
|
vi.mocked(prisma.membership.findMany).mockResolvedValue([
|
||||||
|
{
|
||||||
|
userId,
|
||||||
|
organizationId: organizationId1,
|
||||||
|
role: "member" as any,
|
||||||
|
accepted: true,
|
||||||
|
deprecatedRole: null,
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
vi.mocked(prisma.project.findMany).mockResolvedValue(mockProjects as any);
|
||||||
|
|
||||||
|
const result = await getUserProjectEnvironmentsByOrganizationIds(
|
||||||
|
[organizationId1, organizationId2],
|
||||||
|
userId
|
||||||
|
);
|
||||||
|
|
||||||
|
expect(result).toEqual(mockProjects);
|
||||||
|
expect(prisma.project.findMany).toHaveBeenCalledWith({
|
||||||
|
where: {
|
||||||
|
OR: [
|
||||||
|
{
|
||||||
|
organizationId: organizationId1,
|
||||||
|
projectTeams: {
|
||||||
|
some: {
|
||||||
|
team: {
|
||||||
|
teamUsers: {
|
||||||
|
some: {
|
||||||
|
userId,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
select: { environments: true },
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -170,3 +170,67 @@ export const getOrganizationProjectsCount = reactCache(async (organizationId: st
|
|||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const getUserProjectEnvironmentsByOrganizationIds = reactCache(
|
||||||
|
async (organizationIds: string[], userId: string): Promise<Pick<TProject, "environments">[]> => {
|
||||||
|
validateInputs([organizationIds, ZId.array()], [userId, ZId]);
|
||||||
|
try {
|
||||||
|
if (organizationIds.length === 0) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const memberships = await prisma.membership.findMany({
|
||||||
|
where: {
|
||||||
|
userId,
|
||||||
|
organizationId: {
|
||||||
|
in: organizationIds,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (memberships.length === 0) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const whereConditions: Prisma.ProjectWhereInput[] = memberships.map((membership) => {
|
||||||
|
let projectWhereClause: Prisma.ProjectWhereInput = {
|
||||||
|
organizationId: membership.organizationId,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (membership.role === "member") {
|
||||||
|
projectWhereClause = {
|
||||||
|
...projectWhereClause,
|
||||||
|
projectTeams: {
|
||||||
|
some: {
|
||||||
|
team: {
|
||||||
|
teamUsers: {
|
||||||
|
some: {
|
||||||
|
userId,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return projectWhereClause;
|
||||||
|
});
|
||||||
|
|
||||||
|
const projects = await prisma.project.findMany({
|
||||||
|
where: {
|
||||||
|
OR: whereConditions,
|
||||||
|
},
|
||||||
|
select: { environments: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
return projects;
|
||||||
|
} catch (err) {
|
||||||
|
if (err instanceof Prisma.PrismaClientKnownRequestError) {
|
||||||
|
throw new DatabaseError(err.message);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|||||||
@@ -15,6 +15,15 @@ vi.mock("@aws-sdk/client-s3", () => ({
|
|||||||
GetObjectCommand: vi.fn(),
|
GetObjectCommand: vi.fn(),
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
vi.mock("@aws-sdk/s3-presigned-post", () => ({
|
||||||
|
createPresignedPost: vi.fn(() =>
|
||||||
|
Promise.resolve({
|
||||||
|
url: "https://test-bucket.s3.test-region.amazonaws.com",
|
||||||
|
fields: { key: "test-key", policy: "test-policy" },
|
||||||
|
})
|
||||||
|
),
|
||||||
|
}));
|
||||||
|
|
||||||
// Mock environment variables
|
// Mock environment variables
|
||||||
vi.mock("../constants", () => ({
|
vi.mock("../constants", () => ({
|
||||||
S3_ACCESS_KEY: "test-access-key",
|
S3_ACCESS_KEY: "test-access-key",
|
||||||
@@ -34,11 +43,32 @@ vi.mock("../constants", () => ({
|
|||||||
UPLOADS_DIR: "/tmp/uploads",
|
UPLOADS_DIR: "/tmp/uploads",
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
// Mock getPublicDomain
|
||||||
|
vi.mock("../getPublicUrl", () => ({
|
||||||
|
getPublicDomain: () => "https://public-domain.com",
|
||||||
|
}));
|
||||||
|
|
||||||
// Mock crypto functions
|
// Mock crypto functions
|
||||||
vi.mock("crypto", () => ({
|
vi.mock("crypto", () => ({
|
||||||
randomUUID: () => "test-uuid",
|
randomUUID: () => "test-uuid",
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
// Mock local signed url generation
|
||||||
|
vi.mock("../crypto", () => ({
|
||||||
|
generateLocalSignedUrl: () => ({
|
||||||
|
signature: "test-signature",
|
||||||
|
timestamp: 123456789,
|
||||||
|
uuid: "test-uuid",
|
||||||
|
}),
|
||||||
|
}));
|
||||||
|
|
||||||
|
// Mock env
|
||||||
|
vi.mock("../env", () => ({
|
||||||
|
env: {
|
||||||
|
S3_BUCKET_NAME: "test-bucket",
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
describe("Storage Service", () => {
|
describe("Storage Service", () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
vi.clearAllMocks();
|
vi.clearAllMocks();
|
||||||
@@ -131,4 +161,38 @@ describe("Storage Service", () => {
|
|||||||
await expect(putFile(fileName, fileBuffer, accessType, environmentId)).rejects.toThrow("Upload failed");
|
await expect(putFile(fileName, fileBuffer, accessType, environmentId)).rejects.toThrow("Upload failed");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("getUploadSignedUrl", () => {
|
||||||
|
let getUploadSignedUrl: any;
|
||||||
|
|
||||||
|
beforeEach(async () => {
|
||||||
|
const serviceModule = await import("./service");
|
||||||
|
getUploadSignedUrl = serviceModule.getUploadSignedUrl;
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should use PUBLIC_URL for public files with S3", async () => {
|
||||||
|
const result = await getUploadSignedUrl("test.jpg", "env123", "image/jpeg", "public");
|
||||||
|
|
||||||
|
expect(result.fileUrl).toContain("https://public-domain.com");
|
||||||
|
expect(result.fileUrl).toMatch(
|
||||||
|
/https:\/\/public-domain\.com\/storage\/env123\/public\/test--fid--test-uuid\.jpg/
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should use WEBAPP_URL for private files with S3", async () => {
|
||||||
|
const result = await getUploadSignedUrl("test.jpg", "env123", "image/jpeg", "private");
|
||||||
|
|
||||||
|
expect(result.fileUrl).toContain("http://test-webapp");
|
||||||
|
expect(result.fileUrl).toMatch(
|
||||||
|
/http:\/\/test-webapp\/storage\/env123\/private\/test--fid--test-uuid\.jpg/
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should contain signed URL and presigned fields for S3", async () => {
|
||||||
|
const result = await getUploadSignedUrl("test.jpg", "env123", "image/jpeg", "public");
|
||||||
|
|
||||||
|
expect(result.signedUrl).toBe("https://test-bucket.s3.test-region.amazonaws.com");
|
||||||
|
expect(result.presignedFields).toEqual({ key: "test-key", policy: "test-policy" });
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ import {
|
|||||||
} from "../constants";
|
} from "../constants";
|
||||||
import { generateLocalSignedUrl } from "../crypto";
|
import { generateLocalSignedUrl } from "../crypto";
|
||||||
import { env } from "../env";
|
import { env } from "../env";
|
||||||
|
import { getPublicDomain } from "../getPublicUrl";
|
||||||
|
|
||||||
// S3Client Singleton
|
// S3Client Singleton
|
||||||
let s3ClientInstance: S3Client | null = null;
|
let s3ClientInstance: S3Client | null = null;
|
||||||
@@ -165,6 +166,10 @@ export const getUploadSignedUrl = async (
|
|||||||
|
|
||||||
const updatedFileName = `${fileNameWithoutExtension}--fid--${randomUUID()}.${fileExtension}`;
|
const updatedFileName = `${fileNameWithoutExtension}--fid--${randomUUID()}.${fileExtension}`;
|
||||||
|
|
||||||
|
// Use PUBLIC_URL for public files, WEBAPP_URL for private files
|
||||||
|
const publicDomain = getPublicDomain();
|
||||||
|
const baseUrl = accessType === "public" ? getPublicDomain() : WEBAPP_URL;
|
||||||
|
|
||||||
// handle the local storage case first
|
// handle the local storage case first
|
||||||
if (!isS3Configured()) {
|
if (!isS3Configured()) {
|
||||||
try {
|
try {
|
||||||
@@ -173,7 +178,7 @@ export const getUploadSignedUrl = async (
|
|||||||
return {
|
return {
|
||||||
signedUrl:
|
signedUrl:
|
||||||
accessType === "private"
|
accessType === "private"
|
||||||
? new URL(`${WEBAPP_URL}/api/v1/client/${environmentId}/storage/local`).href
|
? new URL(`${publicDomain}/api/v1/client/${environmentId}/storage/local`).href
|
||||||
: new URL(`${WEBAPP_URL}/api/v1/management/storage/local`).href,
|
: new URL(`${WEBAPP_URL}/api/v1/management/storage/local`).href,
|
||||||
signingData: {
|
signingData: {
|
||||||
signature,
|
signature,
|
||||||
@@ -181,7 +186,7 @@ export const getUploadSignedUrl = async (
|
|||||||
uuid,
|
uuid,
|
||||||
},
|
},
|
||||||
updatedFileName,
|
updatedFileName,
|
||||||
fileUrl: new URL(`${WEBAPP_URL}/storage/${environmentId}/${accessType}/${updatedFileName}`).href,
|
fileUrl: new URL(`${baseUrl}/storage/${environmentId}/${accessType}/${updatedFileName}`).href,
|
||||||
};
|
};
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
throw err;
|
throw err;
|
||||||
@@ -200,7 +205,7 @@ export const getUploadSignedUrl = async (
|
|||||||
return {
|
return {
|
||||||
signedUrl,
|
signedUrl,
|
||||||
presignedFields,
|
presignedFields,
|
||||||
fileUrl: new URL(`${WEBAPP_URL}/storage/${environmentId}/${accessType}/${updatedFileName}`).href,
|
fileUrl: new URL(`${baseUrl}/storage/${environmentId}/${accessType}/${updatedFileName}`).href,
|
||||||
};
|
};
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
throw err;
|
throw err;
|
||||||
|
|||||||
@@ -197,13 +197,15 @@ export const getFallbackValues = (text: string): fallbacks => {
|
|||||||
|
|
||||||
// Transforms headlines in a text to their corresponding recall information.
|
// Transforms headlines in a text to their corresponding recall information.
|
||||||
export const headlineToRecall = (
|
export const headlineToRecall = (
|
||||||
text: string,
|
text: string | undefined,
|
||||||
recallItems: TSurveyRecallItem[],
|
recallItems: TSurveyRecallItem[],
|
||||||
fallbacks: fallbacks
|
fallbacks: fallbacks
|
||||||
): string => {
|
): string => {
|
||||||
|
if (!text) return "";
|
||||||
|
|
||||||
recallItems.forEach((recallItem) => {
|
recallItems.forEach((recallItem) => {
|
||||||
const recallInfo = `#recall:${recallItem.id}/fallback:${fallbacks[recallItem.id]}#`;
|
const recallInfo = `#recall:${recallItem.id}/fallback:${fallbacks[recallItem.id]}#`;
|
||||||
text = text.replace(`@${recallItem.label}`, recallInfo);
|
text = text?.replace(`@${recallItem.label}`, recallInfo);
|
||||||
});
|
});
|
||||||
return text;
|
return text;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -34,7 +34,8 @@
|
|||||||
"text": "Du kannst Dich jetzt mit deinem neuen Passwort einloggen"
|
"text": "Du kannst Dich jetzt mit deinem neuen Passwort einloggen"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"reset_password": "Passwort zurücksetzen"
|
"reset_password": "Passwort zurücksetzen",
|
||||||
|
"reset_password_description": "Du wirst abgemeldet, um dein Passwort zurückzusetzen."
|
||||||
},
|
},
|
||||||
"invite": {
|
"invite": {
|
||||||
"create_account": "Konto erstellen",
|
"create_account": "Konto erstellen",
|
||||||
@@ -79,7 +80,7 @@
|
|||||||
},
|
},
|
||||||
"signup_without_verification_success": {
|
"signup_without_verification_success": {
|
||||||
"user_successfully_created": "Benutzer erfolgreich erstellt",
|
"user_successfully_created": "Benutzer erfolgreich erstellt",
|
||||||
"user_successfully_created_description": "Dein neuer Benutzer wurde erfolgreich erstellt. Bitte klicke auf den untenstehenden Button und melde Dich in deinem Konto an."
|
"user_successfully_created_info": "Wir haben nach einem Konto gesucht, das mit {email} verknüpft ist. Wenn keines existierte, haben wir eines für Dich erstellt. Wenn bereits ein Konto existierte, wurden keine Änderungen vorgenommen. Bitte melde Dich unten an, um fortzufahren."
|
||||||
},
|
},
|
||||||
"testimonial_1": "Als open-source Firma ist uns Datenschutz extrem wichtig! Formbricks bietet die perfekte Mischung aus modernster Technologie und solidem Datenschutz.",
|
"testimonial_1": "Als open-source Firma ist uns Datenschutz extrem wichtig! Formbricks bietet die perfekte Mischung aus modernster Technologie und solidem Datenschutz.",
|
||||||
"testimonial_all_features_included": "Alle Funktionen enthalten",
|
"testimonial_all_features_included": "Alle Funktionen enthalten",
|
||||||
@@ -91,11 +92,10 @@
|
|||||||
"invalid_token": "Ungültiges Token ☹️",
|
"invalid_token": "Ungültiges Token ☹️",
|
||||||
"new_email_verification_success": "Wenn die Adresse gültig ist, wurde eine Bestätigungs-E-Mail gesendet.",
|
"new_email_verification_success": "Wenn die Adresse gültig ist, wurde eine Bestätigungs-E-Mail gesendet.",
|
||||||
"no_email_provided": "Keine E-Mail bereitgestellt",
|
"no_email_provided": "Keine E-Mail bereitgestellt",
|
||||||
"please_click_the_link_in_the_email_to_activate_your_account": "Bitte klicke auf den Link in der E-Mail, um dein Konto zu aktivieren.",
|
|
||||||
"please_confirm_your_email_address": "Bitte bestätige deine E-Mail-Adresse",
|
"please_confirm_your_email_address": "Bitte bestätige deine E-Mail-Adresse",
|
||||||
"resend_verification_email": "Bestätigungs-E-Mail erneut senden",
|
"resend_verification_email": "Bestätigungs-E-Mail erneut senden",
|
||||||
"verification_email_resent_successfully": "Bestätigungs-E-Mail gesendet! Bitte überprüfe dein Postfach.",
|
"verification_email_resent_successfully": "Bestätigungs-E-Mail gesendet! Bitte überprüfe dein Postfach.",
|
||||||
"we_sent_an_email_to": "Wir haben eine E-Mail an {email} gesendet",
|
"verification_email_successfully_sent_info": "Wenn ein Konto mit {email} verknüpft ist, haben wir einen Bestätigungslink an diese Adresse gesendet. Bitte überprüfe dein Postfach, um die Anmeldung abzuschließen.",
|
||||||
"you_didnt_receive_an_email_or_your_link_expired": "Hast Du keine E-Mail erhalten oder ist dein Link abgelaufen?"
|
"you_didnt_receive_an_email_or_your_link_expired": "Hast Du keine E-Mail erhalten oder ist dein Link abgelaufen?"
|
||||||
},
|
},
|
||||||
"verify": {
|
"verify": {
|
||||||
@@ -192,7 +192,6 @@
|
|||||||
"e_commerce": "E-Commerce",
|
"e_commerce": "E-Commerce",
|
||||||
"edit": "Bearbeiten",
|
"edit": "Bearbeiten",
|
||||||
"email": "E-Mail",
|
"email": "E-Mail",
|
||||||
"embed": "Einbetten",
|
|
||||||
"enterprise_license": "Enterprise Lizenz",
|
"enterprise_license": "Enterprise Lizenz",
|
||||||
"environment_not_found": "Umgebung nicht gefunden",
|
"environment_not_found": "Umgebung nicht gefunden",
|
||||||
"environment_notice": "Du befindest dich derzeit in der {environment}-Umgebung.",
|
"environment_notice": "Du befindest dich derzeit in der {environment}-Umgebung.",
|
||||||
@@ -310,7 +309,6 @@
|
|||||||
"project_not_found": "Projekt nicht gefunden",
|
"project_not_found": "Projekt nicht gefunden",
|
||||||
"project_permission_not_found": "Projekt-Berechtigung nicht gefunden",
|
"project_permission_not_found": "Projekt-Berechtigung nicht gefunden",
|
||||||
"projects": "Projekte",
|
"projects": "Projekte",
|
||||||
"projects_limit_reached": "Projektlimit erreicht",
|
|
||||||
"question": "Frage",
|
"question": "Frage",
|
||||||
"question_id": "Frage-ID",
|
"question_id": "Frage-ID",
|
||||||
"questions": "Fragen",
|
"questions": "Fragen",
|
||||||
@@ -1788,6 +1786,7 @@
|
|||||||
"setup_instructions": "Einrichtung",
|
"setup_instructions": "Einrichtung",
|
||||||
"setup_integrations": "Integrationen einrichten",
|
"setup_integrations": "Integrationen einrichten",
|
||||||
"share_results": "Ergebnisse teilen",
|
"share_results": "Ergebnisse teilen",
|
||||||
|
"share_survey": "Umfrage teilen",
|
||||||
"share_the_link": "Teile den Link",
|
"share_the_link": "Teile den Link",
|
||||||
"share_the_link_to_get_responses": "Teile den Link, um Antworten einzusammeln",
|
"share_the_link_to_get_responses": "Teile den Link, um Antworten einzusammeln",
|
||||||
"show_all_responses_that_match": "Zeige alle Antworten, die übereinstimmen",
|
"show_all_responses_that_match": "Zeige alle Antworten, die übereinstimmen",
|
||||||
@@ -2651,6 +2650,7 @@
|
|||||||
"professional_development_survey_description": "Bewerte die Zufriedenheit der Mitarbeiter mit beruflichen Entwicklungsmöglichkeiten.",
|
"professional_development_survey_description": "Bewerte die Zufriedenheit der Mitarbeiter mit beruflichen Entwicklungsmöglichkeiten.",
|
||||||
"professional_development_survey_name": "Berufliche Entwicklungsbewertung",
|
"professional_development_survey_name": "Berufliche Entwicklungsbewertung",
|
||||||
"professional_development_survey_question_1_choice_1": "Ja",
|
"professional_development_survey_question_1_choice_1": "Ja",
|
||||||
|
"professional_development_survey_question_1_choice_2": "Nein",
|
||||||
"professional_development_survey_question_1_headline": "Sind Sie an beruflichen Entwicklungsmöglichkeiten interessiert?",
|
"professional_development_survey_question_1_headline": "Sind Sie an beruflichen Entwicklungsmöglichkeiten interessiert?",
|
||||||
"professional_development_survey_question_2_choice_1": "Networking-Veranstaltungen",
|
"professional_development_survey_question_2_choice_1": "Networking-Veranstaltungen",
|
||||||
"professional_development_survey_question_2_choice_2": "Konferenzen oder Seminare",
|
"professional_development_survey_question_2_choice_2": "Konferenzen oder Seminare",
|
||||||
|
|||||||
@@ -34,7 +34,8 @@
|
|||||||
"text": "You can now log in with your new password"
|
"text": "You can now log in with your new password"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"reset_password": "Reset password"
|
"reset_password": "Reset password",
|
||||||
|
"reset_password_description": "You will be logged out to reset your password."
|
||||||
},
|
},
|
||||||
"invite": {
|
"invite": {
|
||||||
"create_account": "Create an account",
|
"create_account": "Create an account",
|
||||||
@@ -79,7 +80,7 @@
|
|||||||
},
|
},
|
||||||
"signup_without_verification_success": {
|
"signup_without_verification_success": {
|
||||||
"user_successfully_created": "User successfully created",
|
"user_successfully_created": "User successfully created",
|
||||||
"user_successfully_created_description": "Your new user has been created successfully. Please click the button below and sign in to your account."
|
"user_successfully_created_info": "We’ve checked for an account associated with {email}. If none existed, we’ve created one for you. If an account already existed, no changes were made. Please log in below to continue."
|
||||||
},
|
},
|
||||||
"testimonial_1": "We measure the clarity of our docs and learn from churn all on one platform. Great product, very responsive team!",
|
"testimonial_1": "We measure the clarity of our docs and learn from churn all on one platform. Great product, very responsive team!",
|
||||||
"testimonial_all_features_included": "All features included",
|
"testimonial_all_features_included": "All features included",
|
||||||
@@ -91,11 +92,10 @@
|
|||||||
"invalid_token": "Invalid token ☹️",
|
"invalid_token": "Invalid token ☹️",
|
||||||
"new_email_verification_success": "If the address is valid, a verification email has been sent.",
|
"new_email_verification_success": "If the address is valid, a verification email has been sent.",
|
||||||
"no_email_provided": "No email provided",
|
"no_email_provided": "No email provided",
|
||||||
"please_click_the_link_in_the_email_to_activate_your_account": "Please click the link in the email to activate your account.",
|
|
||||||
"please_confirm_your_email_address": "Please confirm your email address",
|
"please_confirm_your_email_address": "Please confirm your email address",
|
||||||
"resend_verification_email": "Resend verification email",
|
"resend_verification_email": "Resend verification email",
|
||||||
"verification_email_resent_successfully": "Verification email sent! Please check your inbox.",
|
"verification_email_resent_successfully": "Verification email sent! Please check your inbox.",
|
||||||
"we_sent_an_email_to": "We sent an email to {email}. ",
|
"verification_email_successfully_sent_info": "If there’s an account associated with {email}, we’ve sent a verification link to that address. Please check your inbox to complete the sign-up.",
|
||||||
"you_didnt_receive_an_email_or_your_link_expired": "You didn't receive an email or your link expired?"
|
"you_didnt_receive_an_email_or_your_link_expired": "You didn't receive an email or your link expired?"
|
||||||
},
|
},
|
||||||
"verify": {
|
"verify": {
|
||||||
@@ -192,7 +192,6 @@
|
|||||||
"e_commerce": "E-Commerce",
|
"e_commerce": "E-Commerce",
|
||||||
"edit": "Edit",
|
"edit": "Edit",
|
||||||
"email": "Email",
|
"email": "Email",
|
||||||
"embed": "Embed",
|
|
||||||
"enterprise_license": "Enterprise License",
|
"enterprise_license": "Enterprise License",
|
||||||
"environment_not_found": "Environment not found",
|
"environment_not_found": "Environment not found",
|
||||||
"environment_notice": "You're currently in the {environment} environment.",
|
"environment_notice": "You're currently in the {environment} environment.",
|
||||||
@@ -310,7 +309,6 @@
|
|||||||
"project_not_found": "Project not found",
|
"project_not_found": "Project not found",
|
||||||
"project_permission_not_found": "Project permission not found",
|
"project_permission_not_found": "Project permission not found",
|
||||||
"projects": "Projects",
|
"projects": "Projects",
|
||||||
"projects_limit_reached": "Projects limit reached",
|
|
||||||
"question": "Question",
|
"question": "Question",
|
||||||
"question_id": "Question ID",
|
"question_id": "Question ID",
|
||||||
"questions": "Questions",
|
"questions": "Questions",
|
||||||
@@ -1788,6 +1786,7 @@
|
|||||||
"setup_instructions": "Setup instructions",
|
"setup_instructions": "Setup instructions",
|
||||||
"setup_integrations": "Setup integrations",
|
"setup_integrations": "Setup integrations",
|
||||||
"share_results": "Share results",
|
"share_results": "Share results",
|
||||||
|
"share_survey": "Share survey",
|
||||||
"share_the_link": "Share the link",
|
"share_the_link": "Share the link",
|
||||||
"share_the_link_to_get_responses": "Share the link to get responses",
|
"share_the_link_to_get_responses": "Share the link to get responses",
|
||||||
"show_all_responses_that_match": "Show all responses that match",
|
"show_all_responses_that_match": "Show all responses that match",
|
||||||
@@ -1893,12 +1892,12 @@
|
|||||||
},
|
},
|
||||||
"s": {
|
"s": {
|
||||||
"check_inbox_or_spam": "Please also check your spam folder if you don't see the email in your inbox.",
|
"check_inbox_or_spam": "Please also check your spam folder if you don't see the email in your inbox.",
|
||||||
"completed": "This free & open-source survey has been closed.",
|
"completed": "This survey is closed.",
|
||||||
"create_your_own": "Create your own",
|
"create_your_own": "Create your own open-source survey",
|
||||||
"enter_pin": "This survey is protected. Enter the PIN below",
|
"enter_pin": "This survey is protected. Enter the PIN below",
|
||||||
"just_curious": "Just curious?",
|
"just_curious": "Just curious?",
|
||||||
"link_invalid": "This survey can only be taken by invitation.",
|
"link_invalid": "This survey can only be taken by invitation.",
|
||||||
"paused": "This free & open-source survey is temporarily paused.",
|
"paused": "This survey is temporarily paused.",
|
||||||
"please_try_again_with_the_original_link": "Please try again with the original link",
|
"please_try_again_with_the_original_link": "Please try again with the original link",
|
||||||
"preview_survey_questions": "Preview survey questions.",
|
"preview_survey_questions": "Preview survey questions.",
|
||||||
"question_preview": "Question Preview",
|
"question_preview": "Question Preview",
|
||||||
@@ -2651,6 +2650,7 @@
|
|||||||
"professional_development_survey_description": "Assess employee satisfaction with professional growth and development opportunities.",
|
"professional_development_survey_description": "Assess employee satisfaction with professional growth and development opportunities.",
|
||||||
"professional_development_survey_name": "Professional Development Survey",
|
"professional_development_survey_name": "Professional Development Survey",
|
||||||
"professional_development_survey_question_1_choice_1": "Yes",
|
"professional_development_survey_question_1_choice_1": "Yes",
|
||||||
|
"professional_development_survey_question_1_choice_2": "No",
|
||||||
"professional_development_survey_question_1_headline": "Are you interested in professional development activities?",
|
"professional_development_survey_question_1_headline": "Are you interested in professional development activities?",
|
||||||
"professional_development_survey_question_2_choice_1": "Networking events",
|
"professional_development_survey_question_2_choice_1": "Networking events",
|
||||||
"professional_development_survey_question_2_choice_2": "Conferences or seminars",
|
"professional_development_survey_question_2_choice_2": "Conferences or seminars",
|
||||||
|
|||||||
@@ -34,7 +34,8 @@
|
|||||||
"text": "Vous pouvez maintenant vous connecter avec votre nouveau mot de passe."
|
"text": "Vous pouvez maintenant vous connecter avec votre nouveau mot de passe."
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"reset_password": "Réinitialiser le mot de passe"
|
"reset_password": "Réinitialiser le mot de passe",
|
||||||
|
"reset_password_description": "Vous serez déconnecté pour réinitialiser votre mot de passe."
|
||||||
},
|
},
|
||||||
"invite": {
|
"invite": {
|
||||||
"create_account": "Créer un compte",
|
"create_account": "Créer un compte",
|
||||||
@@ -79,7 +80,7 @@
|
|||||||
},
|
},
|
||||||
"signup_without_verification_success": {
|
"signup_without_verification_success": {
|
||||||
"user_successfully_created": "Utilisateur créé avec succès",
|
"user_successfully_created": "Utilisateur créé avec succès",
|
||||||
"user_successfully_created_description": "Votre nouvel utilisateur a été créé avec succès. Veuillez cliquer sur le bouton ci-dessous et vous connecter à votre compte."
|
"user_successfully_created_info": "Nous avons vérifié s'il existait un compte associé à {email}. Si aucun n'existait, nous en avons créé un pour vous. Si un compte existait déjà, aucune modification n'a été apportée. Veuillez vous connecter ci-dessous pour continuer."
|
||||||
},
|
},
|
||||||
"testimonial_1": "Nous mesurons la clarté de nos documents et apprenons des abandons, le tout sur une seule plateforme. Excellent produit, équipe très réactive !",
|
"testimonial_1": "Nous mesurons la clarté de nos documents et apprenons des abandons, le tout sur une seule plateforme. Excellent produit, équipe très réactive !",
|
||||||
"testimonial_all_features_included": "Toutes les fonctionnalités incluses",
|
"testimonial_all_features_included": "Toutes les fonctionnalités incluses",
|
||||||
@@ -91,11 +92,10 @@
|
|||||||
"invalid_token": "Jeton non valide ☹️",
|
"invalid_token": "Jeton non valide ☹️",
|
||||||
"new_email_verification_success": "Si l'adresse est valide, un email de vérification a été envoyé.",
|
"new_email_verification_success": "Si l'adresse est valide, un email de vérification a été envoyé.",
|
||||||
"no_email_provided": "Aucun e-mail fourni",
|
"no_email_provided": "Aucun e-mail fourni",
|
||||||
"please_click_the_link_in_the_email_to_activate_your_account": "Veuillez cliquer sur le lien dans l'e-mail pour activer votre compte.",
|
|
||||||
"please_confirm_your_email_address": "Veuillez confirmer votre adresse e-mail.",
|
"please_confirm_your_email_address": "Veuillez confirmer votre adresse e-mail.",
|
||||||
"resend_verification_email": "Renvoyer l'email de vérification",
|
"resend_verification_email": "Renvoyer l'email de vérification",
|
||||||
"verification_email_resent_successfully": "E-mail de vérification envoyé ! Veuillez vérifier votre boîte de réception.",
|
"verification_email_resent_successfully": "E-mail de vérification envoyé ! Veuillez vérifier votre boîte de réception.",
|
||||||
"we_sent_an_email_to": "Nous avons envoyé un email à {email}",
|
"verification_email_successfully_sent_info": "Si un compte est associé à {email}, nous avons envoyé un lien de vérification à cette adresse. Veuillez vérifier votre boîte de réception pour terminer l'inscription.",
|
||||||
"you_didnt_receive_an_email_or_your_link_expired": "Vous n'avez pas reçu d'email ou votre lien a expiré ?"
|
"you_didnt_receive_an_email_or_your_link_expired": "Vous n'avez pas reçu d'email ou votre lien a expiré ?"
|
||||||
},
|
},
|
||||||
"verify": {
|
"verify": {
|
||||||
@@ -192,7 +192,6 @@
|
|||||||
"e_commerce": "E-commerce",
|
"e_commerce": "E-commerce",
|
||||||
"edit": "Modifier",
|
"edit": "Modifier",
|
||||||
"email": "Email",
|
"email": "Email",
|
||||||
"embed": "Intégrer",
|
|
||||||
"enterprise_license": "Licence d'entreprise",
|
"enterprise_license": "Licence d'entreprise",
|
||||||
"environment_not_found": "Environnement non trouvé",
|
"environment_not_found": "Environnement non trouvé",
|
||||||
"environment_notice": "Vous êtes actuellement dans l'environnement {environment}.",
|
"environment_notice": "Vous êtes actuellement dans l'environnement {environment}.",
|
||||||
@@ -310,7 +309,6 @@
|
|||||||
"project_not_found": "Projet non trouvé",
|
"project_not_found": "Projet non trouvé",
|
||||||
"project_permission_not_found": "Autorisation de projet non trouvée",
|
"project_permission_not_found": "Autorisation de projet non trouvée",
|
||||||
"projects": "Projets",
|
"projects": "Projets",
|
||||||
"projects_limit_reached": "Limite de projets atteinte",
|
|
||||||
"question": "Question",
|
"question": "Question",
|
||||||
"question_id": "ID de la question",
|
"question_id": "ID de la question",
|
||||||
"questions": "Questions",
|
"questions": "Questions",
|
||||||
@@ -1788,6 +1786,7 @@
|
|||||||
"setup_instructions": "Instructions d'installation",
|
"setup_instructions": "Instructions d'installation",
|
||||||
"setup_integrations": "Configurer les intégrations",
|
"setup_integrations": "Configurer les intégrations",
|
||||||
"share_results": "Partager les résultats",
|
"share_results": "Partager les résultats",
|
||||||
|
"share_survey": "Partager l'enquête",
|
||||||
"share_the_link": "Partager le lien",
|
"share_the_link": "Partager le lien",
|
||||||
"share_the_link_to_get_responses": "Partagez le lien pour obtenir des réponses",
|
"share_the_link_to_get_responses": "Partagez le lien pour obtenir des réponses",
|
||||||
"show_all_responses_that_match": "Afficher toutes les réponses correspondantes",
|
"show_all_responses_that_match": "Afficher toutes les réponses correspondantes",
|
||||||
@@ -2651,6 +2650,7 @@
|
|||||||
"professional_development_survey_description": "Évaluer la satisfaction des employés concernant les opportunités de croissance et de développement professionnel.",
|
"professional_development_survey_description": "Évaluer la satisfaction des employés concernant les opportunités de croissance et de développement professionnel.",
|
||||||
"professional_development_survey_name": "Sondage sur le développement professionnel",
|
"professional_development_survey_name": "Sondage sur le développement professionnel",
|
||||||
"professional_development_survey_question_1_choice_1": "Oui",
|
"professional_development_survey_question_1_choice_1": "Oui",
|
||||||
|
"professional_development_survey_question_1_choice_2": "Non",
|
||||||
"professional_development_survey_question_1_headline": "Êtes-vous intéressé par des activités de développement professionnel ?",
|
"professional_development_survey_question_1_headline": "Êtes-vous intéressé par des activités de développement professionnel ?",
|
||||||
"professional_development_survey_question_2_choice_1": "Événements de réseautage",
|
"professional_development_survey_question_2_choice_1": "Événements de réseautage",
|
||||||
"professional_development_survey_question_2_choice_2": "Conférences ou séminaires",
|
"professional_development_survey_question_2_choice_2": "Conférences ou séminaires",
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user