mirror of
				https://github.com/lingble/chatwoot.git
				synced 2025-11-04 04:57:51 +00:00 
			
		
		
		
	Merge branch 'develop' into pr/12259
This commit is contained in:
		@@ -1,6 +1,7 @@
 | 
			
		||||
version: 2.1
 | 
			
		||||
orbs:
 | 
			
		||||
  node: circleci/node@6.1.0
 | 
			
		||||
  qlty-orb: qltysh/qlty-orb@0.0
 | 
			
		||||
 | 
			
		||||
defaults: &defaults
 | 
			
		||||
  working_directory: ~/build
 | 
			
		||||
@@ -89,14 +90,6 @@ jobs:
 | 
			
		||||
          command: |
 | 
			
		||||
            source ~/.rvm/scripts/rvm
 | 
			
		||||
            bundle install
 | 
			
		||||
            # pnpm install
 | 
			
		||||
 | 
			
		||||
      - run:
 | 
			
		||||
          name: Download cc-test-reporter
 | 
			
		||||
          command: |
 | 
			
		||||
            mkdir -p ~/tmp
 | 
			
		||||
            curl -L https://codeclimate.com/downloads/test-reporter/test-reporter-latest-linux-amd64 > ~/tmp/cc-test-reporter
 | 
			
		||||
            chmod +x ~/tmp/cc-test-reporter
 | 
			
		||||
 | 
			
		||||
      # Swagger verification
 | 
			
		||||
      - run:
 | 
			
		||||
@@ -108,10 +101,11 @@ jobs:
 | 
			
		||||
              echo "ERROR: The swagger.json file is not in sync with the yaml specification. Run 'rake swagger:build' and commit 'swagger/swagger.json'."
 | 
			
		||||
              exit 1
 | 
			
		||||
            fi
 | 
			
		||||
            mkdir -p ~/tmp
 | 
			
		||||
            curl -L https://repo1.maven.org/maven2/org/openapitools/openapi-generator-cli/6.3.0/openapi-generator-cli-6.3.0.jar > ~/tmp/openapi-generator-cli-6.3.0.jar
 | 
			
		||||
            java -jar ~/tmp/openapi-generator-cli-6.3.0.jar validate -i swagger/swagger.json
 | 
			
		||||
 | 
			
		||||
      # we remove the FRONTED_URL from the .env before running the tests
 | 
			
		||||
      # Configure environment and database
 | 
			
		||||
      - run:
 | 
			
		||||
          name: Database Setup and Configure Environment Variables
 | 
			
		||||
          command: |
 | 
			
		||||
@@ -149,17 +143,11 @@ jobs:
 | 
			
		||||
          command: pnpm run eslint
 | 
			
		||||
 | 
			
		||||
      - run:
 | 
			
		||||
          name: Run frontend tests
 | 
			
		||||
          name: Run frontend tests (with coverage)
 | 
			
		||||
          command: |
 | 
			
		||||
            mkdir -p ~/build/coverage/frontend
 | 
			
		||||
            ~/tmp/cc-test-reporter before-build
 | 
			
		||||
            pnpm run test:coverage
 | 
			
		||||
 | 
			
		||||
      - run:
 | 
			
		||||
          name: Code Climate Test Coverage (Frontend)
 | 
			
		||||
          command: |
 | 
			
		||||
            ~/tmp/cc-test-reporter format-coverage -t lcov -o "~/build/coverage/frontend/codeclimate.frontend_$CIRCLE_NODE_INDEX.json"
 | 
			
		||||
 | 
			
		||||
      # Run backend tests
 | 
			
		||||
      - run:
 | 
			
		||||
          name: Run backend tests
 | 
			
		||||
@@ -167,18 +155,18 @@ jobs:
 | 
			
		||||
            mkdir -p ~/tmp/test-results/rspec
 | 
			
		||||
            mkdir -p ~/tmp/test-artifacts
 | 
			
		||||
            mkdir -p ~/build/coverage/backend
 | 
			
		||||
            ~/tmp/cc-test-reporter before-build
 | 
			
		||||
            TESTFILES=$(circleci tests glob "spec/**/*_spec.rb" | circleci tests split --split-by=timings)
 | 
			
		||||
            bundle exec rspec --format progress \
 | 
			
		||||
            bundle exec rspec -I ./spec --require coverage_helper --require spec_helper --format progress \
 | 
			
		||||
                              --format RspecJunitFormatter \
 | 
			
		||||
                              --out ~/tmp/test-results/rspec.xml \
 | 
			
		||||
                              -- ${TESTFILES}
 | 
			
		||||
          no_output_timeout: 30m
 | 
			
		||||
 | 
			
		||||
      - run:
 | 
			
		||||
          name: Code Climate Test Coverage (Backend)
 | 
			
		||||
          command: |
 | 
			
		||||
            ~/tmp/cc-test-reporter format-coverage -t simplecov -o "~/build/coverage/backend/codeclimate.$CIRCLE_NODE_INDEX.json"
 | 
			
		||||
      # Qlty coverage publish
 | 
			
		||||
      - qlty-orb/coverage_publish:
 | 
			
		||||
          files: |
 | 
			
		||||
            coverage/coverage.json
 | 
			
		||||
            coverage/lcov.info
 | 
			
		||||
 | 
			
		||||
      - run:
 | 
			
		||||
          name: List coverage directory contents
 | 
			
		||||
@@ -189,3 +177,7 @@ jobs:
 | 
			
		||||
          root: ~/build
 | 
			
		||||
          paths:
 | 
			
		||||
            - coverage
 | 
			
		||||
 | 
			
		||||
      - store_artifacts:
 | 
			
		||||
          path: coverage
 | 
			
		||||
          destination: coverage
 | 
			
		||||
 
 | 
			
		||||
@@ -1,62 +0,0 @@
 | 
			
		||||
version: '2'
 | 
			
		||||
plugins:
 | 
			
		||||
  rubocop:
 | 
			
		||||
    enabled: false
 | 
			
		||||
    channel: rubocop-0-73
 | 
			
		||||
  eslint:
 | 
			
		||||
    enabled: false
 | 
			
		||||
  csslint:
 | 
			
		||||
    enabled: true
 | 
			
		||||
  scss-lint:
 | 
			
		||||
    enabled: true
 | 
			
		||||
  brakeman:
 | 
			
		||||
    enabled: false
 | 
			
		||||
checks:
 | 
			
		||||
  similar-code:
 | 
			
		||||
    enabled: false
 | 
			
		||||
  method-count:
 | 
			
		||||
    enabled: true
 | 
			
		||||
    config:
 | 
			
		||||
      threshold: 32
 | 
			
		||||
  file-lines:
 | 
			
		||||
    enabled: true
 | 
			
		||||
    config:
 | 
			
		||||
      threshold: 300
 | 
			
		||||
  method-lines:
 | 
			
		||||
    config:
 | 
			
		||||
      threshold: 50
 | 
			
		||||
exclude_patterns:
 | 
			
		||||
  - 'spec/'
 | 
			
		||||
  - '**/specs/**/**'
 | 
			
		||||
  - '**/spec/**/**'
 | 
			
		||||
  - 'db/*'
 | 
			
		||||
  - 'bin/**/*'
 | 
			
		||||
  - 'db/**/*'
 | 
			
		||||
  - 'config/**/*'
 | 
			
		||||
  - 'public/**/*'
 | 
			
		||||
  - 'vendor/**/*'
 | 
			
		||||
  - 'node_modules/**/*'
 | 
			
		||||
  - 'lib/tasks/auto_annotate_models.rake'
 | 
			
		||||
  - 'app/test-matchers.js'
 | 
			
		||||
  - 'docs/*'
 | 
			
		||||
  - '**/*.md'
 | 
			
		||||
  - '**/*.yml'
 | 
			
		||||
  - 'app/javascript/dashboard/i18n/locale'
 | 
			
		||||
  - '**/*.stories.js'
 | 
			
		||||
  - 'stories/'
 | 
			
		||||
  - 'app/javascript/dashboard/components/widgets/conversation/advancedFilterItems/index.js'
 | 
			
		||||
  - 'app/javascript/shared/constants/countries.js'
 | 
			
		||||
  - 'app/javascript/dashboard/components/widgets/conversation/advancedFilterItems/languages.js'
 | 
			
		||||
  - 'app/javascript/dashboard/routes/dashboard/contacts/contactFilterItems/index.js'
 | 
			
		||||
  - 'app/javascript/dashboard/routes/dashboard/settings/automation/constants.js'
 | 
			
		||||
  - 'app/javascript/dashboard/components/widgets/FilterInput/FilterOperatorTypes.js'
 | 
			
		||||
  - 'app/javascript/dashboard/routes/dashboard/settings/reports/constants.js'
 | 
			
		||||
  - 'app/javascript/dashboard/store/captain/storeFactory.js'
 | 
			
		||||
  - 'app/javascript/dashboard/i18n/index.js'
 | 
			
		||||
  - 'app/javascript/widget/i18n/index.js'
 | 
			
		||||
  - 'app/javascript/survey/i18n/index.js'
 | 
			
		||||
  - 'app/javascript/shared/constants/locales.js'
 | 
			
		||||
  - 'app/javascript/dashboard/helper/specs/macrosFixtures.js'
 | 
			
		||||
  - 'app/javascript/dashboard/routes/dashboard/settings/macros/constants.js'
 | 
			
		||||
  - '**/fixtures/**'
 | 
			
		||||
  - '**/*/fixtures.js'
 | 
			
		||||
@@ -77,8 +77,8 @@ RUN curl -L https://github.com/DarthSim/overmind/releases/download/v2.1.0/overmi
 | 
			
		||||
WORKDIR /workspace
 | 
			
		||||
RUN chown vscode:vscode /workspace
 | 
			
		||||
 | 
			
		||||
# set up node js and pnpm in single layer
 | 
			
		||||
RUN npm install -g pnpm@${PNPM_VERSION} \
 | 
			
		||||
# set up node js, pnpm and claude code in single layer
 | 
			
		||||
RUN npm install -g pnpm@${PNPM_VERSION} @anthropic-ai/claude-code \
 | 
			
		||||
    && npm cache clean --force
 | 
			
		||||
 | 
			
		||||
# Switch to vscode user
 | 
			
		||||
 
 | 
			
		||||
@@ -4,17 +4,26 @@
 | 
			
		||||
  "dockerComposeFile": "docker-compose.yml",
 | 
			
		||||
 | 
			
		||||
  "settings": {
 | 
			
		||||
    "terminal.integrated.shell.linux": "/bin/zsh"
 | 
			
		||||
    "terminal.integrated.shell.linux": "/bin/zsh",
 | 
			
		||||
    "extensions.showRecommendationsOnlyOnDemand": true,
 | 
			
		||||
    "editor.formatOnSave": true,
 | 
			
		||||
    "files.trimTrailingWhitespace": true,
 | 
			
		||||
    "files.insertFinalNewline": true,
 | 
			
		||||
    "search.exclude": {
 | 
			
		||||
      "**/node_modules": true,
 | 
			
		||||
      "**/tmp": true,
 | 
			
		||||
      "**/log": true,
 | 
			
		||||
      "**/coverage": true,
 | 
			
		||||
      "**/public/packs": true
 | 
			
		||||
    }
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  // Add the IDs of extensions you want installed when the container is created.
 | 
			
		||||
  "extensions": [
 | 
			
		||||
    "rebornix.Ruby",
 | 
			
		||||
    "Shopify.ruby-lsp",
 | 
			
		||||
    "misogi.ruby-rubocop",
 | 
			
		||||
    "wingrunr21.vscode-ruby",
 | 
			
		||||
    "davidpallinder.rails-test-runner",
 | 
			
		||||
    "eamodio.gitlens",
 | 
			
		||||
    "github.copilot",
 | 
			
		||||
    "mrmlnc.vscode-duplicate"
 | 
			
		||||
  ],
 | 
			
		||||
 
 | 
			
		||||
@@ -4,5 +4,13 @@ sed -i -e '/POSTGRES_HOST/ s/=.*/=localhost/' .env
 | 
			
		||||
sed -i -e '/SMTP_ADDRESS/ s/=.*/=localhost/' .env
 | 
			
		||||
sed -i -e "/FRONTEND_URL/ s/=.*/=https:\/\/$CODESPACE_NAME-3000.app.github.dev/" .env
 | 
			
		||||
 | 
			
		||||
# Setup Claude Code API key if available
 | 
			
		||||
if [ -n "$CLAUDE_CODE_API_KEY" ]; then
 | 
			
		||||
  mkdir -p ~/.claude
 | 
			
		||||
  echo '{"apiKeyHelper": "~/.claude/anthropic_key.sh"}' > ~/.claude/settings.json
 | 
			
		||||
  echo "echo \"$CLAUDE_CODE_API_KEY\"" > ~/.claude/anthropic_key.sh
 | 
			
		||||
  chmod +x ~/.claude/anthropic_key.sh
 | 
			
		||||
fi
 | 
			
		||||
 | 
			
		||||
# codespaces make the ports public
 | 
			
		||||
gh codespace ports visibility 3000:public 3036:public 8025:public -c $CODESPACE_NAME
 | 
			
		||||
 
 | 
			
		||||
@@ -103,6 +103,7 @@ module.exports = {
 | 
			
		||||
          '⌘',
 | 
			
		||||
          '📄',
 | 
			
		||||
          '🎉',
 | 
			
		||||
          '🚀',
 | 
			
		||||
          '💬',
 | 
			
		||||
          '👥',
 | 
			
		||||
          '📥',
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										28
									
								
								.github/workflows/auto-assign-pr.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								.github/workflows/auto-assign-pr.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,28 @@
 | 
			
		||||
name: Auto-assign PR to Author
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  pull_request:
 | 
			
		||||
    types: [opened]
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  auto-assign:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    permissions:
 | 
			
		||||
      pull-requests: write
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Auto-assign PR to author
 | 
			
		||||
        uses: actions/github-script@v7
 | 
			
		||||
        with:
 | 
			
		||||
          script: |
 | 
			
		||||
            const { owner, repo } = context.repo;
 | 
			
		||||
            const pull_number = context.payload.pull_request.number;
 | 
			
		||||
            const author = context.payload.pull_request.user.login;
 | 
			
		||||
            
 | 
			
		||||
            await github.rest.issues.addAssignees({
 | 
			
		||||
              owner,
 | 
			
		||||
              repo,
 | 
			
		||||
              issue_number: pull_number,
 | 
			
		||||
              assignees: [author]
 | 
			
		||||
            });
 | 
			
		||||
            
 | 
			
		||||
            console.log(`Assigned PR #${pull_number} to ${author}`);
 | 
			
		||||
							
								
								
									
										5
									
								
								.github/workflows/deploy_check.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/deploy_check.yml
									
									
									
									
										vendored
									
									
								
							@@ -6,6 +6,11 @@ name: Deploy Check
 | 
			
		||||
on:
 | 
			
		||||
  pull_request:
 | 
			
		||||
 | 
			
		||||
# If two pushes happen within a short time in the same PR, cancel the run of the oldest push
 | 
			
		||||
concurrency:
 | 
			
		||||
  group: pr-${{ github.workflow }}-${{ github.head_ref }}
 | 
			
		||||
  cancel-in-progress: true
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  deployment_check:
 | 
			
		||||
    name: Check Deployment
 | 
			
		||||
 
 | 
			
		||||
@@ -5,6 +5,11 @@ on:
 | 
			
		||||
    branches:
 | 
			
		||||
      - develop
 | 
			
		||||
 | 
			
		||||
# If two pushes happen within a short time in the same PR, cancel the run of the oldest push
 | 
			
		||||
concurrency:
 | 
			
		||||
  group: pr-${{ github.workflow }}-${{ github.head_ref }}
 | 
			
		||||
  cancel-in-progress: true
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  log_lines_check:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										5
									
								
								.github/workflows/size-limit.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/size-limit.yml
									
									
									
									
										vendored
									
									
								
							@@ -5,6 +5,11 @@ on:
 | 
			
		||||
    branches:
 | 
			
		||||
      - develop
 | 
			
		||||
 | 
			
		||||
# If two pushes happen within a short time in the same PR, cancel the run of the oldest push
 | 
			
		||||
concurrency:
 | 
			
		||||
  group: pr-${{ github.workflow }}-${{ github.head_ref }}
 | 
			
		||||
  cancel-in-progress: true
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  test:
 | 
			
		||||
    runs-on: ubuntu-22.04
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -94,3 +94,8 @@ yarn-debug.log*
 | 
			
		||||
.vscode
 | 
			
		||||
.claude/settings.local.json
 | 
			
		||||
.cursor
 | 
			
		||||
CLAUDE.local.md
 | 
			
		||||
 | 
			
		||||
# Histoire deployment
 | 
			
		||||
.netlify
 | 
			
		||||
.histoire
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										7
									
								
								.qlty/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								.qlty/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,7 @@
 | 
			
		||||
*
 | 
			
		||||
!configs
 | 
			
		||||
!configs/**
 | 
			
		||||
!hooks
 | 
			
		||||
!hooks/**
 | 
			
		||||
!qlty.toml
 | 
			
		||||
!.gitignore
 | 
			
		||||
							
								
								
									
										2
									
								
								.qlty/configs/.hadolint.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								.qlty/configs/.hadolint.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,2 @@
 | 
			
		||||
ignored:
 | 
			
		||||
  - DL3008
 | 
			
		||||
							
								
								
									
										1
									
								
								.qlty/configs/.shellcheckrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.qlty/configs/.shellcheckrc
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1 @@
 | 
			
		||||
source-path=SCRIPTDIR
 | 
			
		||||
							
								
								
									
										8
									
								
								.qlty/configs/.yamllint.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.qlty/configs/.yamllint.yaml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,8 @@
 | 
			
		||||
rules:
 | 
			
		||||
  document-start: disable
 | 
			
		||||
  quoted-strings:
 | 
			
		||||
    required: only-when-needed
 | 
			
		||||
    extra-allowed: ["{|}"]
 | 
			
		||||
  key-duplicates: {}
 | 
			
		||||
  octal-values:
 | 
			
		||||
    forbid-implicit-octal: true
 | 
			
		||||
							
								
								
									
										84
									
								
								.qlty/qlty.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								.qlty/qlty.toml
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,84 @@
 | 
			
		||||
# This file was automatically generated by `qlty init`.
 | 
			
		||||
# You can modify it to suit your needs.
 | 
			
		||||
# We recommend you to commit this file to your repository.
 | 
			
		||||
#
 | 
			
		||||
# This configuration is used by both Qlty CLI and Qlty Cloud.
 | 
			
		||||
#
 | 
			
		||||
#     Qlty CLI -- Code quality toolkit for developers
 | 
			
		||||
#     Qlty Cloud -- Fully automated Code Health Platform
 | 
			
		||||
#
 | 
			
		||||
# Try Qlty Cloud: https://qlty.sh
 | 
			
		||||
#
 | 
			
		||||
# For a guide to configuration, visit https://qlty.sh/d/config
 | 
			
		||||
# Or for a full reference, visit https://qlty.sh/d/qlty-toml
 | 
			
		||||
config_version = "0"
 | 
			
		||||
 | 
			
		||||
exclude_patterns = [
 | 
			
		||||
  "*_min.*",
 | 
			
		||||
  "*-min.*",
 | 
			
		||||
  "*.min.*",
 | 
			
		||||
  "**/.yarn/**",
 | 
			
		||||
  "**/*.d.ts",
 | 
			
		||||
  "**/assets/**",
 | 
			
		||||
  "**/bower_components/**",
 | 
			
		||||
  "**/build/**",
 | 
			
		||||
  "**/cache/**",
 | 
			
		||||
  "**/config/**",
 | 
			
		||||
  "**/db/**",
 | 
			
		||||
  "**/deps/**",
 | 
			
		||||
  "**/dist/**",
 | 
			
		||||
  "**/extern/**",
 | 
			
		||||
  "**/external/**",
 | 
			
		||||
  "**/generated/**",
 | 
			
		||||
  "**/Godeps/**",
 | 
			
		||||
  "**/gradlew/**",
 | 
			
		||||
  "**/mvnw/**",
 | 
			
		||||
  "**/node_modules/**",
 | 
			
		||||
  "**/protos/**",
 | 
			
		||||
  "**/seed/**",
 | 
			
		||||
  "**/target/**",
 | 
			
		||||
  "**/templates/**",
 | 
			
		||||
  "**/testdata/**",
 | 
			
		||||
  "**/vendor/**", "spec/", "**/specs/**/**", "**/spec/**/**", "db/*", "bin/**/*", "db/**/*", "config/**/*", "public/**/*", "vendor/**/*", "node_modules/**/*", "lib/tasks/auto_annotate_models.rake", "app/test-matchers.js", "docs/*", "**/*.md", "**/*.yml", "app/javascript/dashboard/i18n/locale", "**/*.stories.js", "stories/", "app/javascript/dashboard/components/widgets/conversation/advancedFilterItems/index.js", "app/javascript/shared/constants/countries.js", "app/javascript/dashboard/components/widgets/conversation/advancedFilterItems/languages.js", "app/javascript/dashboard/routes/dashboard/contacts/contactFilterItems/index.js", "app/javascript/dashboard/routes/dashboard/settings/automation/constants.js", "app/javascript/dashboard/components/widgets/FilterInput/FilterOperatorTypes.js", "app/javascript/dashboard/routes/dashboard/settings/reports/constants.js", "app/javascript/dashboard/store/captain/storeFactory.js", "app/javascript/dashboard/i18n/index.js", "app/javascript/widget/i18n/index.js", "app/javascript/survey/i18n/index.js", "app/javascript/shared/constants/locales.js", "app/javascript/dashboard/helper/specs/macrosFixtures.js", "app/javascript/dashboard/routes/dashboard/settings/macros/constants.js", "**/fixtures/**", "**/*/fixtures.js",
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
test_patterns = [
 | 
			
		||||
  "**/test/**",
 | 
			
		||||
  "**/spec/**",
 | 
			
		||||
  "**/*.test.*",
 | 
			
		||||
  "**/*.spec.*",
 | 
			
		||||
  "**/*_test.*",
 | 
			
		||||
  "**/*_spec.*",
 | 
			
		||||
  "**/test_*.*",
 | 
			
		||||
  "**/spec_*.*",
 | 
			
		||||
]
 | 
			
		||||
 | 
			
		||||
[smells]
 | 
			
		||||
mode = "comment"
 | 
			
		||||
 | 
			
		||||
[smells.boolean_logic]
 | 
			
		||||
threshold = 4
 | 
			
		||||
 | 
			
		||||
[smells.file_complexity]
 | 
			
		||||
threshold = 66
 | 
			
		||||
enabled = true
 | 
			
		||||
 | 
			
		||||
[smells.return_statements]
 | 
			
		||||
threshold = 4
 | 
			
		||||
 | 
			
		||||
[smells.nested_control_flow]
 | 
			
		||||
threshold = 4
 | 
			
		||||
 | 
			
		||||
[smells.function_parameters]
 | 
			
		||||
threshold = 4
 | 
			
		||||
 | 
			
		||||
[smells.function_complexity]
 | 
			
		||||
threshold = 5
 | 
			
		||||
 | 
			
		||||
[smells.duplication]
 | 
			
		||||
enabled = true
 | 
			
		||||
threshold = 20
 | 
			
		||||
 | 
			
		||||
[[source]]
 | 
			
		||||
name = "default"
 | 
			
		||||
default = true
 | 
			
		||||
@@ -283,7 +283,7 @@ Rails/RedundantActiveRecordAllMethod:
 | 
			
		||||
  Enabled: false
 | 
			
		||||
 | 
			
		||||
Layout/TrailingEmptyLines:
 | 
			
		||||
  Enabled: false
 | 
			
		||||
  Enabled: true
 | 
			
		||||
 | 
			
		||||
Style/SafeNavigationChainLength:
 | 
			
		||||
  Enabled: false
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										17
									
								
								AGENTS.md
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								AGENTS.md
									
									
									
									
									
								
							@@ -56,3 +56,20 @@
 | 
			
		||||
## Ruby Best Practices
 | 
			
		||||
 | 
			
		||||
- Use compact `module/class` definitions; avoid nested styles
 | 
			
		||||
 | 
			
		||||
## Enterprise Edition Notes
 | 
			
		||||
 | 
			
		||||
- Chatwoot has an Enterprise overlay under `enterprise/` that extends/overrides OSS code.
 | 
			
		||||
- When you add or modify core functionality, always check for corresponding files in `enterprise/` and keep behavior compatible.
 | 
			
		||||
- Follow the Enterprise development practices documented here:
 | 
			
		||||
  - https://chatwoot.help/hc/handbook/articles/developing-enterprise-edition-features-38
 | 
			
		||||
 | 
			
		||||
Practical checklist for any change impacting core logic or public APIs
 | 
			
		||||
- Search for related files in both trees before editing (e.g., `rg -n "FooService|ControllerName|ModelName" app enterprise`).
 | 
			
		||||
- If adding new endpoints, services, or models, consider whether Enterprise needs:
 | 
			
		||||
  - An override (e.g., `enterprise/app/...`), or
 | 
			
		||||
  - An extension point (e.g., `prepend_mod_with`, hooks, configuration) to avoid hard forks.
 | 
			
		||||
- Avoid hardcoding instance- or plan-specific behavior in OSS; prefer configuration, feature flags, or extension points consumed by Enterprise.
 | 
			
		||||
- Keep request/response contracts stable across OSS and Enterprise; update both sets of routes/controllers when introducing new APIs.
 | 
			
		||||
- When renaming/moving shared code, mirror the change in `enterprise/` to prevent drift.
 | 
			
		||||
- Tests: Add Enterprise-specific specs under `spec/enterprise`, mirroring OSS spec layout where applicable.
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										21
									
								
								Gemfile
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								Gemfile
									
									
									
									
									
								
							@@ -62,6 +62,10 @@ gem 'redis-namespace'
 | 
			
		||||
# super fast record imports in bulk
 | 
			
		||||
gem 'activerecord-import'
 | 
			
		||||
 | 
			
		||||
gem 'searchkick'
 | 
			
		||||
gem 'opensearch-ruby'
 | 
			
		||||
gem 'faraday_middleware-aws-sigv4'
 | 
			
		||||
 | 
			
		||||
##--- gems for server & infra configuration ---##
 | 
			
		||||
gem 'dotenv-rails', '>= 3.0.0'
 | 
			
		||||
gem 'foreman'
 | 
			
		||||
@@ -77,6 +81,7 @@ gem 'devise_token_auth', '>= 1.2.3'
 | 
			
		||||
# authorization
 | 
			
		||||
gem 'jwt'
 | 
			
		||||
gem 'pundit'
 | 
			
		||||
 | 
			
		||||
# super admin
 | 
			
		||||
gem 'administrate', '>= 0.20.1'
 | 
			
		||||
gem 'administrate-field-active_storage', '>= 1.0.3'
 | 
			
		||||
@@ -89,7 +94,7 @@ gem 'wisper', '2.0.0'
 | 
			
		||||
##--- gems for channels ---##
 | 
			
		||||
gem 'facebook-messenger'
 | 
			
		||||
gem 'line-bot-api'
 | 
			
		||||
gem 'twilio-ruby', '~> 5.66'
 | 
			
		||||
gem 'twilio-ruby'
 | 
			
		||||
# twitty will handle subscription of twitter account events
 | 
			
		||||
# gem 'twitty', git: 'https://github.com/chatwoot/twitty'
 | 
			
		||||
gem 'twitty', '~> 0.1.5'
 | 
			
		||||
@@ -108,7 +113,7 @@ gem 'google-cloud-translate-v3', '>= 0.7.0'
 | 
			
		||||
##-- apm and error monitoring ---#
 | 
			
		||||
# loaded only when environment variables are set.
 | 
			
		||||
# ref application.rb
 | 
			
		||||
gem 'ddtrace', require: false
 | 
			
		||||
gem 'datadog', '~> 2.0', require: false
 | 
			
		||||
gem 'elastic-apm', require: false
 | 
			
		||||
gem 'newrelic_rpm', require: false
 | 
			
		||||
gem 'newrelic-sidekiq-metrics', '>= 1.6.2', require: false
 | 
			
		||||
@@ -121,6 +126,8 @@ gem 'sentry-sidekiq', '>= 5.19.0', require: false
 | 
			
		||||
gem 'sidekiq', '>= 7.3.1'
 | 
			
		||||
# We want cron jobs
 | 
			
		||||
gem 'sidekiq-cron', '>= 1.12.0'
 | 
			
		||||
# for sidekiq healthcheck
 | 
			
		||||
gem 'sidekiq_alive'
 | 
			
		||||
 | 
			
		||||
##-- Push notification service --##
 | 
			
		||||
gem 'fcm'
 | 
			
		||||
@@ -165,6 +172,7 @@ gem 'audited', '~> 5.4', '>= 5.4.1'
 | 
			
		||||
 | 
			
		||||
# need for google auth
 | 
			
		||||
gem 'omniauth', '>= 2.1.2'
 | 
			
		||||
gem 'omniauth-saml'
 | 
			
		||||
gem 'omniauth-google-oauth2', '>= 1.1.3'
 | 
			
		||||
gem 'omniauth-rails_csrf_protection', '~> 1.0', '>= 1.0.2'
 | 
			
		||||
 | 
			
		||||
@@ -177,6 +185,10 @@ gem 'reverse_markdown'
 | 
			
		||||
 | 
			
		||||
gem 'iso-639'
 | 
			
		||||
gem 'ruby-openai'
 | 
			
		||||
gem 'ai-agents', '>= 0.4.3'
 | 
			
		||||
 | 
			
		||||
# TODO: Move this gem as a dependency of ai-agents
 | 
			
		||||
gem 'ruby_llm-schema'
 | 
			
		||||
 | 
			
		||||
gem 'shopify_api'
 | 
			
		||||
 | 
			
		||||
@@ -206,6 +218,8 @@ group :development do
 | 
			
		||||
  gem 'stackprof'
 | 
			
		||||
  # Should install the associated chrome extension to view query logs
 | 
			
		||||
  gem 'meta_request', '>= 0.8.3'
 | 
			
		||||
 | 
			
		||||
  gem 'tidewave'
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
group :test do
 | 
			
		||||
@@ -215,6 +229,7 @@ group :test do
 | 
			
		||||
  gem 'webmock'
 | 
			
		||||
  # test profiling
 | 
			
		||||
  gem 'test-prof'
 | 
			
		||||
  gem 'simplecov_json_formatter', require: false
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
group :development, :test do
 | 
			
		||||
@@ -239,7 +254,7 @@ group :development, :test do
 | 
			
		||||
  gem 'rubocop-factory_bot', require: false
 | 
			
		||||
  gem 'seed_dump'
 | 
			
		||||
  gem 'shoulda-matchers'
 | 
			
		||||
  gem 'simplecov', '0.17.1', require: false
 | 
			
		||||
  gem 'simplecov', '>= 0.21', require: false
 | 
			
		||||
  gem 'spring'
 | 
			
		||||
  gem 'spring-watcher-listen'
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										292
									
								
								Gemfile.lock
									
									
									
									
									
								
							
							
						
						
									
										292
									
								
								Gemfile.lock
									
									
									
									
									
								
							@@ -25,35 +25,35 @@ GIT
 | 
			
		||||
GEM
 | 
			
		||||
  remote: https://rubygems.org/
 | 
			
		||||
  specs:
 | 
			
		||||
    actioncable (7.1.5.1)
 | 
			
		||||
      actionpack (= 7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    actioncable (7.1.5.2)
 | 
			
		||||
      actionpack (= 7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      nio4r (~> 2.0)
 | 
			
		||||
      websocket-driver (>= 0.6.1)
 | 
			
		||||
      zeitwerk (~> 2.6)
 | 
			
		||||
    actionmailbox (7.1.5.1)
 | 
			
		||||
      actionpack (= 7.1.5.1)
 | 
			
		||||
      activejob (= 7.1.5.1)
 | 
			
		||||
      activerecord (= 7.1.5.1)
 | 
			
		||||
      activestorage (= 7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    actionmailbox (7.1.5.2)
 | 
			
		||||
      actionpack (= 7.1.5.2)
 | 
			
		||||
      activejob (= 7.1.5.2)
 | 
			
		||||
      activerecord (= 7.1.5.2)
 | 
			
		||||
      activestorage (= 7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      mail (>= 2.7.1)
 | 
			
		||||
      net-imap
 | 
			
		||||
      net-pop
 | 
			
		||||
      net-smtp
 | 
			
		||||
    actionmailer (7.1.5.1)
 | 
			
		||||
      actionpack (= 7.1.5.1)
 | 
			
		||||
      actionview (= 7.1.5.1)
 | 
			
		||||
      activejob (= 7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    actionmailer (7.1.5.2)
 | 
			
		||||
      actionpack (= 7.1.5.2)
 | 
			
		||||
      actionview (= 7.1.5.2)
 | 
			
		||||
      activejob (= 7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      mail (~> 2.5, >= 2.5.4)
 | 
			
		||||
      net-imap
 | 
			
		||||
      net-pop
 | 
			
		||||
      net-smtp
 | 
			
		||||
      rails-dom-testing (~> 2.2)
 | 
			
		||||
    actionpack (7.1.5.1)
 | 
			
		||||
      actionview (= 7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    actionpack (7.1.5.2)
 | 
			
		||||
      actionview (= 7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      nokogiri (>= 1.8.5)
 | 
			
		||||
      racc
 | 
			
		||||
      rack (>= 2.2.4)
 | 
			
		||||
@@ -61,38 +61,38 @@ GEM
 | 
			
		||||
      rack-test (>= 0.6.3)
 | 
			
		||||
      rails-dom-testing (~> 2.2)
 | 
			
		||||
      rails-html-sanitizer (~> 1.6)
 | 
			
		||||
    actiontext (7.1.5.1)
 | 
			
		||||
      actionpack (= 7.1.5.1)
 | 
			
		||||
      activerecord (= 7.1.5.1)
 | 
			
		||||
      activestorage (= 7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    actiontext (7.1.5.2)
 | 
			
		||||
      actionpack (= 7.1.5.2)
 | 
			
		||||
      activerecord (= 7.1.5.2)
 | 
			
		||||
      activestorage (= 7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      globalid (>= 0.6.0)
 | 
			
		||||
      nokogiri (>= 1.8.5)
 | 
			
		||||
    actionview (7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    actionview (7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      builder (~> 3.1)
 | 
			
		||||
      erubi (~> 1.11)
 | 
			
		||||
      rails-dom-testing (~> 2.2)
 | 
			
		||||
      rails-html-sanitizer (~> 1.6)
 | 
			
		||||
    active_record_query_trace (1.8)
 | 
			
		||||
    activejob (7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    activejob (7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      globalid (>= 0.3.6)
 | 
			
		||||
    activemodel (7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    activerecord (7.1.5.1)
 | 
			
		||||
      activemodel (= 7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    activemodel (7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
    activerecord (7.1.5.2)
 | 
			
		||||
      activemodel (= 7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      timeout (>= 0.4.0)
 | 
			
		||||
    activerecord-import (2.1.0)
 | 
			
		||||
      activerecord (>= 4.2)
 | 
			
		||||
    activestorage (7.1.5.1)
 | 
			
		||||
      actionpack (= 7.1.5.1)
 | 
			
		||||
      activejob (= 7.1.5.1)
 | 
			
		||||
      activerecord (= 7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    activestorage (7.1.5.2)
 | 
			
		||||
      actionpack (= 7.1.5.2)
 | 
			
		||||
      activejob (= 7.1.5.2)
 | 
			
		||||
      activerecord (= 7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      marcel (~> 1.0)
 | 
			
		||||
    activesupport (7.1.5.1)
 | 
			
		||||
    activesupport (7.1.5.2)
 | 
			
		||||
      base64
 | 
			
		||||
      benchmark (>= 0.3)
 | 
			
		||||
      bigdecimal
 | 
			
		||||
@@ -126,6 +126,8 @@ GEM
 | 
			
		||||
      jbuilder (~> 2)
 | 
			
		||||
      rails (>= 4.2, < 7.2)
 | 
			
		||||
      selectize-rails (~> 0.6)
 | 
			
		||||
    ai-agents (0.4.3)
 | 
			
		||||
      ruby_llm (~> 1.3)
 | 
			
		||||
    annotate (3.2.0)
 | 
			
		||||
      activerecord (>= 3.2, < 8.0)
 | 
			
		||||
      rake (>= 10.4, < 14.0)
 | 
			
		||||
@@ -153,10 +155,10 @@ GEM
 | 
			
		||||
    barnes (0.0.9)
 | 
			
		||||
      multi_json (~> 1)
 | 
			
		||||
      statsd-ruby (~> 1.1)
 | 
			
		||||
    base64 (0.2.0)
 | 
			
		||||
    base64 (0.3.0)
 | 
			
		||||
    bcrypt (3.1.20)
 | 
			
		||||
    benchmark (0.4.0)
 | 
			
		||||
    bigdecimal (3.1.9)
 | 
			
		||||
    benchmark (0.4.1)
 | 
			
		||||
    bigdecimal (3.2.2)
 | 
			
		||||
    bindex (0.8.1)
 | 
			
		||||
    bootsnap (1.16.0)
 | 
			
		||||
      msgpack (~> 1.2)
 | 
			
		||||
@@ -172,6 +174,8 @@ GEM
 | 
			
		||||
      bundler (>= 1.2.0, < 3)
 | 
			
		||||
      thor (~> 1.0)
 | 
			
		||||
    byebug (11.1.3)
 | 
			
		||||
    childprocess (5.1.0)
 | 
			
		||||
      logger (~> 1.5)
 | 
			
		||||
    climate_control (1.2.0)
 | 
			
		||||
    coderay (1.1.3)
 | 
			
		||||
    commonmarker (0.23.10)
 | 
			
		||||
@@ -190,10 +194,14 @@ GEM
 | 
			
		||||
      activerecord (>= 5.a)
 | 
			
		||||
      database_cleaner-core (~> 2.0.0)
 | 
			
		||||
    database_cleaner-core (2.0.1)
 | 
			
		||||
    date (3.4.1)
 | 
			
		||||
    ddtrace (0.48.0)
 | 
			
		||||
      ffi (~> 1.0)
 | 
			
		||||
    datadog (2.19.0)
 | 
			
		||||
      datadog-ruby_core_source (~> 3.4, >= 3.4.1)
 | 
			
		||||
      libdatadog (~> 18.1.0.1.0)
 | 
			
		||||
      libddwaf (~> 1.24.1.0.3)
 | 
			
		||||
      logger
 | 
			
		||||
      msgpack
 | 
			
		||||
    datadog-ruby_core_source (3.4.1)
 | 
			
		||||
    date (3.4.1)
 | 
			
		||||
    debug (1.8.0)
 | 
			
		||||
      irb (>= 1.5.0)
 | 
			
		||||
      reline (>= 0.3.1)
 | 
			
		||||
@@ -211,7 +219,7 @@ GEM
 | 
			
		||||
    diff-lcs (1.5.1)
 | 
			
		||||
    digest-crc (0.6.5)
 | 
			
		||||
      rake (>= 12.0.0, < 14.0.0)
 | 
			
		||||
    docile (1.4.0)
 | 
			
		||||
    docile (1.4.1)
 | 
			
		||||
    domain_name (0.5.20190701)
 | 
			
		||||
      unf (>= 0.0.5, < 1.0.0)
 | 
			
		||||
    dotenv (3.1.2)
 | 
			
		||||
@@ -222,6 +230,35 @@ GEM
 | 
			
		||||
      addressable (~> 2.8)
 | 
			
		||||
    drb (2.2.3)
 | 
			
		||||
    dry-cli (1.1.0)
 | 
			
		||||
    dry-configurable (1.3.0)
 | 
			
		||||
      dry-core (~> 1.1)
 | 
			
		||||
      zeitwerk (~> 2.6)
 | 
			
		||||
    dry-core (1.1.0)
 | 
			
		||||
      concurrent-ruby (~> 1.0)
 | 
			
		||||
      logger
 | 
			
		||||
      zeitwerk (~> 2.6)
 | 
			
		||||
    dry-inflector (1.2.0)
 | 
			
		||||
    dry-initializer (3.2.0)
 | 
			
		||||
    dry-logic (1.6.0)
 | 
			
		||||
      bigdecimal
 | 
			
		||||
      concurrent-ruby (~> 1.0)
 | 
			
		||||
      dry-core (~> 1.1)
 | 
			
		||||
      zeitwerk (~> 2.6)
 | 
			
		||||
    dry-schema (1.14.1)
 | 
			
		||||
      concurrent-ruby (~> 1.0)
 | 
			
		||||
      dry-configurable (~> 1.0, >= 1.0.1)
 | 
			
		||||
      dry-core (~> 1.1)
 | 
			
		||||
      dry-initializer (~> 3.2)
 | 
			
		||||
      dry-logic (~> 1.5)
 | 
			
		||||
      dry-types (~> 1.8)
 | 
			
		||||
      zeitwerk (~> 2.6)
 | 
			
		||||
    dry-types (1.8.3)
 | 
			
		||||
      bigdecimal (~> 3.0)
 | 
			
		||||
      concurrent-ruby (~> 1.0)
 | 
			
		||||
      dry-core (~> 1.0)
 | 
			
		||||
      dry-inflector (~> 1.0)
 | 
			
		||||
      dry-logic (~> 1.4)
 | 
			
		||||
      zeitwerk (~> 2.6)
 | 
			
		||||
    ecma-re-validator (0.4.0)
 | 
			
		||||
      regexp_parser (~> 2.2)
 | 
			
		||||
    elastic-apm (4.6.2)
 | 
			
		||||
@@ -244,8 +281,10 @@ GEM
 | 
			
		||||
      railties (>= 5.0.0)
 | 
			
		||||
    faker (3.2.0)
 | 
			
		||||
      i18n (>= 1.8.11, < 2)
 | 
			
		||||
    faraday (2.9.0)
 | 
			
		||||
      faraday-net_http (>= 2.0, < 3.2)
 | 
			
		||||
    faraday (2.13.1)
 | 
			
		||||
      faraday-net_http (>= 2.0, < 3.5)
 | 
			
		||||
      json
 | 
			
		||||
      logger
 | 
			
		||||
    faraday-follow_redirects (0.3.0)
 | 
			
		||||
      faraday (>= 1, < 3)
 | 
			
		||||
    faraday-mashify (0.1.1)
 | 
			
		||||
@@ -253,13 +292,23 @@ GEM
 | 
			
		||||
      hashie
 | 
			
		||||
    faraday-multipart (1.0.4)
 | 
			
		||||
      multipart-post (~> 2)
 | 
			
		||||
    faraday-net_http (3.1.0)
 | 
			
		||||
      net-http
 | 
			
		||||
    faraday-net_http (3.4.0)
 | 
			
		||||
      net-http (>= 0.5.0)
 | 
			
		||||
    faraday-net_http_persistent (2.1.0)
 | 
			
		||||
      faraday (~> 2.5)
 | 
			
		||||
      net-http-persistent (~> 4.0)
 | 
			
		||||
    faraday-retry (2.2.1)
 | 
			
		||||
      faraday (~> 2.0)
 | 
			
		||||
    faraday_middleware-aws-sigv4 (1.0.1)
 | 
			
		||||
      aws-sigv4 (~> 1.0)
 | 
			
		||||
      faraday (>= 2.0, < 3)
 | 
			
		||||
    fast-mcp (1.5.0)
 | 
			
		||||
      addressable (~> 2.8)
 | 
			
		||||
      base64
 | 
			
		||||
      dry-schema (~> 1.14)
 | 
			
		||||
      json (~> 2.0)
 | 
			
		||||
      mime-types (~> 3.4)
 | 
			
		||||
      rack (~> 3.1)
 | 
			
		||||
    fcm (1.0.8)
 | 
			
		||||
      faraday (>= 1.0.0, < 3.0)
 | 
			
		||||
      googleauth (~> 1)
 | 
			
		||||
@@ -355,6 +404,7 @@ GEM
 | 
			
		||||
    grpc (1.72.0-x86_64-linux)
 | 
			
		||||
      google-protobuf (>= 3.25, < 5.0)
 | 
			
		||||
      googleapis-common-protos-types (~> 1.0)
 | 
			
		||||
    gserver (0.0.1)
 | 
			
		||||
    haikunator (1.1.1)
 | 
			
		||||
    hairtrigger (1.0.0)
 | 
			
		||||
      activerecord (>= 6.0, < 8)
 | 
			
		||||
@@ -397,7 +447,7 @@ GEM
 | 
			
		||||
      rails-dom-testing (>= 1, < 3)
 | 
			
		||||
      railties (>= 4.2.0)
 | 
			
		||||
      thor (>= 0.14, < 2.0)
 | 
			
		||||
    json (2.12.0)
 | 
			
		||||
    json (2.13.2)
 | 
			
		||||
    json_refs (0.1.8)
 | 
			
		||||
      hana
 | 
			
		||||
    json_schemer (0.2.24)
 | 
			
		||||
@@ -412,7 +462,7 @@ GEM
 | 
			
		||||
    judoscale-sidekiq (1.8.2)
 | 
			
		||||
      judoscale-ruby (= 1.8.2)
 | 
			
		||||
      sidekiq (>= 5.0)
 | 
			
		||||
    jwt (2.8.1)
 | 
			
		||||
    jwt (2.10.1)
 | 
			
		||||
      base64
 | 
			
		||||
    kaminari (1.2.2)
 | 
			
		||||
      activesupport (>= 4.1.0)
 | 
			
		||||
@@ -433,10 +483,22 @@ GEM
 | 
			
		||||
      json (>= 1.8)
 | 
			
		||||
      rexml
 | 
			
		||||
    language_server-protocol (3.17.0.5)
 | 
			
		||||
    launchy (2.5.2)
 | 
			
		||||
    launchy (3.1.1)
 | 
			
		||||
      addressable (~> 2.8)
 | 
			
		||||
    letter_opener (1.8.1)
 | 
			
		||||
      launchy (>= 2.2, < 3)
 | 
			
		||||
      childprocess (~> 5.0)
 | 
			
		||||
      logger (~> 1.6)
 | 
			
		||||
    letter_opener (1.10.0)
 | 
			
		||||
      launchy (>= 2.2, < 4)
 | 
			
		||||
    libdatadog (18.1.0.1.0)
 | 
			
		||||
    libdatadog (18.1.0.1.0-x86_64-linux)
 | 
			
		||||
    libddwaf (1.24.1.0.3)
 | 
			
		||||
      ffi (~> 1.0)
 | 
			
		||||
    libddwaf (1.24.1.0.3-arm64-darwin)
 | 
			
		||||
      ffi (~> 1.0)
 | 
			
		||||
    libddwaf (1.24.1.0.3-x86_64-darwin)
 | 
			
		||||
      ffi (~> 1.0)
 | 
			
		||||
    libddwaf (1.24.1.0.3-x86_64-linux)
 | 
			
		||||
      ffi (~> 1.0)
 | 
			
		||||
    line-bot-api (1.28.0)
 | 
			
		||||
    lint_roller (1.1.0)
 | 
			
		||||
    liquid (5.4.0)
 | 
			
		||||
@@ -471,7 +533,7 @@ GEM
 | 
			
		||||
    mime-types-data (3.2023.0218.1)
 | 
			
		||||
    mini_magick (4.12.0)
 | 
			
		||||
    mini_mime (1.1.5)
 | 
			
		||||
    mini_portile2 (2.8.8)
 | 
			
		||||
    mini_portile2 (2.8.9)
 | 
			
		||||
    minitest (5.25.5)
 | 
			
		||||
    mock_redis (0.36.0)
 | 
			
		||||
      ruby2_keywords
 | 
			
		||||
@@ -482,7 +544,7 @@ GEM
 | 
			
		||||
    mutex_m (0.3.0)
 | 
			
		||||
    neighbor (0.2.3)
 | 
			
		||||
      activerecord (>= 5.2)
 | 
			
		||||
    net-http (0.4.1)
 | 
			
		||||
    net-http (0.6.0)
 | 
			
		||||
      uri
 | 
			
		||||
    net-http-persistent (4.0.2)
 | 
			
		||||
      connection_pool (~> 2.2)
 | 
			
		||||
@@ -502,14 +564,14 @@ GEM
 | 
			
		||||
    newrelic_rpm (9.6.0)
 | 
			
		||||
      base64
 | 
			
		||||
    nio4r (2.7.3)
 | 
			
		||||
    nokogiri (1.18.8)
 | 
			
		||||
    nokogiri (1.18.9)
 | 
			
		||||
      mini_portile2 (~> 2.8.2)
 | 
			
		||||
      racc (~> 1.4)
 | 
			
		||||
    nokogiri (1.18.8-arm64-darwin)
 | 
			
		||||
    nokogiri (1.18.9-arm64-darwin)
 | 
			
		||||
      racc (~> 1.4)
 | 
			
		||||
    nokogiri (1.18.8-x86_64-darwin)
 | 
			
		||||
    nokogiri (1.18.9-x86_64-darwin)
 | 
			
		||||
      racc (~> 1.4)
 | 
			
		||||
    nokogiri (1.18.8-x86_64-linux-gnu)
 | 
			
		||||
    nokogiri (1.18.9-x86_64-linux-gnu)
 | 
			
		||||
      racc (~> 1.4)
 | 
			
		||||
    oauth (1.1.0)
 | 
			
		||||
      oauth-tty (~> 1.0, >= 1.0.1)
 | 
			
		||||
@@ -527,8 +589,9 @@ GEM
 | 
			
		||||
    oj (3.16.10)
 | 
			
		||||
      bigdecimal (>= 3.0)
 | 
			
		||||
      ostruct (>= 0.2)
 | 
			
		||||
    omniauth (2.1.2)
 | 
			
		||||
    omniauth (2.1.3)
 | 
			
		||||
      hashie (>= 3.4.6)
 | 
			
		||||
      logger
 | 
			
		||||
      rack (>= 2.2.3)
 | 
			
		||||
      rack-protection
 | 
			
		||||
    omniauth-google-oauth2 (1.1.3)
 | 
			
		||||
@@ -542,6 +605,12 @@ GEM
 | 
			
		||||
    omniauth-rails_csrf_protection (1.0.2)
 | 
			
		||||
      actionpack (>= 4.2)
 | 
			
		||||
      omniauth (~> 2.0)
 | 
			
		||||
    omniauth-saml (2.2.4)
 | 
			
		||||
      omniauth (~> 2.1)
 | 
			
		||||
      ruby-saml (~> 1.18)
 | 
			
		||||
    opensearch-ruby (3.4.0)
 | 
			
		||||
      faraday (>= 1.0, < 3)
 | 
			
		||||
      multi_json (>= 1.0)
 | 
			
		||||
    openssl (3.2.0)
 | 
			
		||||
    orm_adapter (0.5.0)
 | 
			
		||||
    os (1.1.4)
 | 
			
		||||
@@ -563,14 +632,14 @@ GEM
 | 
			
		||||
      method_source (~> 1.0)
 | 
			
		||||
    pry-rails (0.3.9)
 | 
			
		||||
      pry (>= 0.10.4)
 | 
			
		||||
    public_suffix (6.0.0)
 | 
			
		||||
    public_suffix (6.0.2)
 | 
			
		||||
    puma (6.4.3)
 | 
			
		||||
      nio4r (~> 2.0)
 | 
			
		||||
    pundit (2.3.0)
 | 
			
		||||
      activesupport (>= 3.0.0)
 | 
			
		||||
    raabro (1.4.0)
 | 
			
		||||
    racc (1.8.1)
 | 
			
		||||
    rack (2.2.15)
 | 
			
		||||
    rack (3.2.0)
 | 
			
		||||
    rack-attack (6.7.0)
 | 
			
		||||
      rack (>= 1.0, < 4)
 | 
			
		||||
    rack-contrib (2.5.0)
 | 
			
		||||
@@ -579,33 +648,34 @@ GEM
 | 
			
		||||
      rack (>= 2.0.0)
 | 
			
		||||
    rack-mini-profiler (3.2.0)
 | 
			
		||||
      rack (>= 1.2.0)
 | 
			
		||||
    rack-protection (3.2.0)
 | 
			
		||||
    rack-protection (4.1.1)
 | 
			
		||||
      base64 (>= 0.1.0)
 | 
			
		||||
      rack (~> 2.2, >= 2.2.4)
 | 
			
		||||
      logger (>= 1.6.0)
 | 
			
		||||
      rack (>= 3.0.0, < 4)
 | 
			
		||||
    rack-proxy (0.7.7)
 | 
			
		||||
      rack
 | 
			
		||||
    rack-session (1.0.2)
 | 
			
		||||
      rack (< 3)
 | 
			
		||||
    rack-session (2.1.1)
 | 
			
		||||
      base64 (>= 0.1.0)
 | 
			
		||||
      rack (>= 3.0.0)
 | 
			
		||||
    rack-test (2.1.0)
 | 
			
		||||
      rack (>= 1.3)
 | 
			
		||||
    rack-timeout (0.6.3)
 | 
			
		||||
    rackup (1.0.1)
 | 
			
		||||
      rack (< 3)
 | 
			
		||||
      webrick
 | 
			
		||||
    rails (7.1.5.1)
 | 
			
		||||
      actioncable (= 7.1.5.1)
 | 
			
		||||
      actionmailbox (= 7.1.5.1)
 | 
			
		||||
      actionmailer (= 7.1.5.1)
 | 
			
		||||
      actionpack (= 7.1.5.1)
 | 
			
		||||
      actiontext (= 7.1.5.1)
 | 
			
		||||
      actionview (= 7.1.5.1)
 | 
			
		||||
      activejob (= 7.1.5.1)
 | 
			
		||||
      activemodel (= 7.1.5.1)
 | 
			
		||||
      activerecord (= 7.1.5.1)
 | 
			
		||||
      activestorage (= 7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    rackup (2.2.1)
 | 
			
		||||
      rack (>= 3)
 | 
			
		||||
    rails (7.1.5.2)
 | 
			
		||||
      actioncable (= 7.1.5.2)
 | 
			
		||||
      actionmailbox (= 7.1.5.2)
 | 
			
		||||
      actionmailer (= 7.1.5.2)
 | 
			
		||||
      actionpack (= 7.1.5.2)
 | 
			
		||||
      actiontext (= 7.1.5.2)
 | 
			
		||||
      actionview (= 7.1.5.2)
 | 
			
		||||
      activejob (= 7.1.5.2)
 | 
			
		||||
      activemodel (= 7.1.5.2)
 | 
			
		||||
      activerecord (= 7.1.5.2)
 | 
			
		||||
      activestorage (= 7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      bundler (>= 1.15.0)
 | 
			
		||||
      railties (= 7.1.5.1)
 | 
			
		||||
      railties (= 7.1.5.2)
 | 
			
		||||
    rails-dom-testing (2.2.0)
 | 
			
		||||
      activesupport (>= 5.0.0)
 | 
			
		||||
      minitest
 | 
			
		||||
@@ -613,9 +683,9 @@ GEM
 | 
			
		||||
    rails-html-sanitizer (1.6.1)
 | 
			
		||||
      loofah (~> 2.21)
 | 
			
		||||
      nokogiri (>= 1.15.7, != 1.16.7, != 1.16.6, != 1.16.5, != 1.16.4, != 1.16.3, != 1.16.2, != 1.16.1, != 1.16.0.rc1, != 1.16.0)
 | 
			
		||||
    railties (7.1.5.1)
 | 
			
		||||
      actionpack (= 7.1.5.1)
 | 
			
		||||
      activesupport (= 7.1.5.1)
 | 
			
		||||
    railties (7.1.5.2)
 | 
			
		||||
      actionpack (= 7.1.5.2)
 | 
			
		||||
      activesupport (= 7.1.5.2)
 | 
			
		||||
      irb
 | 
			
		||||
      rackup (>= 1.0.0)
 | 
			
		||||
      rake (>= 12.2)
 | 
			
		||||
@@ -707,12 +777,25 @@ GEM
 | 
			
		||||
      faraday (>= 1)
 | 
			
		||||
      faraday-multipart (>= 1)
 | 
			
		||||
    ruby-progressbar (1.13.0)
 | 
			
		||||
    ruby-saml (1.18.1)
 | 
			
		||||
      nokogiri (>= 1.13.10)
 | 
			
		||||
      rexml
 | 
			
		||||
    ruby-vips (2.1.4)
 | 
			
		||||
      ffi (~> 1.12)
 | 
			
		||||
    ruby2_keywords (0.0.5)
 | 
			
		||||
    ruby2ruby (2.5.0)
 | 
			
		||||
      ruby_parser (~> 3.1)
 | 
			
		||||
      sexp_processor (~> 4.6)
 | 
			
		||||
    ruby_llm (1.5.1)
 | 
			
		||||
      base64
 | 
			
		||||
      event_stream_parser (~> 1)
 | 
			
		||||
      faraday (>= 1.10.0)
 | 
			
		||||
      faraday-multipart (>= 1)
 | 
			
		||||
      faraday-net_http (>= 1)
 | 
			
		||||
      faraday-retry (>= 1)
 | 
			
		||||
      marcel (~> 1.0)
 | 
			
		||||
      zeitwerk (~> 2)
 | 
			
		||||
    ruby_llm-schema (0.1.0)
 | 
			
		||||
    ruby_parser (3.20.0)
 | 
			
		||||
      sexp_processor (~> 4.16)
 | 
			
		||||
    sass (3.7.4)
 | 
			
		||||
@@ -732,6 +815,9 @@ GEM
 | 
			
		||||
      parser
 | 
			
		||||
    scss_lint (0.60.0)
 | 
			
		||||
      sass (~> 3.5, >= 3.5.5)
 | 
			
		||||
    searchkick (5.5.2)
 | 
			
		||||
      activemodel (>= 7.1)
 | 
			
		||||
      hashie
 | 
			
		||||
    securerandom (0.4.1)
 | 
			
		||||
    seed_dump (3.3.1)
 | 
			
		||||
      activerecord (>= 4)
 | 
			
		||||
@@ -770,16 +856,20 @@ GEM
 | 
			
		||||
      fugit (~> 1.8)
 | 
			
		||||
      globalid (>= 1.0.1)
 | 
			
		||||
      sidekiq (>= 6)
 | 
			
		||||
    sidekiq_alive (2.5.0)
 | 
			
		||||
      gserver (~> 0.0.1)
 | 
			
		||||
      sidekiq (>= 5, < 9)
 | 
			
		||||
    signet (0.17.0)
 | 
			
		||||
      addressable (~> 2.8)
 | 
			
		||||
      faraday (>= 0.17.5, < 3.a)
 | 
			
		||||
      jwt (>= 1.5, < 3.0)
 | 
			
		||||
      multi_json (~> 1.10)
 | 
			
		||||
    simplecov (0.17.1)
 | 
			
		||||
    simplecov (0.22.0)
 | 
			
		||||
      docile (~> 1.1)
 | 
			
		||||
      json (>= 1.8, < 3)
 | 
			
		||||
      simplecov-html (~> 0.10.0)
 | 
			
		||||
    simplecov-html (0.10.2)
 | 
			
		||||
      simplecov-html (~> 0.11)
 | 
			
		||||
      simplecov_json_formatter (~> 0.1)
 | 
			
		||||
    simplecov-html (0.13.2)
 | 
			
		||||
    simplecov_json_formatter (0.1.4)
 | 
			
		||||
    slack-ruby-client (2.5.2)
 | 
			
		||||
      faraday (>= 2.0)
 | 
			
		||||
      faraday-mashify
 | 
			
		||||
@@ -808,14 +898,18 @@ GEM
 | 
			
		||||
    stripe (8.5.0)
 | 
			
		||||
    telephone_number (1.4.20)
 | 
			
		||||
    test-prof (1.2.1)
 | 
			
		||||
    thor (1.3.1)
 | 
			
		||||
    thor (1.4.0)
 | 
			
		||||
    tidewave (0.2.0)
 | 
			
		||||
      fast-mcp (~> 1.5.0)
 | 
			
		||||
      rack (>= 2.0)
 | 
			
		||||
      rails (>= 7.1.0)
 | 
			
		||||
    tilt (2.3.0)
 | 
			
		||||
    time_diff (0.3.0)
 | 
			
		||||
      activesupport
 | 
			
		||||
      i18n
 | 
			
		||||
    timeout (0.4.3)
 | 
			
		||||
    trailblazer-option (0.1.2)
 | 
			
		||||
    twilio-ruby (5.77.0)
 | 
			
		||||
    twilio-ruby (7.6.0)
 | 
			
		||||
      faraday (>= 0.9, < 3.0)
 | 
			
		||||
      jwt (>= 1.5, < 3.0)
 | 
			
		||||
      nokogiri (>= 1.6, < 2.0)
 | 
			
		||||
@@ -862,7 +956,6 @@ GEM
 | 
			
		||||
      addressable (>= 2.8.0)
 | 
			
		||||
      crack (>= 0.3.2)
 | 
			
		||||
      hashdiff (>= 0.4.0, < 2.0.0)
 | 
			
		||||
    webrick (1.9.1)
 | 
			
		||||
    websocket-driver (0.7.7)
 | 
			
		||||
      base64
 | 
			
		||||
      websocket-extensions (>= 0.1.0)
 | 
			
		||||
@@ -891,6 +984,7 @@ DEPENDENCIES
 | 
			
		||||
  administrate (>= 0.20.1)
 | 
			
		||||
  administrate-field-active_storage (>= 1.0.3)
 | 
			
		||||
  administrate-field-belongs_to_search (>= 0.9.0)
 | 
			
		||||
  ai-agents (>= 0.4.3)
 | 
			
		||||
  annotate
 | 
			
		||||
  attr_extras
 | 
			
		||||
  audited (~> 5.4, >= 5.4.1)
 | 
			
		||||
@@ -907,7 +1001,7 @@ DEPENDENCIES
 | 
			
		||||
  commonmarker
 | 
			
		||||
  csv-safe
 | 
			
		||||
  database_cleaner
 | 
			
		||||
  ddtrace
 | 
			
		||||
  datadog (~> 2.0)
 | 
			
		||||
  debug (~> 1.8)
 | 
			
		||||
  devise (>= 4.9.4)
 | 
			
		||||
  devise-secure_password!
 | 
			
		||||
@@ -919,6 +1013,7 @@ DEPENDENCIES
 | 
			
		||||
  facebook-messenger
 | 
			
		||||
  factory_bot_rails (>= 6.4.3)
 | 
			
		||||
  faker
 | 
			
		||||
  faraday_middleware-aws-sigv4
 | 
			
		||||
  fcm
 | 
			
		||||
  flag_shih_tzu
 | 
			
		||||
  foreman
 | 
			
		||||
@@ -959,6 +1054,8 @@ DEPENDENCIES
 | 
			
		||||
  omniauth-google-oauth2 (>= 1.1.3)
 | 
			
		||||
  omniauth-oauth2
 | 
			
		||||
  omniauth-rails_csrf_protection (~> 1.0, >= 1.0.2)
 | 
			
		||||
  omniauth-saml
 | 
			
		||||
  opensearch-ruby
 | 
			
		||||
  pg
 | 
			
		||||
  pg_search
 | 
			
		||||
  pgvector
 | 
			
		||||
@@ -984,8 +1081,10 @@ DEPENDENCIES
 | 
			
		||||
  rubocop-rails
 | 
			
		||||
  rubocop-rspec
 | 
			
		||||
  ruby-openai
 | 
			
		||||
  ruby_llm-schema
 | 
			
		||||
  scout_apm
 | 
			
		||||
  scss_lint
 | 
			
		||||
  searchkick
 | 
			
		||||
  seed_dump
 | 
			
		||||
  sentry-rails (>= 5.19.0)
 | 
			
		||||
  sentry-ruby
 | 
			
		||||
@@ -994,7 +1093,9 @@ DEPENDENCIES
 | 
			
		||||
  shoulda-matchers
 | 
			
		||||
  sidekiq (>= 7.3.1)
 | 
			
		||||
  sidekiq-cron (>= 1.12.0)
 | 
			
		||||
  simplecov (= 0.17.1)
 | 
			
		||||
  sidekiq_alive
 | 
			
		||||
  simplecov (>= 0.21)
 | 
			
		||||
  simplecov_json_formatter
 | 
			
		||||
  slack-ruby-client (~> 2.5.2)
 | 
			
		||||
  spring
 | 
			
		||||
  spring-watcher-listen
 | 
			
		||||
@@ -1003,8 +1104,9 @@ DEPENDENCIES
 | 
			
		||||
  stripe
 | 
			
		||||
  telephone_number
 | 
			
		||||
  test-prof
 | 
			
		||||
  tidewave
 | 
			
		||||
  time_diff
 | 
			
		||||
  twilio-ruby (~> 5.66)
 | 
			
		||||
  twilio-ruby
 | 
			
		||||
  twitty (~> 0.1.5)
 | 
			
		||||
  tzinfo-data
 | 
			
		||||
  uglifier
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										9
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										9
									
								
								Makefile
									
									
									
									
									
								
							@@ -41,8 +41,15 @@ run:
 | 
			
		||||
 | 
			
		||||
force_run:
 | 
			
		||||
	rm -f ./.overmind.sock
 | 
			
		||||
	rm -f tmp/pids/*.pid
 | 
			
		||||
	overmind start -f Procfile.dev
 | 
			
		||||
 | 
			
		||||
force_run_tunnel:
 | 
			
		||||
	lsof -ti:3000 | xargs kill -9 2>/dev/null || true
 | 
			
		||||
	rm -f ./.overmind.sock
 | 
			
		||||
	rm -f tmp/pids/*.pid
 | 
			
		||||
	overmind start -f Procfile.tunnel
 | 
			
		||||
 | 
			
		||||
debug:
 | 
			
		||||
	overmind connect backend
 | 
			
		||||
 | 
			
		||||
@@ -52,4 +59,4 @@ debug_worker:
 | 
			
		||||
docker: 
 | 
			
		||||
	docker build -t $(APP_NAME) -f ./docker/Dockerfile .
 | 
			
		||||
 | 
			
		||||
.PHONY: setup db_create db_migrate db_seed db_reset db console server burn docker run force_run debug debug_worker
 | 
			
		||||
.PHONY: setup db_create db_migrate db_seed db_reset db console server burn docker run force_run force_run_tunnel debug debug_worker
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										4
									
								
								Procfile.tunnel
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								Procfile.tunnel
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,4 @@
 | 
			
		||||
backend: DISABLE_MINI_PROFILER=true bin/rails s -p 3000
 | 
			
		||||
# https://github.com/mperham/sidekiq/issues/3090#issuecomment-389748695
 | 
			
		||||
worker: dotenv bundle exec sidekiq -C config/sidekiq.yml
 | 
			
		||||
vite: bin/vite build --watch
 | 
			
		||||
							
								
								
									
										3
									
								
								Rakefile
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								Rakefile
									
									
									
									
									
								
							@@ -2,5 +2,8 @@
 | 
			
		||||
# for example lib/tasks/capistrano.rake, and they will automatically be available to Rake.
 | 
			
		||||
 | 
			
		||||
require_relative 'config/application'
 | 
			
		||||
# Load Enterprise Edition rake tasks if they exist
 | 
			
		||||
enterprise_tasks_path = Rails.root.join('enterprise/tasks_railtie.rb').to_s
 | 
			
		||||
require enterprise_tasks_path if File.exist?(enterprise_tasks_path)
 | 
			
		||||
 | 
			
		||||
Rails.application.load_tasks
 | 
			
		||||
 
 | 
			
		||||
@@ -1 +1 @@
 | 
			
		||||
3.13.0
 | 
			
		||||
4.4.0
 | 
			
		||||
 
 | 
			
		||||
@@ -1 +1 @@
 | 
			
		||||
3.2.0
 | 
			
		||||
3.4.3
 | 
			
		||||
 
 | 
			
		||||
@@ -6,6 +6,7 @@
 | 
			
		||||
# We don't want to update the name of the identified original contact.
 | 
			
		||||
 | 
			
		||||
class ContactIdentifyAction
 | 
			
		||||
  include UrlHelper
 | 
			
		||||
  pattr_initialize [:contact!, :params!, { retain_original_contact_name: false, discard_invalid_attrs: false }]
 | 
			
		||||
 | 
			
		||||
  def perform
 | 
			
		||||
@@ -104,7 +105,14 @@ class ContactIdentifyAction
 | 
			
		||||
    # TODO: replace reject { |_k, v| v.blank? } with compact_blank when rails is upgraded
 | 
			
		||||
    @contact.discard_invalid_attrs if discard_invalid_attrs
 | 
			
		||||
    @contact.save!
 | 
			
		||||
    Avatar::AvatarFromUrlJob.perform_later(@contact, params[:avatar_url]) if params[:avatar_url].present? && !@contact.avatar.attached?
 | 
			
		||||
    enqueue_avatar_job
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def enqueue_avatar_job
 | 
			
		||||
    return unless params[:avatar_url].present? && !@contact.avatar.attached?
 | 
			
		||||
    return unless url_valid?(params[:avatar_url])
 | 
			
		||||
 | 
			
		||||
    Avatar::AvatarFromUrlJob.perform_later(@contact, params[:avatar_url])
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def merge_contact(base_contact, merge_contact)
 | 
			
		||||
 
 | 
			
		||||
@@ -10,7 +10,8 @@ function toggleSecretField(e) {
 | 
			
		||||
  if (!textElement) return;
 | 
			
		||||
 | 
			
		||||
  if (textElement.dataset.secretMasked === 'false') {
 | 
			
		||||
    textElement.textContent = '•'.repeat(10);
 | 
			
		||||
    const maskedLength = secretField.dataset.secretText?.length || 10;
 | 
			
		||||
    textElement.textContent = '•'.repeat(maskedLength);
 | 
			
		||||
    textElement.dataset.secretMasked = 'true';
 | 
			
		||||
    toggler.querySelector('svg use').setAttribute('xlink:href', '#eye-show');
 | 
			
		||||
 | 
			
		||||
@@ -32,3 +33,13 @@ function copySecretField(e) {
 | 
			
		||||
 | 
			
		||||
  navigator.clipboard.writeText(secretField.dataset.secretText);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
document.addEventListener('DOMContentLoaded', () => {
 | 
			
		||||
  document.querySelectorAll('.cell-data__secret-field').forEach(field => {
 | 
			
		||||
    const span = field.querySelector('[data-secret-masked]');
 | 
			
		||||
    if (span && span.dataset.secretMasked === 'true') {
 | 
			
		||||
      const len = field.dataset.secretText?.length || 10;
 | 
			
		||||
      span.textContent = '•'.repeat(len);
 | 
			
		||||
    }
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
 
 | 
			
		||||
@@ -46,17 +46,25 @@
 | 
			
		||||
 | 
			
		||||
.cell-data__secret-field {
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  color: $hint-grey;
 | 
			
		||||
  display: flex;
 | 
			
		||||
 | 
			
		||||
  span {
 | 
			
		||||
    flex: 1;
 | 
			
		||||
    flex: 0 0 auto;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  button {
 | 
			
		||||
    margin-left: 5px;
 | 
			
		||||
  [data-secret-toggler],
 | 
			
		||||
  [data-secret-copier] {
 | 
			
		||||
    background: transparent;
 | 
			
		||||
    border: 0;
 | 
			
		||||
    color: inherit;
 | 
			
		||||
    margin-left: 0.5rem;
 | 
			
		||||
    padding: 0;
 | 
			
		||||
 | 
			
		||||
    svg {
 | 
			
		||||
      fill: currentColor;
 | 
			
		||||
      height: 1.25rem;
 | 
			
		||||
      width: 1.25rem;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										112
									
								
								app/builders/v2/reports/label_summary_builder.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								app/builders/v2/reports/label_summary_builder.rb
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,112 @@
 | 
			
		||||
class V2::Reports::LabelSummaryBuilder < V2::Reports::BaseSummaryBuilder
 | 
			
		||||
  attr_reader :account, :params
 | 
			
		||||
 | 
			
		||||
  # rubocop:disable Lint/MissingSuper
 | 
			
		||||
  # the parent class has no initialize
 | 
			
		||||
  def initialize(account:, params:)
 | 
			
		||||
    @account = account
 | 
			
		||||
    @params = params
 | 
			
		||||
 | 
			
		||||
    timezone_offset = (params[:timezone_offset] || 0).to_f
 | 
			
		||||
    @timezone = ActiveSupport::TimeZone[timezone_offset]&.name
 | 
			
		||||
  end
 | 
			
		||||
  # rubocop:enable Lint/MissingSuper
 | 
			
		||||
 | 
			
		||||
  def build
 | 
			
		||||
    labels = account.labels.to_a
 | 
			
		||||
    return [] if labels.empty?
 | 
			
		||||
 | 
			
		||||
    report_data = collect_report_data
 | 
			
		||||
    labels.map { |label| build_label_report(label, report_data) }
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def collect_report_data
 | 
			
		||||
    conversation_filter = build_conversation_filter
 | 
			
		||||
    use_business_hours = use_business_hours?
 | 
			
		||||
 | 
			
		||||
    {
 | 
			
		||||
      conversation_counts: fetch_conversation_counts(conversation_filter),
 | 
			
		||||
      resolved_counts: fetch_resolved_counts,
 | 
			
		||||
      resolution_metrics: fetch_metrics(conversation_filter, 'conversation_resolved', use_business_hours),
 | 
			
		||||
      first_response_metrics: fetch_metrics(conversation_filter, 'first_response', use_business_hours),
 | 
			
		||||
      reply_metrics: fetch_metrics(conversation_filter, 'reply_time', use_business_hours)
 | 
			
		||||
    }
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def build_label_report(label, report_data)
 | 
			
		||||
    {
 | 
			
		||||
      id: label.id,
 | 
			
		||||
      name: label.title,
 | 
			
		||||
      conversations_count: report_data[:conversation_counts][label.title] || 0,
 | 
			
		||||
      avg_resolution_time: report_data[:resolution_metrics][label.title] || 0,
 | 
			
		||||
      avg_first_response_time: report_data[:first_response_metrics][label.title] || 0,
 | 
			
		||||
      avg_reply_time: report_data[:reply_metrics][label.title] || 0,
 | 
			
		||||
      resolved_conversations_count: report_data[:resolved_counts][label.title] || 0
 | 
			
		||||
    }
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def use_business_hours?
 | 
			
		||||
    ActiveModel::Type::Boolean.new.cast(params[:business_hours])
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def build_conversation_filter
 | 
			
		||||
    conversation_filter = { account_id: account.id }
 | 
			
		||||
    conversation_filter[:created_at] = range if range.present?
 | 
			
		||||
 | 
			
		||||
    conversation_filter
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def fetch_conversation_counts(conversation_filter)
 | 
			
		||||
    fetch_counts(conversation_filter)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def fetch_resolved_counts
 | 
			
		||||
    # Count resolution events, not conversations currently in resolved status
 | 
			
		||||
    # Filter by reporting_event.created_at, not conversation.created_at
 | 
			
		||||
    reporting_event_filter = { name: 'conversation_resolved', account_id: account.id }
 | 
			
		||||
    reporting_event_filter[:created_at] = range if range.present?
 | 
			
		||||
 | 
			
		||||
    ReportingEvent
 | 
			
		||||
      .joins(conversation: { taggings: :tag })
 | 
			
		||||
      .where(
 | 
			
		||||
        reporting_event_filter.merge(
 | 
			
		||||
          taggings: { taggable_type: 'Conversation', context: 'labels' }
 | 
			
		||||
        )
 | 
			
		||||
      )
 | 
			
		||||
      .group('tags.name')
 | 
			
		||||
      .count
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def fetch_counts(conversation_filter)
 | 
			
		||||
    ActsAsTaggableOn::Tagging
 | 
			
		||||
      .joins('INNER JOIN conversations ON taggings.taggable_id = conversations.id')
 | 
			
		||||
      .joins('INNER JOIN tags ON taggings.tag_id = tags.id')
 | 
			
		||||
      .where(
 | 
			
		||||
        taggable_type: 'Conversation',
 | 
			
		||||
        context: 'labels',
 | 
			
		||||
        conversations: conversation_filter
 | 
			
		||||
      )
 | 
			
		||||
      .select('tags.name, COUNT(taggings.*) AS count')
 | 
			
		||||
      .group('tags.name')
 | 
			
		||||
      .each_with_object({}) { |record, hash| hash[record.name] = record.count }
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def fetch_metrics(conversation_filter, event_name, use_business_hours)
 | 
			
		||||
    ReportingEvent
 | 
			
		||||
      .joins(conversation: { taggings: :tag })
 | 
			
		||||
      .where(
 | 
			
		||||
        conversations: conversation_filter,
 | 
			
		||||
        name: event_name,
 | 
			
		||||
        taggings: { taggable_type: 'Conversation', context: 'labels' }
 | 
			
		||||
      )
 | 
			
		||||
      .group('tags.name')
 | 
			
		||||
      .order('tags.name')
 | 
			
		||||
      .select(
 | 
			
		||||
        'tags.name',
 | 
			
		||||
        use_business_hours ? 'AVG(reporting_events.value_in_business_hours) as avg_value' : 'AVG(reporting_events.value) as avg_value'
 | 
			
		||||
      )
 | 
			
		||||
      .each_with_object({}) { |record, hash| hash[record.name] = record.avg_value.to_f }
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -38,27 +38,34 @@ class V2::Reports::Timeseries::CountReportBuilder < V2::Reports::Timeseries::Bas
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def scope_for_resolutions_count
 | 
			
		||||
    scope.reporting_events.joins(:conversation).select(:conversation_id).where(
 | 
			
		||||
    scope.reporting_events.where(
 | 
			
		||||
      name: :conversation_resolved,
 | 
			
		||||
      conversations: { status: :resolved }, created_at: range
 | 
			
		||||
    ).distinct
 | 
			
		||||
      account_id: account.id,
 | 
			
		||||
      created_at: range
 | 
			
		||||
    )
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def scope_for_bot_resolutions_count
 | 
			
		||||
    scope.reporting_events.joins(:conversation).select(:conversation_id).where(
 | 
			
		||||
    scope.reporting_events.where(
 | 
			
		||||
      name: :conversation_bot_resolved,
 | 
			
		||||
      conversations: { status: :resolved }, created_at: range
 | 
			
		||||
    ).distinct
 | 
			
		||||
      account_id: account.id,
 | 
			
		||||
      created_at: range
 | 
			
		||||
    )
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def scope_for_bot_handoffs_count
 | 
			
		||||
    scope.reporting_events.joins(:conversation).select(:conversation_id).where(
 | 
			
		||||
      name: :conversation_bot_handoff,
 | 
			
		||||
      account_id: account.id,
 | 
			
		||||
      created_at: range
 | 
			
		||||
    ).distinct
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def grouped_count
 | 
			
		||||
    # IMPORTANT: time_zone parameter affects both data grouping AND output timestamps
 | 
			
		||||
    # It converts timestamps to the target timezone before grouping, which means
 | 
			
		||||
    # the same event can fall into different day buckets depending on timezone
 | 
			
		||||
    # Example: 2024-01-15 00:00 UTC becomes 2024-01-14 16:00 PST (falls on different day)
 | 
			
		||||
    @grouped_values = object_scope.group_by_period(
 | 
			
		||||
      group_by,
 | 
			
		||||
      :created_at,
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,20 @@
 | 
			
		||||
class Api::V1::Accounts::AssignmentPolicies::InboxesController < Api::V1::Accounts::BaseController
 | 
			
		||||
  before_action :fetch_assignment_policy
 | 
			
		||||
  before_action -> { check_authorization(AssignmentPolicy) }
 | 
			
		||||
 | 
			
		||||
  def index
 | 
			
		||||
    @inboxes = @assignment_policy.inboxes
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def fetch_assignment_policy
 | 
			
		||||
    @assignment_policy = Current.account.assignment_policies.find(
 | 
			
		||||
      params[:assignment_policy_id]
 | 
			
		||||
    )
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def permitted_params
 | 
			
		||||
    params.permit(:assignment_policy_id)
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -0,0 +1,36 @@
 | 
			
		||||
class Api::V1::Accounts::AssignmentPoliciesController < Api::V1::Accounts::BaseController
 | 
			
		||||
  before_action :fetch_assignment_policy, only: [:show, :update, :destroy]
 | 
			
		||||
  before_action :check_authorization
 | 
			
		||||
 | 
			
		||||
  def index
 | 
			
		||||
    @assignment_policies = Current.account.assignment_policies
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def show; end
 | 
			
		||||
 | 
			
		||||
  def create
 | 
			
		||||
    @assignment_policy = Current.account.assignment_policies.create!(assignment_policy_params)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def update
 | 
			
		||||
    @assignment_policy.update!(assignment_policy_params)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def destroy
 | 
			
		||||
    @assignment_policy.destroy!
 | 
			
		||||
    head :ok
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def fetch_assignment_policy
 | 
			
		||||
    @assignment_policy = Current.account.assignment_policies.find(params[:id])
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def assignment_policy_params
 | 
			
		||||
    params.require(:assignment_policy).permit(
 | 
			
		||||
      :name, :description, :assignment_order, :conversation_priority,
 | 
			
		||||
      :fair_distribution_limit, :fair_distribution_window, :enabled
 | 
			
		||||
    )
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -30,7 +30,14 @@ class Api::V1::Accounts::CallbacksController < Api::V1::Accounts::BaseController
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def facebook_pages
 | 
			
		||||
    @page_details = mark_already_existing_facebook_pages(fb_object.get_connections('me', 'accounts'))
 | 
			
		||||
    pages = []
 | 
			
		||||
    fb_pages = fb_object.get_connections('me', 'accounts')
 | 
			
		||||
    pages.concat(fb_pages)
 | 
			
		||||
    while fb_pages.respond_to?(:next_page) && (next_page = fb_pages.next_page)
 | 
			
		||||
      fb_pages = next_page
 | 
			
		||||
      pages.concat(fb_pages)
 | 
			
		||||
    end
 | 
			
		||||
    @page_details = mark_already_existing_facebook_pages(pages)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def set_instagram_id(page_access_token, facebook_channel)
 | 
			
		||||
 
 | 
			
		||||
@@ -29,6 +29,6 @@ class Api::V1::Accounts::CampaignsController < Api::V1::Accounts::BaseController
 | 
			
		||||
 | 
			
		||||
  def campaign_params
 | 
			
		||||
    params.require(:campaign).permit(:title, :description, :message, :enabled, :trigger_only_during_business_hours, :inbox_id, :sender_id,
 | 
			
		||||
                                     :scheduled_at, audience: [:type, :id], trigger_rules: {})
 | 
			
		||||
                                     :scheduled_at, audience: [:type, :id], trigger_rules: {}, template_params: {})
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -14,7 +14,7 @@ class Api::V1::Accounts::ContactsController < Api::V1::Accounts::BaseController
 | 
			
		||||
  before_action :check_authorization
 | 
			
		||||
  before_action :set_current_page, only: [:index, :active, :search, :filter]
 | 
			
		||||
  before_action :fetch_contact, only: [:show, :update, :destroy, :avatar, :contactable_inboxes, :destroy_custom_attributes]
 | 
			
		||||
  before_action :set_include_contact_inboxes, only: [:index, :search, :filter, :show, :update]
 | 
			
		||||
  before_action :set_include_contact_inboxes, only: [:index, :active, :search, :filter, :show, :update]
 | 
			
		||||
 | 
			
		||||
  def index
 | 
			
		||||
    @contacts_count = resolved_contacts.count
 | 
			
		||||
@@ -56,7 +56,7 @@ class Api::V1::Accounts::ContactsController < Api::V1::Accounts::BaseController
 | 
			
		||||
    contacts = Current.account.contacts.where(id: ::OnlineStatusTracker
 | 
			
		||||
                  .get_available_contact_ids(Current.account.id))
 | 
			
		||||
    @contacts_count = contacts.count
 | 
			
		||||
    @contacts = contacts.page(@current_page)
 | 
			
		||||
    @contacts = fetch_contacts(contacts)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def show; end
 | 
			
		||||
@@ -122,7 +122,7 @@ class Api::V1::Accounts::ContactsController < Api::V1::Accounts::BaseController
 | 
			
		||||
  def resolved_contacts
 | 
			
		||||
    return @resolved_contacts if @resolved_contacts
 | 
			
		||||
 | 
			
		||||
    @resolved_contacts = Current.account.contacts.resolved_contacts
 | 
			
		||||
    @resolved_contacts = Current.account.contacts.resolved_contacts(use_crm_v2: Current.account.feature_enabled?('crm_v2'))
 | 
			
		||||
 | 
			
		||||
    @resolved_contacts = @resolved_contacts.tagged_with(params[:labels], any: true) if params[:labels].present?
 | 
			
		||||
    @resolved_contacts
 | 
			
		||||
 
 | 
			
		||||
@@ -124,6 +124,12 @@ class Api::V1::Accounts::ConversationsController < Api::V1::Accounts::BaseContro
 | 
			
		||||
    @conversation.save!
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def destroy
 | 
			
		||||
    authorize @conversation, :destroy?
 | 
			
		||||
    ::DeleteObjectJob.perform_later(@conversation, Current.user, request.ip)
 | 
			
		||||
    head :ok
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def permitted_update_params
 | 
			
		||||
 
 | 
			
		||||
@@ -1,32 +1,23 @@
 | 
			
		||||
class Api::V1::Accounts::Google::AuthorizationsController < Api::V1::Accounts::BaseController
 | 
			
		||||
class Api::V1::Accounts::Google::AuthorizationsController < Api::V1::Accounts::OauthAuthorizationController
 | 
			
		||||
  include GoogleConcern
 | 
			
		||||
  before_action :check_authorization
 | 
			
		||||
 | 
			
		||||
  def create
 | 
			
		||||
    email = params[:authorization][:email]
 | 
			
		||||
    redirect_url = google_client.auth_code.authorize_url(
 | 
			
		||||
      {
 | 
			
		||||
        redirect_uri: "#{base_url}/google/callback",
 | 
			
		||||
        scope: 'email profile https://mail.google.com/',
 | 
			
		||||
        scope: scope,
 | 
			
		||||
        response_type: 'code',
 | 
			
		||||
        prompt: 'consent', # the oauth flow does not return a refresh token, this is supposed to fix it
 | 
			
		||||
        access_type: 'offline', # the default is 'online'
 | 
			
		||||
        state: state,
 | 
			
		||||
        client_id: GlobalConfigService.load('GOOGLE_OAUTH_CLIENT_ID', nil)
 | 
			
		||||
      }
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    if redirect_url
 | 
			
		||||
      cache_key = "google::#{email.downcase}"
 | 
			
		||||
      ::Redis::Alfred.setex(cache_key, Current.account.id, 5.minutes)
 | 
			
		||||
      render json: { success: true, url: redirect_url }
 | 
			
		||||
    else
 | 
			
		||||
      render json: { success: false }, status: :unprocessable_entity
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def check_authorization
 | 
			
		||||
    raise Pundit::NotAuthorizedError unless Current.account_user.administrator?
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,46 @@
 | 
			
		||||
class Api::V1::Accounts::Inboxes::AssignmentPoliciesController < Api::V1::Accounts::BaseController
 | 
			
		||||
  before_action :fetch_inbox
 | 
			
		||||
  before_action :fetch_assignment_policy, only: [:create]
 | 
			
		||||
  before_action -> { check_authorization(AssignmentPolicy) }
 | 
			
		||||
  before_action :validate_assignment_policy, only: [:show, :destroy]
 | 
			
		||||
 | 
			
		||||
  def show
 | 
			
		||||
    @assignment_policy = @inbox.assignment_policy
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def create
 | 
			
		||||
    # There should be only one assignment policy for an inbox.
 | 
			
		||||
    # If there is a new request to add an assignment policy, we will
 | 
			
		||||
    # delete the old one and attach the new policy
 | 
			
		||||
    remove_inbox_assignment_policy
 | 
			
		||||
    @inbox_assignment_policy = @inbox.create_inbox_assignment_policy!(assignment_policy: @assignment_policy)
 | 
			
		||||
    @assignment_policy = @inbox.assignment_policy
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def destroy
 | 
			
		||||
    remove_inbox_assignment_policy
 | 
			
		||||
    head :ok
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def remove_inbox_assignment_policy
 | 
			
		||||
    @inbox.inbox_assignment_policy&.destroy
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def fetch_inbox
 | 
			
		||||
    @inbox = Current.account.inboxes.find(permitted_params[:inbox_id])
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def fetch_assignment_policy
 | 
			
		||||
    @assignment_policy = Current.account.assignment_policies.find(permitted_params[:assignment_policy_id])
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def permitted_params
 | 
			
		||||
    params.permit(:assignment_policy_id, :inbox_id)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def validate_assignment_policy
 | 
			
		||||
    return render_not_found_error(I18n.t('errors.assignment_policy.not_found')) unless @inbox.assignment_policy
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -69,6 +69,15 @@ class Api::V1::Accounts::InboxesController < Api::V1::Accounts::BaseController
 | 
			
		||||
    render status: :ok, json: { message: I18n.t('messages.inbox_deletetion_response') }
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def sync_templates
 | 
			
		||||
    return render status: :unprocessable_entity, json: { error: 'Template sync is only available for WhatsApp channels' } unless whatsapp_channel?
 | 
			
		||||
 | 
			
		||||
    trigger_template_sync
 | 
			
		||||
    render status: :ok, json: { message: 'Template sync initiated successfully' }
 | 
			
		||||
  rescue StandardError => e
 | 
			
		||||
    render status: :internal_server_error, json: { error: e.message }
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def fetch_inbox
 | 
			
		||||
@@ -81,11 +90,15 @@ class Api::V1::Accounts::InboxesController < Api::V1::Accounts::BaseController
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def create_channel
 | 
			
		||||
    return unless %w[web_widget api email line telegram whatsapp sms].include?(permitted_params[:channel][:type])
 | 
			
		||||
    return unless allowed_channel_types.include?(permitted_params[:channel][:type])
 | 
			
		||||
 | 
			
		||||
    account_channels_method.create!(permitted_params(channel_type_from_params::EDITABLE_ATTRS)[:channel].except(:type))
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def allowed_channel_types
 | 
			
		||||
    %w[web_widget api email line telegram whatsapp sms]
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def update_inbox_working_hours
 | 
			
		||||
    @inbox.update_working_hours(params.permit(working_hours: Inbox::OFFISABLE_ATTRS)[:working_hours]) if params[:working_hours]
 | 
			
		||||
  end
 | 
			
		||||
@@ -170,6 +183,18 @@ class Api::V1::Accounts::InboxesController < Api::V1::Accounts::BaseController
 | 
			
		||||
      []
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def whatsapp_channel?
 | 
			
		||||
    @inbox.whatsapp? || (@inbox.twilio? && @inbox.channel.whatsapp?)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def trigger_template_sync
 | 
			
		||||
    if @inbox.whatsapp?
 | 
			
		||||
      Channels::Whatsapp::TemplatesSyncJob.perform_later(@inbox.channel)
 | 
			
		||||
    elsif @inbox.twilio? && @inbox.channel.whatsapp?
 | 
			
		||||
      Channels::Twilio::TemplatesSyncJob.perform_later(@inbox.channel)
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
Api::V1::Accounts::InboxesController.prepend_mod_with('Api::V1::Accounts::InboxesController')
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,6 @@
 | 
			
		||||
class Api::V1::Accounts::Instagram::AuthorizationsController < Api::V1::Accounts::BaseController
 | 
			
		||||
class Api::V1::Accounts::Instagram::AuthorizationsController < Api::V1::Accounts::OauthAuthorizationController
 | 
			
		||||
  include InstagramConcern
 | 
			
		||||
  include Instagram::IntegrationHelper
 | 
			
		||||
  before_action :check_authorization
 | 
			
		||||
 | 
			
		||||
  def create
 | 
			
		||||
    # https://developers.facebook.com/docs/instagram-platform/instagram-api-with-instagram-login/business-login#step-1--get-authorization
 | 
			
		||||
@@ -21,10 +20,4 @@ class Api::V1::Accounts::Instagram::AuthorizationsController < Api::V1::Accounts
 | 
			
		||||
      render json: { success: false }, status: :unprocessable_entity
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def check_authorization
 | 
			
		||||
    raise Pundit::NotAuthorizedError unless Current.account_user.administrator?
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -1,8 +1,9 @@
 | 
			
		||||
class Api::V1::Accounts::Integrations::LinearController < Api::V1::Accounts::BaseController
 | 
			
		||||
  before_action :fetch_conversation, only: [:link_issue, :linked_issues]
 | 
			
		||||
  before_action :fetch_conversation, only: [:create_issue, :link_issue, :unlink_issue, :linked_issues]
 | 
			
		||||
  before_action :fetch_hook, only: [:destroy]
 | 
			
		||||
 | 
			
		||||
  def destroy
 | 
			
		||||
    revoke_linear_token
 | 
			
		||||
    @hook.destroy!
 | 
			
		||||
    head :ok
 | 
			
		||||
  end
 | 
			
		||||
@@ -27,10 +28,16 @@ class Api::V1::Accounts::Integrations::LinearController < Api::V1::Accounts::Bas
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def create_issue
 | 
			
		||||
    issue = linear_processor_service.create_issue(permitted_params)
 | 
			
		||||
    issue = linear_processor_service.create_issue(permitted_params, Current.user)
 | 
			
		||||
    if issue[:error]
 | 
			
		||||
      render json: { error: issue[:error] }, status: :unprocessable_entity
 | 
			
		||||
    else
 | 
			
		||||
      Linear::ActivityMessageService.new(
 | 
			
		||||
        conversation: @conversation,
 | 
			
		||||
        action_type: :issue_created,
 | 
			
		||||
        issue_data: { id: issue[:data][:identifier] },
 | 
			
		||||
        user: Current.user
 | 
			
		||||
      ).perform
 | 
			
		||||
      render json: issue[:data], status: :ok
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
@@ -38,21 +45,34 @@ class Api::V1::Accounts::Integrations::LinearController < Api::V1::Accounts::Bas
 | 
			
		||||
  def link_issue
 | 
			
		||||
    issue_id = permitted_params[:issue_id]
 | 
			
		||||
    title = permitted_params[:title]
 | 
			
		||||
    issue = linear_processor_service.link_issue(conversation_link, issue_id, title)
 | 
			
		||||
    issue = linear_processor_service.link_issue(conversation_link, issue_id, title, Current.user)
 | 
			
		||||
    if issue[:error]
 | 
			
		||||
      render json: { error: issue[:error] }, status: :unprocessable_entity
 | 
			
		||||
    else
 | 
			
		||||
      Linear::ActivityMessageService.new(
 | 
			
		||||
        conversation: @conversation,
 | 
			
		||||
        action_type: :issue_linked,
 | 
			
		||||
        issue_data: { id: issue_id },
 | 
			
		||||
        user: Current.user
 | 
			
		||||
      ).perform
 | 
			
		||||
      render json: issue[:data], status: :ok
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def unlink_issue
 | 
			
		||||
    link_id = permitted_params[:link_id]
 | 
			
		||||
    issue_id = permitted_params[:issue_id]
 | 
			
		||||
    issue = linear_processor_service.unlink_issue(link_id)
 | 
			
		||||
 | 
			
		||||
    if issue[:error]
 | 
			
		||||
      render json: { error: issue[:error] }, status: :unprocessable_entity
 | 
			
		||||
    else
 | 
			
		||||
      Linear::ActivityMessageService.new(
 | 
			
		||||
        conversation: @conversation,
 | 
			
		||||
        action_type: :issue_unlinked,
 | 
			
		||||
        issue_data: { id: issue_id },
 | 
			
		||||
        user: Current.user
 | 
			
		||||
      ).perform
 | 
			
		||||
      render json: issue[:data], status: :ok
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
@@ -101,4 +121,15 @@ class Api::V1::Accounts::Integrations::LinearController < Api::V1::Accounts::Bas
 | 
			
		||||
  def fetch_hook
 | 
			
		||||
    @hook = Integrations::Hook.where(account: Current.account).find_by(app_id: 'linear')
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def revoke_linear_token
 | 
			
		||||
    return unless @hook&.access_token
 | 
			
		||||
 | 
			
		||||
    begin
 | 
			
		||||
      linear_client = Linear.new(@hook.access_token)
 | 
			
		||||
      linear_client.revoke_token
 | 
			
		||||
    rescue StandardError => e
 | 
			
		||||
      Rails.logger.error "Failed to revoke Linear token: #{e.message}"
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,14 @@
 | 
			
		||||
class Api::V1::Accounts::Integrations::NotionController < Api::V1::Accounts::BaseController
 | 
			
		||||
  before_action :fetch_hook, only: [:destroy]
 | 
			
		||||
 | 
			
		||||
  def destroy
 | 
			
		||||
    @hook.destroy!
 | 
			
		||||
    head :ok
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def fetch_hook
 | 
			
		||||
    @hook = Integrations::Hook.where(account: Current.account).find_by(app_id: 'notion')
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -1,28 +1,19 @@
 | 
			
		||||
class Api::V1::Accounts::Microsoft::AuthorizationsController < Api::V1::Accounts::BaseController
 | 
			
		||||
class Api::V1::Accounts::Microsoft::AuthorizationsController < Api::V1::Accounts::OauthAuthorizationController
 | 
			
		||||
  include MicrosoftConcern
 | 
			
		||||
  before_action :check_authorization
 | 
			
		||||
 | 
			
		||||
  def create
 | 
			
		||||
    email = params[:authorization][:email]
 | 
			
		||||
    redirect_url = microsoft_client.auth_code.authorize_url(
 | 
			
		||||
      {
 | 
			
		||||
        redirect_uri: "#{base_url}/microsoft/callback",
 | 
			
		||||
        scope: 'offline_access https://outlook.office.com/IMAP.AccessAsUser.All https://outlook.office.com/SMTP.Send openid profile',
 | 
			
		||||
        scope: scope,
 | 
			
		||||
        state: state,
 | 
			
		||||
        prompt: 'consent'
 | 
			
		||||
      }
 | 
			
		||||
    )
 | 
			
		||||
    if redirect_url
 | 
			
		||||
      cache_key = "microsoft::#{email.downcase}"
 | 
			
		||||
      ::Redis::Alfred.setex(cache_key, Current.account.id, 5.minutes)
 | 
			
		||||
      render json: { success: true, url: redirect_url }
 | 
			
		||||
    else
 | 
			
		||||
      render json: { success: false }, status: :unprocessable_entity
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def check_authorization
 | 
			
		||||
    raise Pundit::NotAuthorizedError unless Current.account_user.administrator?
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,21 @@
 | 
			
		||||
class Api::V1::Accounts::Notion::AuthorizationsController < Api::V1::Accounts::OauthAuthorizationController
 | 
			
		||||
  include NotionConcern
 | 
			
		||||
 | 
			
		||||
  def create
 | 
			
		||||
    redirect_url = notion_client.auth_code.authorize_url(
 | 
			
		||||
      {
 | 
			
		||||
        redirect_uri: "#{base_url}/notion/callback",
 | 
			
		||||
        response_type: 'code',
 | 
			
		||||
        owner: 'user',
 | 
			
		||||
        state: state,
 | 
			
		||||
        client_id: GlobalConfigService.load('NOTION_CLIENT_ID', nil)
 | 
			
		||||
      }
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    if redirect_url
 | 
			
		||||
      render json: { success: true, url: redirect_url }
 | 
			
		||||
    else
 | 
			
		||||
      render json: { success: false }, status: :unprocessable_entity
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -0,0 +1,23 @@
 | 
			
		||||
class Api::V1::Accounts::OauthAuthorizationController < Api::V1::Accounts::BaseController
 | 
			
		||||
  before_action :check_authorization
 | 
			
		||||
 | 
			
		||||
  protected
 | 
			
		||||
 | 
			
		||||
  def scope
 | 
			
		||||
    ''
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def state
 | 
			
		||||
    Current.account.to_sgid(expires_in: 15.minutes).to_s
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def base_url
 | 
			
		||||
    ENV.fetch('FRONTEND_URL', 'http://localhost:3000')
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def check_authorization
 | 
			
		||||
    raise Pundit::NotAuthorizedError unless Current.account_user.administrator?
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -26,9 +26,8 @@ class Api::V1::Accounts::PortalsController < Api::V1::Accounts::BaseController
 | 
			
		||||
      @portal.update!(portal_params.merge(live_chat_widget_params)) if params[:portal].present?
 | 
			
		||||
      # @portal.custom_domain = parsed_custom_domain
 | 
			
		||||
      process_attached_logo if params[:blob_id].present?
 | 
			
		||||
    rescue StandardError => e
 | 
			
		||||
      Rails.logger.error e
 | 
			
		||||
      render json: { error: @portal.errors.messages }.to_json, status: :unprocessable_entity
 | 
			
		||||
    rescue ActiveRecord::RecordInvalid => e
 | 
			
		||||
      render_record_invalid(e)
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
@@ -47,6 +46,20 @@ class Api::V1::Accounts::PortalsController < Api::V1::Accounts::BaseController
 | 
			
		||||
    head :ok
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def send_instructions
 | 
			
		||||
    email = permitted_params[:email]
 | 
			
		||||
    return render_could_not_create_error(I18n.t('portals.send_instructions.email_required')) if email.blank?
 | 
			
		||||
    return render_could_not_create_error(I18n.t('portals.send_instructions.invalid_email_format')) unless valid_email?(email)
 | 
			
		||||
    return render_could_not_create_error(I18n.t('portals.send_instructions.custom_domain_not_configured')) if @portal.custom_domain.blank?
 | 
			
		||||
 | 
			
		||||
    PortalInstructionsMailer.send_cname_instructions(
 | 
			
		||||
      portal: @portal,
 | 
			
		||||
      recipient_email: email
 | 
			
		||||
    ).deliver_later
 | 
			
		||||
 | 
			
		||||
    render json: { message: I18n.t('portals.send_instructions.instructions_sent_successfully') }, status: :ok
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def process_attached_logo
 | 
			
		||||
    blob_id = params[:blob_id]
 | 
			
		||||
    blob = ActiveStorage::Blob.find_by(id: blob_id)
 | 
			
		||||
@@ -60,12 +73,12 @@ class Api::V1::Accounts::PortalsController < Api::V1::Accounts::BaseController
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def permitted_params
 | 
			
		||||
    params.permit(:id)
 | 
			
		||||
    params.permit(:id, :email)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def portal_params
 | 
			
		||||
    params.require(:portal).permit(
 | 
			
		||||
      :account_id, :color, :custom_domain, :header_text, :homepage_link,
 | 
			
		||||
      :id, :account_id, :color, :custom_domain, :header_text, :homepage_link,
 | 
			
		||||
      :name, :page_title, :slug, :archived, { config: [:default_locale, { allowed_locales: [] }] }
 | 
			
		||||
    )
 | 
			
		||||
  end
 | 
			
		||||
@@ -88,4 +101,10 @@ class Api::V1::Accounts::PortalsController < Api::V1::Accounts::BaseController
 | 
			
		||||
    domain = URI.parse(@portal.custom_domain)
 | 
			
		||||
    domain.is_a?(URI::HTTP) ? domain.host : @portal.custom_domain
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def valid_email?(email)
 | 
			
		||||
    ValidEmail2::Address.new(email).valid?
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
Api::V1::Accounts::PortalsController.prepend_mod_with('Api::V1::Accounts::PortalsController')
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,77 @@
 | 
			
		||||
class Api::V1::Accounts::Whatsapp::AuthorizationsController < Api::V1::Accounts::BaseController
 | 
			
		||||
  before_action :fetch_and_validate_inbox, if: -> { params[:inbox_id].present? }
 | 
			
		||||
 | 
			
		||||
  # POST /api/v1/accounts/:account_id/whatsapp/authorization
 | 
			
		||||
  # Handles both initial authorization and reauthorization
 | 
			
		||||
  # If inbox_id is present in params, it performs reauthorization
 | 
			
		||||
  def create
 | 
			
		||||
    validate_embedded_signup_params!
 | 
			
		||||
    channel = process_embedded_signup
 | 
			
		||||
    render_success_response(channel.inbox)
 | 
			
		||||
  rescue StandardError => e
 | 
			
		||||
    render_error_response(e)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def process_embedded_signup
 | 
			
		||||
    service = Whatsapp::EmbeddedSignupService.new(
 | 
			
		||||
      account: Current.account,
 | 
			
		||||
      params: params.permit(:code, :business_id, :waba_id, :phone_number_id).to_h.symbolize_keys,
 | 
			
		||||
      inbox_id: params[:inbox_id]
 | 
			
		||||
    )
 | 
			
		||||
    service.perform
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def fetch_and_validate_inbox
 | 
			
		||||
    @inbox = Current.account.inboxes.find(params[:inbox_id])
 | 
			
		||||
    validate_reauthorization_required
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def validate_reauthorization_required
 | 
			
		||||
    return if @inbox.channel.reauthorization_required? || can_upgrade_to_embedded_signup?
 | 
			
		||||
 | 
			
		||||
    render json: {
 | 
			
		||||
      success: false,
 | 
			
		||||
      message: I18n.t('inbox.reauthorization.not_required')
 | 
			
		||||
    }, status: :unprocessable_entity
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def can_upgrade_to_embedded_signup?
 | 
			
		||||
    channel = @inbox.channel
 | 
			
		||||
    return false unless channel.provider == 'whatsapp_cloud'
 | 
			
		||||
 | 
			
		||||
    true
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def render_success_response(inbox)
 | 
			
		||||
    response = {
 | 
			
		||||
      success: true,
 | 
			
		||||
      id: inbox.id,
 | 
			
		||||
      name: inbox.name,
 | 
			
		||||
      channel_type: 'whatsapp'
 | 
			
		||||
    }
 | 
			
		||||
    response[:message] = I18n.t('inbox.reauthorization.success') if params[:inbox_id].present?
 | 
			
		||||
    render json: response
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def render_error_response(error)
 | 
			
		||||
    Rails.logger.error "[WHATSAPP AUTHORIZATION] Embedded signup error: #{error.message}"
 | 
			
		||||
    Rails.logger.error error.backtrace.join("\n")
 | 
			
		||||
    render json: {
 | 
			
		||||
      success: false,
 | 
			
		||||
      error: error.message
 | 
			
		||||
    }, status: :unprocessable_entity
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def validate_embedded_signup_params!
 | 
			
		||||
    missing_params = []
 | 
			
		||||
    missing_params << 'code' if params[:code].blank?
 | 
			
		||||
    missing_params << 'business_id' if params[:business_id].blank?
 | 
			
		||||
    missing_params << 'waba_id' if params[:waba_id].blank?
 | 
			
		||||
 | 
			
		||||
    return if missing_params.empty?
 | 
			
		||||
 | 
			
		||||
    raise ArgumentError, "Required parameters are missing: #{missing_params.join(', ')}"
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -92,7 +92,7 @@ class Api::V1::AccountsController < Api::BaseController
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def settings_params
 | 
			
		||||
    params.permit(:auto_resolve_after, :auto_resolve_message, :auto_resolve_ignore_waiting)
 | 
			
		||||
    params.permit(:auto_resolve_after, :auto_resolve_message, :auto_resolve_ignore_waiting, :audio_transcriptions, :auto_resolve_label)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def check_signup_enabled
 | 
			
		||||
 
 | 
			
		||||
@@ -9,7 +9,7 @@ class Api::V1::Widget::ConfigsController < Api::V1::Widget::BaseController
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def set_global_config
 | 
			
		||||
    @global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL')
 | 
			
		||||
    @global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL', 'INSTALLATION_NAME')
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def set_contact
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
class Api::V2::Accounts::SummaryReportsController < Api::V1::Accounts::BaseController
 | 
			
		||||
  before_action :check_authorization
 | 
			
		||||
  before_action :prepare_builder_params, only: [:agent, :team, :inbox]
 | 
			
		||||
  before_action :prepare_builder_params, only: [:agent, :team, :inbox, :label]
 | 
			
		||||
 | 
			
		||||
  def agent
 | 
			
		||||
    render_report_with(V2::Reports::AgentSummaryBuilder)
 | 
			
		||||
@@ -14,6 +14,10 @@ class Api::V2::Accounts::SummaryReportsController < Api::V1::Accounts::BaseContr
 | 
			
		||||
    render_report_with(V2::Reports::InboxSummaryBuilder)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def label
 | 
			
		||||
    render_report_with(V2::Reports::LabelSummaryBuilder)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def check_authorization
 | 
			
		||||
 
 | 
			
		||||
@@ -14,7 +14,7 @@ module GoogleConcern
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def base_url
 | 
			
		||||
    ENV.fetch('FRONTEND_URL', 'http://localhost:3000')
 | 
			
		||||
  def scope
 | 
			
		||||
    'email profile https://mail.google.com/'
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -15,7 +15,7 @@ module MicrosoftConcern
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def base_url
 | 
			
		||||
    ENV.fetch('FRONTEND_URL', 'http://localhost:3000')
 | 
			
		||||
  def scope
 | 
			
		||||
    'offline_access https://outlook.office.com/IMAP.AccessAsUser.All https://outlook.office.com/SMTP.Send openid profile email'
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										21
									
								
								app/controllers/concerns/notion_concern.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								app/controllers/concerns/notion_concern.rb
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
module NotionConcern
 | 
			
		||||
  extend ActiveSupport::Concern
 | 
			
		||||
 | 
			
		||||
  def notion_client
 | 
			
		||||
    app_id = GlobalConfigService.load('NOTION_CLIENT_ID', nil)
 | 
			
		||||
    app_secret = GlobalConfigService.load('NOTION_CLIENT_SECRET', nil)
 | 
			
		||||
 | 
			
		||||
    ::OAuth2::Client.new(app_id, app_secret, {
 | 
			
		||||
                           site: 'https://api.notion.com',
 | 
			
		||||
                           authorize_url: 'https://api.notion.com/v1/oauth/authorize',
 | 
			
		||||
                           token_url: 'https://api.notion.com/v1/oauth/token',
 | 
			
		||||
                           auth_scheme: :basic_auth
 | 
			
		||||
                         })
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def scope
 | 
			
		||||
    ''
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -4,17 +4,28 @@ module SwitchLocale
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def switch_locale(&)
 | 
			
		||||
    # priority is for locale set in query string (mostly for widget/from js sdk)
 | 
			
		||||
    # Priority is for locale set in query string (mostly for widget/from js sdk)
 | 
			
		||||
    locale ||= params[:locale]
 | 
			
		||||
 | 
			
		||||
    # Use the user's locale if available
 | 
			
		||||
    locale ||= locale_from_user
 | 
			
		||||
 | 
			
		||||
    # Use the locale from a custom domain if applicable
 | 
			
		||||
    locale ||= locale_from_custom_domain
 | 
			
		||||
 | 
			
		||||
    # if locale is not set in account, let's use DEFAULT_LOCALE env variable
 | 
			
		||||
    locale ||= ENV.fetch('DEFAULT_LOCALE', nil)
 | 
			
		||||
 | 
			
		||||
    set_locale(locale, &)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def switch_locale_using_account_locale(&)
 | 
			
		||||
    locale = locale_from_account(@current_account)
 | 
			
		||||
    # Get the locale from the user first
 | 
			
		||||
    locale = locale_from_user
 | 
			
		||||
 | 
			
		||||
    # Fallback to the account's locale if the user's locale is not set
 | 
			
		||||
    locale ||= locale_from_account(@current_account)
 | 
			
		||||
 | 
			
		||||
    set_locale(locale, &)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
@@ -32,6 +43,12 @@ module SwitchLocale
 | 
			
		||||
    @portal.default_locale
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def locale_from_user
 | 
			
		||||
    return unless @user
 | 
			
		||||
 | 
			
		||||
    @user.ui_settings&.dig('locale')
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def set_locale(locale, &)
 | 
			
		||||
    safe_locale = validate_and_get_locale(locale)
 | 
			
		||||
    # Ensure locale won't bleed into other requests
 | 
			
		||||
 
 | 
			
		||||
@@ -15,7 +15,7 @@ class DashboardController < ActionController::Base
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def ensure_html_format
 | 
			
		||||
    head :not_acceptable unless request.format.html?
 | 
			
		||||
    render json: { error: 'Please use API routes instead of dashboard routes for JSON requests' }, status: :not_acceptable if request.format.json?
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def set_global_config
 | 
			
		||||
@@ -66,7 +66,9 @@ class DashboardController < ActionController::Base
 | 
			
		||||
      ENABLE_ACCOUNT_SIGNUP: GlobalConfigService.load('ENABLE_ACCOUNT_SIGNUP', 'false'),
 | 
			
		||||
      FB_APP_ID: GlobalConfigService.load('FB_APP_ID', ''),
 | 
			
		||||
      INSTAGRAM_APP_ID: GlobalConfigService.load('INSTAGRAM_APP_ID', ''),
 | 
			
		||||
      FACEBOOK_API_VERSION: GlobalConfigService.load('FACEBOOK_API_VERSION', 'v17.0'),
 | 
			
		||||
      FACEBOOK_API_VERSION: GlobalConfigService.load('FACEBOOK_API_VERSION', 'v18.0'),
 | 
			
		||||
      WHATSAPP_APP_ID: GlobalConfigService.load('WHATSAPP_APP_ID', ''),
 | 
			
		||||
      WHATSAPP_CONFIGURATION_ID: GlobalConfigService.load('WHATSAPP_CONFIGURATION_ID', ''),
 | 
			
		||||
      IS_ENTERPRISE: ChatwootApp.enterprise?,
 | 
			
		||||
      AZURE_APP_ID: GlobalConfigService.load('AZURE_APP_ID', ''),
 | 
			
		||||
      GIT_SHA: GIT_HASH
 | 
			
		||||
 
 | 
			
		||||
@@ -47,10 +47,8 @@ class DeviseOverrides::OmniauthCallbacksController < DeviseTokenAuth::OmniauthCa
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def get_resource_from_auth_hash # rubocop:disable Naming/AccessorMethodName
 | 
			
		||||
    # find the user with their email instead of UID and token
 | 
			
		||||
    @resource = resource_class.where(
 | 
			
		||||
      email: auth_hash['info']['email']
 | 
			
		||||
    ).first
 | 
			
		||||
    email = auth_hash.dig('info', 'email')
 | 
			
		||||
    @resource = resource_class.from_email(email)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def validate_signup_email_is_business_domain?
 | 
			
		||||
@@ -75,3 +73,5 @@ class DeviseOverrides::OmniauthCallbacksController < DeviseTokenAuth::OmniauthCa
 | 
			
		||||
    'user'
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
DeviseOverrides::OmniauthCallbacksController.prepend_mod_with('DeviseOverrides::OmniauthCallbacksController')
 | 
			
		||||
 
 | 
			
		||||
@@ -44,3 +44,5 @@ class DeviseOverrides::PasswordsController < Devise::PasswordsController
 | 
			
		||||
    }, status: status
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 | 
			
		||||
DeviseOverrides::PasswordsController.prepend_mod_with('DeviseOverrides::PasswordsController')
 | 
			
		||||
 
 | 
			
		||||
@@ -2,7 +2,7 @@ class MicrosoftController < ApplicationController
 | 
			
		||||
  after_action :set_version_header
 | 
			
		||||
 | 
			
		||||
  def identity_association
 | 
			
		||||
    microsoft_indentity
 | 
			
		||||
    microsoft_identity
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
@@ -11,7 +11,7 @@ class MicrosoftController < ApplicationController
 | 
			
		||||
    response.headers['Content-Length'] = { associatedApplications: [{ applicationId: @identity_json }] }.to_json.length
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def microsoft_indentity
 | 
			
		||||
  def microsoft_identity
 | 
			
		||||
    @identity_json = GlobalConfigService.load('AZURE_APP_ID', nil)
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										36
									
								
								app/controllers/notion/callbacks_controller.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								app/controllers/notion/callbacks_controller.rb
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,36 @@
 | 
			
		||||
class Notion::CallbacksController < OauthCallbackController
 | 
			
		||||
  include NotionConcern
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def provider_name
 | 
			
		||||
    'notion'
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def oauth_client
 | 
			
		||||
    notion_client
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def handle_response
 | 
			
		||||
    hook = account.hooks.new(
 | 
			
		||||
      access_token: parsed_body['access_token'],
 | 
			
		||||
      status: 'enabled',
 | 
			
		||||
      app_id: 'notion',
 | 
			
		||||
      settings: {
 | 
			
		||||
        token_type: parsed_body['token_type'],
 | 
			
		||||
        workspace_name: parsed_body['workspace_name'],
 | 
			
		||||
        workspace_id: parsed_body['workspace_id'],
 | 
			
		||||
        workspace_icon: parsed_body['workspace_icon'],
 | 
			
		||||
        bot_id: parsed_body['bot_id'],
 | 
			
		||||
        owner: parsed_body['owner']
 | 
			
		||||
      }
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    hook.save!
 | 
			
		||||
    redirect_to notion_redirect_uri
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def notion_redirect_uri
 | 
			
		||||
    "#{ENV.fetch('FRONTEND_URL', nil)}/app/accounts/#{account.id}/settings/integrations/notion"
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -6,7 +6,6 @@ class OauthCallbackController < ApplicationController
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    handle_response
 | 
			
		||||
    ::Redis::Alfred.delete(cache_key)
 | 
			
		||||
  rescue StandardError => e
 | 
			
		||||
    ChatwootExceptionTracker.new(e).capture_exception
 | 
			
		||||
    redirect_to '/'
 | 
			
		||||
@@ -64,13 +63,10 @@ class OauthCallbackController < ApplicationController
 | 
			
		||||
    raise NotImplementedError
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def cache_key
 | 
			
		||||
    "#{provider_name}::#{users_data['email'].downcase}"
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def create_channel_with_inbox
 | 
			
		||||
    ActiveRecord::Base.transaction do
 | 
			
		||||
      channel_email = Channel::Email.create!(email: users_data['email'], account: account)
 | 
			
		||||
 | 
			
		||||
      account.inboxes.create!(
 | 
			
		||||
        account: account,
 | 
			
		||||
        channel: channel_email,
 | 
			
		||||
@@ -85,12 +81,17 @@ class OauthCallbackController < ApplicationController
 | 
			
		||||
    decoded_token[0]
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def account_id
 | 
			
		||||
    ::Redis::Alfred.get(cache_key)
 | 
			
		||||
  def account_from_signed_id
 | 
			
		||||
    raise ActionController::BadRequest, 'Missing state variable' if params[:state].blank?
 | 
			
		||||
 | 
			
		||||
    account = GlobalID::Locator.locate_signed(params[:state])
 | 
			
		||||
    raise 'Invalid or expired state' if account.nil?
 | 
			
		||||
 | 
			
		||||
    account
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def account
 | 
			
		||||
    @account ||= Account.find(account_id)
 | 
			
		||||
    @account ||= account_from_signed_id
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  # Fallback name, for when name field is missing from users_data
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,11 @@
 | 
			
		||||
class Platform::Api::V1::AccountsController < PlatformController
 | 
			
		||||
  def index
 | 
			
		||||
    @resources = @platform_app.platform_app_permissibles
 | 
			
		||||
                              .where(permissible_type: 'Account')
 | 
			
		||||
                              .includes(:permissible)
 | 
			
		||||
                              .map(&:permissible)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def show; end
 | 
			
		||||
 | 
			
		||||
  def create
 | 
			
		||||
 
 | 
			
		||||
@@ -7,13 +7,19 @@ class Public::Api::V1::Portals::ArticlesController < Public::Api::V1::Portals::B
 | 
			
		||||
 | 
			
		||||
  def index
 | 
			
		||||
    @articles = @portal.articles.published.includes(:category, :author)
 | 
			
		||||
 | 
			
		||||
    @articles = @articles.where(locale: permitted_params[:locale]) if permitted_params[:locale].present?
 | 
			
		||||
 | 
			
		||||
    @articles_count = @articles.count
 | 
			
		||||
 | 
			
		||||
    search_articles
 | 
			
		||||
    order_by_sort_param
 | 
			
		||||
    limit_results
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def show; end
 | 
			
		||||
  def show
 | 
			
		||||
    @og_image_url = helpers.set_og_image_url(@portal.name, @article.title)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def tracking_pixel
 | 
			
		||||
    @article = @portal.articles.find_by(slug: permitted_params[:article_slug])
 | 
			
		||||
 
 | 
			
		||||
@@ -58,6 +58,6 @@ class Public::Api::V1::Portals::BaseController < PublicController
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def set_global_config
 | 
			
		||||
    @global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'BRAND_URL')
 | 
			
		||||
    @global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'BRAND_URL', 'INSTALLATION_NAME')
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -8,7 +8,9 @@ class Public::Api::V1::Portals::CategoriesController < Public::Api::V1::Portals:
 | 
			
		||||
    @categories = @portal.categories.order(position: :asc)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def show; end
 | 
			
		||||
  def show
 | 
			
		||||
    @og_image_url = helpers.set_og_image_url(@portal.name, @category.name)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -4,7 +4,9 @@ class Public::Api::V1::PortalsController < Public::Api::V1::Portals::BaseControl
 | 
			
		||||
  before_action :redirect_to_portal_with_locale, only: [:show]
 | 
			
		||||
  layout 'portal'
 | 
			
		||||
 | 
			
		||||
  def show; end
 | 
			
		||||
  def show
 | 
			
		||||
    @og_image_url = helpers.set_og_image_url('', @portal.header_text)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def sitemap
 | 
			
		||||
    @help_center_url = @portal.custom_domain || ChatwootApp.help_center_root
 | 
			
		||||
 
 | 
			
		||||
@@ -17,7 +17,12 @@ class SlackUploadsController < ApplicationController
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def blob_url
 | 
			
		||||
    # Only generate representations for images
 | 
			
		||||
    if @blob.content_type.start_with?('image/')
 | 
			
		||||
      url_for(@blob.representation(resize_to_fill: [250, nil]))
 | 
			
		||||
    else
 | 
			
		||||
      url_for(@blob)
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def avatar_url
 | 
			
		||||
 
 | 
			
		||||
@@ -2,6 +2,13 @@ class SuperAdmin::AccountUsersController < SuperAdmin::ApplicationController
 | 
			
		||||
  # Overwrite any of the RESTful controller actions to implement custom behavior
 | 
			
		||||
  # For example, you may want to send an email after a foo is updated.
 | 
			
		||||
  #
 | 
			
		||||
 | 
			
		||||
  # Since account/user page - account user role attribute links to the show page
 | 
			
		||||
  # Handle with a redirect to the user show page
 | 
			
		||||
  def show
 | 
			
		||||
    redirect_to super_admin_user_path(requested_resource.user)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def create
 | 
			
		||||
    resource = resource_class.new(resource_params)
 | 
			
		||||
    authorize_resource(resource)
 | 
			
		||||
 
 | 
			
		||||
@@ -39,7 +39,10 @@ class SuperAdmin::AppConfigsController < SuperAdmin::ApplicationController
 | 
			
		||||
      'email' => ['MAILER_INBOUND_EMAIL_DOMAIN'],
 | 
			
		||||
      'linear' => %w[LINEAR_CLIENT_ID LINEAR_CLIENT_SECRET],
 | 
			
		||||
      'slack' => %w[SLACK_CLIENT_ID SLACK_CLIENT_SECRET],
 | 
			
		||||
      'instagram' => %w[INSTAGRAM_APP_ID INSTAGRAM_APP_SECRET INSTAGRAM_VERIFY_TOKEN INSTAGRAM_API_VERSION ENABLE_INSTAGRAM_CHANNEL_HUMAN_AGENT]
 | 
			
		||||
      'instagram' => %w[INSTAGRAM_APP_ID INSTAGRAM_APP_SECRET INSTAGRAM_VERIFY_TOKEN INSTAGRAM_API_VERSION ENABLE_INSTAGRAM_CHANNEL_HUMAN_AGENT],
 | 
			
		||||
      'whatsapp_embedded' => %w[WHATSAPP_APP_ID WHATSAPP_APP_SECRET WHATSAPP_CONFIGURATION_ID WHATSAPP_API_VERSION],
 | 
			
		||||
      'notion' => %w[NOTION_CLIENT_ID NOTION_CLIENT_SECRET],
 | 
			
		||||
      'google' => %w[GOOGLE_OAUTH_CLIENT_ID GOOGLE_OAUTH_CLIENT_SECRET GOOGLE_OAUTH_REDIRECT_URI]
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @allowed_configs = mapping.fetch(@config, %w[ENABLE_ACCOUNT_SIGNUP FIREBASE_PROJECT_ID FIREBASE_CREDENTIALS])
 | 
			
		||||
 
 | 
			
		||||
@@ -7,8 +7,9 @@
 | 
			
		||||
class SuperAdmin::ApplicationController < Administrate::ApplicationController
 | 
			
		||||
  include ActionView::Helpers::TagHelper
 | 
			
		||||
  include ActionView::Context
 | 
			
		||||
  include SuperAdmin::NavigationHelper
 | 
			
		||||
 | 
			
		||||
  helper_method :render_vue_component
 | 
			
		||||
  helper_method :render_vue_component, :settings_open?, :settings_pages
 | 
			
		||||
  # authenticiation done via devise : SuperAdmin Model
 | 
			
		||||
  before_action :authenticate_super_admin!
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -5,6 +5,6 @@ class Survey::ResponsesController < ActionController::Base
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def set_global_config
 | 
			
		||||
    @global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL')
 | 
			
		||||
    @global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL', 'INSTALLATION_NAME')
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -27,7 +27,11 @@ class Twilio::CallbackController < ApplicationController
 | 
			
		||||
      *Array.new(10) { |i| :"MediaUrl#{i}" },
 | 
			
		||||
      *Array.new(10) { |i| :"MediaContentType#{i}" },
 | 
			
		||||
      :MessagingServiceSid,
 | 
			
		||||
      :NumMedia
 | 
			
		||||
      :NumMedia,
 | 
			
		||||
      :Latitude,
 | 
			
		||||
      :Longitude,
 | 
			
		||||
      :MessageType,
 | 
			
		||||
      :ProfileName
 | 
			
		||||
    )
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
@@ -4,7 +4,16 @@ class Webhooks::InstagramController < ActionController::API
 | 
			
		||||
  def events
 | 
			
		||||
    Rails.logger.info('Instagram webhook received events')
 | 
			
		||||
    if params['object'].casecmp('instagram').zero?
 | 
			
		||||
      ::Webhooks::InstagramEventsJob.perform_later(params.to_unsafe_hash[:entry])
 | 
			
		||||
      entry_params = params.to_unsafe_hash[:entry]
 | 
			
		||||
 | 
			
		||||
      if contains_echo_event?(entry_params)
 | 
			
		||||
        # Add delay to prevent race condition where echo arrives before send message API completes
 | 
			
		||||
        # This avoids duplicate messages when echo comes early during API processing
 | 
			
		||||
        ::Webhooks::InstagramEventsJob.set(wait: 2.seconds).perform_later(entry_params)
 | 
			
		||||
      else
 | 
			
		||||
        ::Webhooks::InstagramEventsJob.perform_later(entry_params)
 | 
			
		||||
      end
 | 
			
		||||
 | 
			
		||||
      render json: :ok
 | 
			
		||||
    else
 | 
			
		||||
      Rails.logger.warn("Message is not received from the instagram webhook event: #{params['object']}")
 | 
			
		||||
@@ -14,6 +23,16 @@ class Webhooks::InstagramController < ActionController::API
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def contains_echo_event?(entry_params)
 | 
			
		||||
    return false unless entry_params.is_a?(Array)
 | 
			
		||||
 | 
			
		||||
    entry_params.any? do |entry|
 | 
			
		||||
      # Check messaging array for echo events
 | 
			
		||||
      messaging_events = entry[:messaging] || []
 | 
			
		||||
      messaging_events.any? { |messaging| messaging.dig(:message, :is_echo).present? }
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def valid_token?(token)
 | 
			
		||||
    # Validates against both IG_VERIFY_TOKEN (Instagram channel via Facebook page) and
 | 
			
		||||
    # INSTAGRAM_VERIFY_TOKEN (Instagram channel via direct Instagram login)
 | 
			
		||||
 
 | 
			
		||||
@@ -15,7 +15,7 @@ class WidgetsController < ActionController::Base
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def set_global_config
 | 
			
		||||
    @global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL', 'DIRECT_UPLOADS_ENABLED')
 | 
			
		||||
    @global_config = GlobalConfig.get('LOGO_THUMBNAIL', 'BRAND_NAME', 'WIDGET_BRAND_URL', 'DIRECT_UPLOADS_ENABLED', 'INSTALLATION_NAME')
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def set_web_widget
 | 
			
		||||
 
 | 
			
		||||
@@ -15,7 +15,13 @@ class NotificationFinder
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def unread_count
 | 
			
		||||
    if type_included?('read')
 | 
			
		||||
      # If we're including read notifications, filter to unread
 | 
			
		||||
      @notifications.where(read_at: nil).count
 | 
			
		||||
    else
 | 
			
		||||
      # Already filtered to unread notifications, just count
 | 
			
		||||
      @notifications.count
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def count
 | 
			
		||||
@@ -27,7 +33,7 @@ class NotificationFinder
 | 
			
		||||
  def set_up
 | 
			
		||||
    find_all_notifications
 | 
			
		||||
    filter_snoozed_notifications
 | 
			
		||||
    fitler_read_notifications
 | 
			
		||||
    filter_read_notifications
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def find_all_notifications
 | 
			
		||||
@@ -38,7 +44,7 @@ class NotificationFinder
 | 
			
		||||
    @notifications = @notifications.where(snoozed_until: nil) unless type_included?('snoozed')
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def fitler_read_notifications
 | 
			
		||||
  def filter_read_notifications
 | 
			
		||||
    @notifications = @notifications.where(read_at: nil) unless type_included?('read')
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -36,9 +36,13 @@ module Api::V2::Accounts::ReportsHelper
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def generate_labels_report
 | 
			
		||||
    Current.account.labels.map do |label|
 | 
			
		||||
      label_report = report_builder({ type: :label, id: label.id }).short_summary
 | 
			
		||||
      [label.title] + generate_readable_report_metrics(label_report)
 | 
			
		||||
    reports = V2::Reports::LabelSummaryBuilder.new(
 | 
			
		||||
      account: Current.account,
 | 
			
		||||
      params: build_params({})
 | 
			
		||||
    ).build
 | 
			
		||||
 | 
			
		||||
    reports.map do |report|
 | 
			
		||||
      [report[:name]] + generate_readable_report_metrics(report)
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,13 @@
 | 
			
		||||
module MessageFormatHelper
 | 
			
		||||
  include RegexHelper
 | 
			
		||||
 | 
			
		||||
  def transform_user_mention_content(message_content)
 | 
			
		||||
    # attachment message without content, message_content is nil
 | 
			
		||||
    message_content.presence ? message_content.gsub(MENTION_REGEX, '\1') : ''
 | 
			
		||||
    return '' unless message_content.presence
 | 
			
		||||
 | 
			
		||||
    # Use CommonMarker to convert markdown to plain text for notifications
 | 
			
		||||
    # This handles all markdown formatting (links, bold, italic, etc.) not just mentions
 | 
			
		||||
    # Converts: [@👍 customer support](mention://team/1/%F0%9F%91%8D%20customer%20support)
 | 
			
		||||
    # To: @👍 customer support
 | 
			
		||||
    CommonMarker.render_doc(message_content).to_plaintext.strip
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def render_message_content(message_content)
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,22 @@
 | 
			
		||||
module PortalHelper
 | 
			
		||||
  include UrlHelper
 | 
			
		||||
  def set_og_image_url(portal_name, title)
 | 
			
		||||
    cdn_url = GlobalConfig.get('OG_IMAGE_CDN_URL')['OG_IMAGE_CDN_URL']
 | 
			
		||||
    return if cdn_url.blank?
 | 
			
		||||
 | 
			
		||||
    client_ref = GlobalConfig.get('OG_IMAGE_CLIENT_REF')['OG_IMAGE_CLIENT_REF']
 | 
			
		||||
 | 
			
		||||
    uri = URI.parse(cdn_url)
 | 
			
		||||
    uri.path = '/og'
 | 
			
		||||
    uri.query = URI.encode_www_form(
 | 
			
		||||
      clientRef: client_ref,
 | 
			
		||||
      title: title,
 | 
			
		||||
      portalName: portal_name
 | 
			
		||||
    )
 | 
			
		||||
 | 
			
		||||
    uri.to_s
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def generate_portal_bg_color(portal_color, theme)
 | 
			
		||||
    base_color = theme == 'dark' ? 'black' : 'white'
 | 
			
		||||
    "color-mix(in srgb, #{portal_color} 20%, #{base_color})"
 | 
			
		||||
@@ -57,6 +75,17 @@ module PortalHelper
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def generate_portal_brand_url(brand_url, referer)
 | 
			
		||||
    url = URI.parse(brand_url.to_s)
 | 
			
		||||
    query_params = Rack::Utils.parse_query(url.query)
 | 
			
		||||
    query_params['utm_medium'] = 'helpcenter'
 | 
			
		||||
    query_params['utm_campaign'] = 'branding'
 | 
			
		||||
    query_params['utm_source'] = URI.parse(referer).host if url_valid?(referer)
 | 
			
		||||
 | 
			
		||||
    url.query = query_params.to_query
 | 
			
		||||
    url.to_s
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def render_category_content(content)
 | 
			
		||||
    ChatwootMarkdownRenderer.new(content).render_markdown_to_plain_text
 | 
			
		||||
  end
 | 
			
		||||
 
 | 
			
		||||
@@ -53,13 +53,13 @@ module ReportHelper
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def resolutions
 | 
			
		||||
    scope.reporting_events.joins(:conversation).select(:conversation_id).where(account_id: account.id, name: :conversation_resolved,
 | 
			
		||||
                                                                               conversations: { status: :resolved }, created_at: range).distinct
 | 
			
		||||
    scope.reporting_events.where(account_id: account.id, name: :conversation_resolved,
 | 
			
		||||
                                 created_at: range)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def bot_resolutions
 | 
			
		||||
    scope.reporting_events.joins(:conversation).select(:conversation_id).where(account_id: account.id, name: :conversation_bot_resolved,
 | 
			
		||||
                                                                               conversations: { status: :resolved }, created_at: range).distinct
 | 
			
		||||
    scope.reporting_events.where(account_id: account.id, name: :conversation_bot_resolved,
 | 
			
		||||
                                 created_at: range)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def bot_handoffs
 | 
			
		||||
 
 | 
			
		||||
@@ -18,12 +18,25 @@ module ReportingEventHelper
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def last_non_human_activity(conversation)
 | 
			
		||||
    # check if a handoff event already exists
 | 
			
		||||
    handoff_event = ReportingEvent.where(conversation_id: conversation.id, name: 'conversation_bot_handoff').last
 | 
			
		||||
    # Try to get either a handoff or reopened event first
 | 
			
		||||
    # These will always take precedence over any other activity
 | 
			
		||||
    # Also, any of these events can happen at any time in the course of a conversation lifecycle.
 | 
			
		||||
    # So we pick the latest event
 | 
			
		||||
    event = ReportingEvent.where(
 | 
			
		||||
      conversation_id: conversation.id,
 | 
			
		||||
      name: %w[conversation_bot_handoff conversation_opened]
 | 
			
		||||
    ).order(event_end_time: :desc).first
 | 
			
		||||
 | 
			
		||||
    # if a handoff exists, last non human activity is when the handoff ended,
 | 
			
		||||
    # otherwise it's when the conversation was created
 | 
			
		||||
    handoff_event&.event_end_time || conversation.created_at
 | 
			
		||||
    return event.event_end_time if event&.event_end_time
 | 
			
		||||
 | 
			
		||||
    # Fallback to bot resolved event
 | 
			
		||||
    # Because this will be closest to the most accurate activity instead of conversation.created_at
 | 
			
		||||
    bot_event = ReportingEvent.where(conversation_id: conversation.id, name: 'conversation_bot_resolved').last
 | 
			
		||||
 | 
			
		||||
    return bot_event.event_end_time if bot_event&.event_end_time
 | 
			
		||||
 | 
			
		||||
    # If no events found, return conversation creation time
 | 
			
		||||
    conversation.created_at
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,7 @@
 | 
			
		||||
# TODO: Move this values to features.yml itself
 | 
			
		||||
# No need to replicate the same values in two places
 | 
			
		||||
 | 
			
		||||
# ------- Premium Features ------- #
 | 
			
		||||
captain:
 | 
			
		||||
  name: 'Captain'
 | 
			
		||||
  description: 'Enable AI-powered conversations with your customers.'
 | 
			
		||||
@@ -32,6 +34,15 @@ disable_branding:
 | 
			
		||||
  enabled: <%= (ChatwootHub.pricing_plan != 'community') %>
 | 
			
		||||
  icon: 'icon-sailbot-fill'
 | 
			
		||||
  enterprise: true
 | 
			
		||||
 | 
			
		||||
# ------- Product Features ------- #
 | 
			
		||||
help_center:
 | 
			
		||||
  name: 'Help Center'
 | 
			
		||||
  description: 'Allow agents to create help center articles and publish them in a portal.'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-book-2-line'
 | 
			
		||||
 | 
			
		||||
# ------- Communication Channels ------- #
 | 
			
		||||
live_chat:
 | 
			
		||||
  name: 'Live Chat'
 | 
			
		||||
  description: 'Improve your customer experience using a live chat on your website.'
 | 
			
		||||
@@ -42,6 +53,12 @@ email:
 | 
			
		||||
  description: 'Manage your email customer interactions from Chatwoot.'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-mail-send-fill'
 | 
			
		||||
  config_key: 'email'
 | 
			
		||||
sms:
 | 
			
		||||
  name: 'SMS'
 | 
			
		||||
  description: 'Manage your SMS customer interactions from Chatwoot.'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-message-line'
 | 
			
		||||
messenger:
 | 
			
		||||
  name: 'Messenger'
 | 
			
		||||
  description: 'Stay connected with your customers on Facebook & Instagram.'
 | 
			
		||||
@@ -69,34 +86,46 @@ line:
 | 
			
		||||
  description: 'Manage your Line customer interactions from Chatwoot.'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-line-line'
 | 
			
		||||
sms:
 | 
			
		||||
  name: 'SMS'
 | 
			
		||||
  description: 'Manage your SMS customer interactions from Chatwoot.'
 | 
			
		||||
 | 
			
		||||
# ------- OAuth & Authentication ------- #
 | 
			
		||||
google:
 | 
			
		||||
  name: 'Google'
 | 
			
		||||
  description: 'Configuration for setting up Google OAuth Integration'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-message-line'
 | 
			
		||||
help_center:
 | 
			
		||||
  name: 'Help Center'
 | 
			
		||||
  description: 'Allow agents to create help center articles and publish them in a portal.'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-book-2-line'
 | 
			
		||||
  icon: 'icon-google'
 | 
			
		||||
  config_key: 'google'
 | 
			
		||||
microsoft:
 | 
			
		||||
  name: 'Microsoft'
 | 
			
		||||
  description: 'Configuration for setting up Microsoft Email'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-microsoft'
 | 
			
		||||
  config_key: 'microsoft'
 | 
			
		||||
 | 
			
		||||
# ------- Third-party Integrations ------- #
 | 
			
		||||
linear:
 | 
			
		||||
  name: 'Linear'
 | 
			
		||||
  description: 'Configuration for setting up Linear Integration'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-linear'
 | 
			
		||||
  config_key: 'linear'
 | 
			
		||||
notion:
 | 
			
		||||
  name: 'Notion'
 | 
			
		||||
  description: 'Configuration for setting up Notion Integration'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-notion'
 | 
			
		||||
  config_key: 'notion'
 | 
			
		||||
slack:
 | 
			
		||||
  name: 'Slack'
 | 
			
		||||
  description: 'Configuration for setting up Slack Integration'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-slack'
 | 
			
		||||
  config_key: 'slack'
 | 
			
		||||
whatsapp_embedded:
 | 
			
		||||
  name: 'WhatsApp Embedded'
 | 
			
		||||
  description: 'Configuration for setting up WhatsApp Embedded Integration'
 | 
			
		||||
  enabled: true
 | 
			
		||||
  icon: 'icon-whatsapp-line'
 | 
			
		||||
  config_key: 'whatsapp_embedded'
 | 
			
		||||
shopify:
 | 
			
		||||
  name: 'Shopify'
 | 
			
		||||
  description: 'Configuration for setting up Shopify Integration'
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
module SuperAdmin::FeaturesHelper
 | 
			
		||||
  def self.available_features
 | 
			
		||||
    YAML.load(ERB.new(Rails.root.join('enterprise/app/helpers/super_admin/features.yml').read).result).with_indifferent_access
 | 
			
		||||
    YAML.load(ERB.new(Rails.root.join('app/helpers/super_admin/features.yml').read).result).with_indifferent_access
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def self.plan_details
 | 
			
		||||
							
								
								
									
										16
									
								
								app/helpers/super_admin/navigation_helper.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								app/helpers/super_admin/navigation_helper.rb
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,16 @@
 | 
			
		||||
module SuperAdmin::NavigationHelper
 | 
			
		||||
  def settings_open?
 | 
			
		||||
    params[:controller].in? %w[super_admin/settings super_admin/app_configs]
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def settings_pages
 | 
			
		||||
    features = SuperAdmin::FeaturesHelper.available_features.select do |_feature, attrs|
 | 
			
		||||
      attrs['config_key'].present? && attrs['enabled']
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    # Add general at the beginning
 | 
			
		||||
    general_feature = [['general', { 'config_key' => 'general', 'name' => 'General' }]]
 | 
			
		||||
 | 
			
		||||
    general_feature + features.to_a
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
<script>
 | 
			
		||||
import { mapGetters } from 'vuex';
 | 
			
		||||
import AddAccountModal from '../dashboard/components/layout/sidebarComponents/AddAccountModal.vue';
 | 
			
		||||
import AddAccountModal from './components/app/AddAccountModal.vue';
 | 
			
		||||
import LoadingState from './components/widgets/LoadingState.vue';
 | 
			
		||||
import NetworkNotification from './components/NetworkNotification.vue';
 | 
			
		||||
import UpdateBanner from './components/app/UpdateBanner.vue';
 | 
			
		||||
@@ -19,6 +19,7 @@ import {
 | 
			
		||||
  verifyServiceWorkerExistence,
 | 
			
		||||
} from './helper/pushHelper';
 | 
			
		||||
import ReconnectService from 'dashboard/helper/ReconnectService';
 | 
			
		||||
import { useUISettings } from 'dashboard/composables/useUISettings';
 | 
			
		||||
 | 
			
		||||
export default {
 | 
			
		||||
  name: 'App',
 | 
			
		||||
@@ -38,12 +39,14 @@ export default {
 | 
			
		||||
    const { accountId } = useAccount();
 | 
			
		||||
    // Use the font size composable (it automatically sets up the watcher)
 | 
			
		||||
    const { currentFontSize } = useFontSize();
 | 
			
		||||
    const { uiSettings } = useUISettings();
 | 
			
		||||
 | 
			
		||||
    return {
 | 
			
		||||
      router,
 | 
			
		||||
      store,
 | 
			
		||||
      currentAccountId: accountId,
 | 
			
		||||
      currentFontSize,
 | 
			
		||||
      uiSettings,
 | 
			
		||||
    };
 | 
			
		||||
  },
 | 
			
		||||
  data() {
 | 
			
		||||
@@ -88,7 +91,10 @@ export default {
 | 
			
		||||
  mounted() {
 | 
			
		||||
    this.initializeColorTheme();
 | 
			
		||||
    this.listenToThemeChanges();
 | 
			
		||||
    this.setLocale(window.chatwootConfig.selectedLocale);
 | 
			
		||||
    // If user locale is set, use it; otherwise use account locale
 | 
			
		||||
    this.setLocale(
 | 
			
		||||
      this.uiSettings?.locale || window.chatwootConfig.selectedLocale
 | 
			
		||||
    );
 | 
			
		||||
  },
 | 
			
		||||
  unmounted() {
 | 
			
		||||
    if (this.reconnectService) {
 | 
			
		||||
@@ -114,7 +120,8 @@ export default {
 | 
			
		||||
      const { locale, latest_chatwoot_version: latestChatwootVersion } =
 | 
			
		||||
        this.getAccount(this.currentAccountId);
 | 
			
		||||
      const { pubsub_token: pubsubToken } = this.currentUser || {};
 | 
			
		||||
      this.setLocale(locale);
 | 
			
		||||
      // If user locale is set, use it; otherwise use account locale
 | 
			
		||||
      this.setLocale(this.uiSettings?.locale || locale);
 | 
			
		||||
      this.latestChatwootVersion = latestChatwootVersion;
 | 
			
		||||
      vueActionCable.init(this.store, pubsubToken);
 | 
			
		||||
      this.reconnectService = new ReconnectService(this.store, this.router);
 | 
			
		||||
@@ -136,8 +143,7 @@ export default {
 | 
			
		||||
  <div
 | 
			
		||||
    v-if="!authUIFlags.isFetching && !accountUIFlags.isFetchingItem"
 | 
			
		||||
    id="app"
 | 
			
		||||
    class="flex-grow-0 w-full h-full min-h-0 app-wrapper"
 | 
			
		||||
    :class="{ 'app-rtl--wrapper': isRTL }"
 | 
			
		||||
    class="flex flex-col w-full h-screen min-h-0"
 | 
			
		||||
    :dir="isRTL ? 'rtl' : 'ltr'"
 | 
			
		||||
  >
 | 
			
		||||
    <UpdateBanner :latest-chatwoot-version="latestChatwootVersion" />
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										43
									
								
								app/javascript/dashboard/api/agentCapacityPolicies.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								app/javascript/dashboard/api/agentCapacityPolicies.js
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,43 @@
 | 
			
		||||
/* global axios */
 | 
			
		||||
 | 
			
		||||
import ApiClient from './ApiClient';
 | 
			
		||||
 | 
			
		||||
class AgentCapacityPolicies extends ApiClient {
 | 
			
		||||
  constructor() {
 | 
			
		||||
    super('agent_capacity_policies', { accountScoped: true });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  getUsers(policyId) {
 | 
			
		||||
    return axios.get(`${this.url}/${policyId}/users`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  addUser(policyId, userData) {
 | 
			
		||||
    return axios.post(`${this.url}/${policyId}/users`, {
 | 
			
		||||
      user_id: userData.id,
 | 
			
		||||
      capacity: userData.capacity,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  removeUser(policyId, userId) {
 | 
			
		||||
    return axios.delete(`${this.url}/${policyId}/users/${userId}`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  createInboxLimit(policyId, limitData) {
 | 
			
		||||
    return axios.post(`${this.url}/${policyId}/inbox_limits`, {
 | 
			
		||||
      inbox_id: limitData.inboxId,
 | 
			
		||||
      conversation_limit: limitData.conversationLimit,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  updateInboxLimit(policyId, limitId, limitData) {
 | 
			
		||||
    return axios.put(`${this.url}/${policyId}/inbox_limits/${limitId}`, {
 | 
			
		||||
      conversation_limit: limitData.conversationLimit,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  deleteInboxLimit(policyId, limitId) {
 | 
			
		||||
    return axios.delete(`${this.url}/${policyId}/inbox_limits/${limitId}`);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default new AgentCapacityPolicies();
 | 
			
		||||
							
								
								
									
										36
									
								
								app/javascript/dashboard/api/assignmentPolicies.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								app/javascript/dashboard/api/assignmentPolicies.js
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,36 @@
 | 
			
		||||
/* global axios */
 | 
			
		||||
 | 
			
		||||
import ApiClient from './ApiClient';
 | 
			
		||||
 | 
			
		||||
class AssignmentPolicies extends ApiClient {
 | 
			
		||||
  constructor() {
 | 
			
		||||
    super('assignment_policies', { accountScoped: true });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  getInboxes(policyId) {
 | 
			
		||||
    return axios.get(`${this.url}/${policyId}/inboxes`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  setInboxPolicy(inboxId, policyId) {
 | 
			
		||||
    return axios.post(
 | 
			
		||||
      `/api/v1/accounts/${this.accountIdFromRoute}/inboxes/${inboxId}/assignment_policy`,
 | 
			
		||||
      {
 | 
			
		||||
        assignment_policy_id: policyId,
 | 
			
		||||
      }
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  getInboxPolicy(inboxId) {
 | 
			
		||||
    return axios.get(
 | 
			
		||||
      `/api/v1/accounts/${this.accountIdFromRoute}/inboxes/${inboxId}/assignment_policy`
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  removeInboxPolicy(inboxId) {
 | 
			
		||||
    return axios.delete(
 | 
			
		||||
      `/api/v1/accounts/${this.accountIdFromRoute}/inboxes/${inboxId}/assignment_policy`
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default new AssignmentPolicies();
 | 
			
		||||
@@ -38,13 +38,7 @@ export default {
 | 
			
		||||
    }
 | 
			
		||||
    return false;
 | 
			
		||||
  },
 | 
			
		||||
  profileUpdate({
 | 
			
		||||
    password,
 | 
			
		||||
    password_confirmation,
 | 
			
		||||
    displayName,
 | 
			
		||||
    avatar,
 | 
			
		||||
    ...profileAttributes
 | 
			
		||||
  }) {
 | 
			
		||||
  profileUpdate({ displayName, avatar, ...profileAttributes }) {
 | 
			
		||||
    const formData = new FormData();
 | 
			
		||||
    Object.keys(profileAttributes).forEach(key => {
 | 
			
		||||
      const hasValue = profileAttributes[key] === undefined;
 | 
			
		||||
@@ -53,16 +47,22 @@ export default {
 | 
			
		||||
      }
 | 
			
		||||
    });
 | 
			
		||||
    formData.append('profile[display_name]', displayName || '');
 | 
			
		||||
    if (password && password_confirmation) {
 | 
			
		||||
      formData.append('profile[password]', password);
 | 
			
		||||
      formData.append('profile[password_confirmation]', password_confirmation);
 | 
			
		||||
    }
 | 
			
		||||
    if (avatar) {
 | 
			
		||||
      formData.append('profile[avatar]', avatar);
 | 
			
		||||
    }
 | 
			
		||||
    return axios.put(endPoints('profileUpdate').url, formData);
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  profilePasswordUpdate({ currentPassword, password, passwordConfirmation }) {
 | 
			
		||||
    return axios.put(endPoints('profileUpdate').url, {
 | 
			
		||||
      profile: {
 | 
			
		||||
        current_password: currentPassword,
 | 
			
		||||
        password,
 | 
			
		||||
        password_confirmation: passwordConfirmation,
 | 
			
		||||
      },
 | 
			
		||||
    });
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  updateUISettings({ uiSettings }) {
 | 
			
		||||
    return axios.put(endPoints('profileUpdate').url, {
 | 
			
		||||
      profile: { ui_settings: uiSettings },
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										36
									
								
								app/javascript/dashboard/api/captain/scenarios.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								app/javascript/dashboard/api/captain/scenarios.js
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,36 @@
 | 
			
		||||
/* global axios */
 | 
			
		||||
import ApiClient from '../ApiClient';
 | 
			
		||||
 | 
			
		||||
class CaptainScenarios extends ApiClient {
 | 
			
		||||
  constructor() {
 | 
			
		||||
    super('captain/assistants', { accountScoped: true });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  get({ assistantId, page = 1, searchKey } = {}) {
 | 
			
		||||
    return axios.get(`${this.url}/${assistantId}/scenarios`, {
 | 
			
		||||
      params: { page, searchKey },
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  show({ assistantId, id }) {
 | 
			
		||||
    return axios.get(`${this.url}/${assistantId}/scenarios/${id}`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  create({ assistantId, ...data } = {}) {
 | 
			
		||||
    return axios.post(`${this.url}/${assistantId}/scenarios`, {
 | 
			
		||||
      scenario: data,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  update({ assistantId, id }, data = {}) {
 | 
			
		||||
    return axios.put(`${this.url}/${assistantId}/scenarios/${id}`, {
 | 
			
		||||
      scenario: data,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  delete({ assistantId, id }) {
 | 
			
		||||
    return axios.delete(`${this.url}/${assistantId}/scenarios/${id}`);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default new CaptainScenarios();
 | 
			
		||||
							
								
								
									
										16
									
								
								app/javascript/dashboard/api/captain/tools.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								app/javascript/dashboard/api/captain/tools.js
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,16 @@
 | 
			
		||||
/* global axios */
 | 
			
		||||
import ApiClient from '../ApiClient';
 | 
			
		||||
 | 
			
		||||
class CaptainTools extends ApiClient {
 | 
			
		||||
  constructor() {
 | 
			
		||||
    super('captain/assistants/tools', { accountScoped: true });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  get(params = {}) {
 | 
			
		||||
    return axios.get(this.url, {
 | 
			
		||||
      params,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default new CaptainTools();
 | 
			
		||||
							
								
								
									
										21
									
								
								app/javascript/dashboard/api/channel/whatsappChannel.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								app/javascript/dashboard/api/channel/whatsappChannel.js
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,21 @@
 | 
			
		||||
/* global axios */
 | 
			
		||||
import ApiClient from '../ApiClient';
 | 
			
		||||
 | 
			
		||||
class WhatsappChannel extends ApiClient {
 | 
			
		||||
  constructor() {
 | 
			
		||||
    super('whatsapp', { accountScoped: true });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  createEmbeddedSignup(params) {
 | 
			
		||||
    return axios.post(`${this.baseUrl()}/whatsapp/authorization`, params);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  reauthorizeWhatsApp({ inboxId, ...params }) {
 | 
			
		||||
    return axios.post(`${this.baseUrl()}/whatsapp/authorization`, {
 | 
			
		||||
      ...params,
 | 
			
		||||
      inbox_id: inboxId,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default new WhatsappChannel();
 | 
			
		||||
@@ -61,6 +61,11 @@ class ContactAPI extends ApiClient {
 | 
			
		||||
    return axios.get(requestURL);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  active(page = 1, sortAttr = 'name') {
 | 
			
		||||
    let requestURL = `${this.url}/active?${buildContactParams(page, sortAttr)}`;
 | 
			
		||||
    return axios.get(requestURL);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // eslint-disable-next-line default-param-last
 | 
			
		||||
  filter(page = 1, sortAttr = 'name', queryPayload) {
 | 
			
		||||
    let requestURL = `${this.url}/filter?${buildContactParams(page, sortAttr)}`;
 | 
			
		||||
 
 | 
			
		||||
@@ -51,6 +51,7 @@ const endPoints = {
 | 
			
		||||
  resendConfirmation: {
 | 
			
		||||
    url: '/api/v1/profile/resend_confirmation',
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  resetAccessToken: {
 | 
			
		||||
    url: '/api/v1/profile/reset_access_token',
 | 
			
		||||
  },
 | 
			
		||||
 
 | 
			
		||||
@@ -21,6 +21,14 @@ class PortalsAPI extends ApiClient {
 | 
			
		||||
  deleteLogo(portalSlug) {
 | 
			
		||||
    return axios.delete(`${this.url}/${portalSlug}/logo`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  sendCnameInstructions(portalSlug, email) {
 | 
			
		||||
    return axios.post(`${this.url}/${portalSlug}/send_instructions`, { email });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  sslStatus(portalSlug) {
 | 
			
		||||
    return axios.get(`${this.url}/${portalSlug}/ssl_status`);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default PortalsAPI;
 | 
			
		||||
 
 | 
			
		||||
@@ -134,13 +134,13 @@ class ConversationApi extends ApiClient {
 | 
			
		||||
    return axios.get(`${this.url}/${conversationId}/attachments`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  requestCopilot(conversationId, body) {
 | 
			
		||||
    return axios.post(`${this.url}/${conversationId}/copilot`, body);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  getInboxAssistant(conversationId) {
 | 
			
		||||
    return axios.get(`${this.url}/${conversationId}/inbox_assistant`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  delete(conversationId) {
 | 
			
		||||
    return axios.delete(`${this.url}/${conversationId}`);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default new ConversationApi();
 | 
			
		||||
 
 | 
			
		||||
@@ -28,6 +28,10 @@ class Inboxes extends CacheEnabledApiClient {
 | 
			
		||||
      agent_bot: botId,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  syncTemplates(inboxId) {
 | 
			
		||||
    return axios.post(`${this.url}/${inboxId}/sync_templates`);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default new Inboxes();
 | 
			
		||||
 
 | 
			
		||||
@@ -33,9 +33,11 @@ class LinearAPI extends ApiClient {
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  unlinkIssue(linkId) {
 | 
			
		||||
  unlinkIssue(linkId, issueIdentifier, conversationId) {
 | 
			
		||||
    return axios.post(`${this.url}/unlink_issue`, {
 | 
			
		||||
      link_id: linkId,
 | 
			
		||||
      issue_id: issueIdentifier,
 | 
			
		||||
      conversation_id: conversationId,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										14
									
								
								app/javascript/dashboard/api/notion_auth.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								app/javascript/dashboard/api/notion_auth.js
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,14 @@
 | 
			
		||||
/* global axios */
 | 
			
		||||
import ApiClient from './ApiClient';
 | 
			
		||||
 | 
			
		||||
class NotionOAuthClient extends ApiClient {
 | 
			
		||||
  constructor() {
 | 
			
		||||
    super('notion', { accountScoped: true });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  generateAuthorization() {
 | 
			
		||||
    return axios.post(`${this.url}/authorization`);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default new NotionOAuthClient();
 | 
			
		||||
							
								
								
									
										26
									
								
								app/javascript/dashboard/api/samlSettings.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								app/javascript/dashboard/api/samlSettings.js
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,26 @@
 | 
			
		||||
/* global axios */
 | 
			
		||||
import ApiClient from './ApiClient';
 | 
			
		||||
 | 
			
		||||
class SamlSettingsAPI extends ApiClient {
 | 
			
		||||
  constructor() {
 | 
			
		||||
    super('saml_settings', { accountScoped: true });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  get() {
 | 
			
		||||
    return axios.get(this.url);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  create(data) {
 | 
			
		||||
    return axios.post(this.url, { saml_settings: data });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  update(data) {
 | 
			
		||||
    return axios.put(this.url, { saml_settings: data });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  delete() {
 | 
			
		||||
    return axios.delete(this.url);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default new SamlSettingsAPI();
 | 
			
		||||
@@ -0,0 +1,98 @@
 | 
			
		||||
import agentCapacityPolicies from '../agentCapacityPolicies';
 | 
			
		||||
import ApiClient from '../ApiClient';
 | 
			
		||||
 | 
			
		||||
describe('#AgentCapacityPoliciesAPI', () => {
 | 
			
		||||
  it('creates correct instance', () => {
 | 
			
		||||
    expect(agentCapacityPolicies).toBeInstanceOf(ApiClient);
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('get');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('show');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('create');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('update');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('delete');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('getUsers');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('addUser');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('removeUser');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('createInboxLimit');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('updateInboxLimit');
 | 
			
		||||
    expect(agentCapacityPolicies).toHaveProperty('deleteInboxLimit');
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('API calls', () => {
 | 
			
		||||
    const originalAxios = window.axios;
 | 
			
		||||
    const axiosMock = {
 | 
			
		||||
      get: vi.fn(() => Promise.resolve()),
 | 
			
		||||
      post: vi.fn(() => Promise.resolve()),
 | 
			
		||||
      put: vi.fn(() => Promise.resolve()),
 | 
			
		||||
      delete: vi.fn(() => Promise.resolve()),
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    beforeEach(() => {
 | 
			
		||||
      window.axios = axiosMock;
 | 
			
		||||
      // Mock accountIdFromRoute
 | 
			
		||||
      Object.defineProperty(agentCapacityPolicies, 'accountIdFromRoute', {
 | 
			
		||||
        get: () => '1',
 | 
			
		||||
        configurable: true,
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    afterEach(() => {
 | 
			
		||||
      window.axios = originalAxios;
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#getUsers', () => {
 | 
			
		||||
      agentCapacityPolicies.getUsers(123);
 | 
			
		||||
      expect(axiosMock.get).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/agent_capacity_policies/123/users'
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#addUser', () => {
 | 
			
		||||
      const userData = { id: 456, capacity: 20 };
 | 
			
		||||
      agentCapacityPolicies.addUser(123, userData);
 | 
			
		||||
      expect(axiosMock.post).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/agent_capacity_policies/123/users',
 | 
			
		||||
        {
 | 
			
		||||
          user_id: 456,
 | 
			
		||||
          capacity: 20,
 | 
			
		||||
        }
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#removeUser', () => {
 | 
			
		||||
      agentCapacityPolicies.removeUser(123, 456);
 | 
			
		||||
      expect(axiosMock.delete).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/agent_capacity_policies/123/users/456'
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#createInboxLimit', () => {
 | 
			
		||||
      const limitData = { inboxId: 1, conversationLimit: 10 };
 | 
			
		||||
      agentCapacityPolicies.createInboxLimit(123, limitData);
 | 
			
		||||
      expect(axiosMock.post).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/agent_capacity_policies/123/inbox_limits',
 | 
			
		||||
        {
 | 
			
		||||
          inbox_id: 1,
 | 
			
		||||
          conversation_limit: 10,
 | 
			
		||||
        }
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#updateInboxLimit', () => {
 | 
			
		||||
      const limitData = { conversationLimit: 15 };
 | 
			
		||||
      agentCapacityPolicies.updateInboxLimit(123, 789, limitData);
 | 
			
		||||
      expect(axiosMock.put).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/agent_capacity_policies/123/inbox_limits/789',
 | 
			
		||||
        {
 | 
			
		||||
          conversation_limit: 15,
 | 
			
		||||
        }
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#deleteInboxLimit', () => {
 | 
			
		||||
      agentCapacityPolicies.deleteInboxLimit(123, 789);
 | 
			
		||||
      expect(axiosMock.delete).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/agent_capacity_policies/123/inbox_limits/789'
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
@@ -0,0 +1,70 @@
 | 
			
		||||
import assignmentPolicies from '../assignmentPolicies';
 | 
			
		||||
import ApiClient from '../ApiClient';
 | 
			
		||||
 | 
			
		||||
describe('#AssignmentPoliciesAPI', () => {
 | 
			
		||||
  it('creates correct instance', () => {
 | 
			
		||||
    expect(assignmentPolicies).toBeInstanceOf(ApiClient);
 | 
			
		||||
    expect(assignmentPolicies).toHaveProperty('get');
 | 
			
		||||
    expect(assignmentPolicies).toHaveProperty('show');
 | 
			
		||||
    expect(assignmentPolicies).toHaveProperty('create');
 | 
			
		||||
    expect(assignmentPolicies).toHaveProperty('update');
 | 
			
		||||
    expect(assignmentPolicies).toHaveProperty('delete');
 | 
			
		||||
    expect(assignmentPolicies).toHaveProperty('getInboxes');
 | 
			
		||||
    expect(assignmentPolicies).toHaveProperty('setInboxPolicy');
 | 
			
		||||
    expect(assignmentPolicies).toHaveProperty('getInboxPolicy');
 | 
			
		||||
    expect(assignmentPolicies).toHaveProperty('removeInboxPolicy');
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('API calls', () => {
 | 
			
		||||
    const originalAxios = window.axios;
 | 
			
		||||
    const axiosMock = {
 | 
			
		||||
      get: vi.fn(() => Promise.resolve()),
 | 
			
		||||
      post: vi.fn(() => Promise.resolve()),
 | 
			
		||||
      delete: vi.fn(() => Promise.resolve()),
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    beforeEach(() => {
 | 
			
		||||
      window.axios = axiosMock;
 | 
			
		||||
      // Mock accountIdFromRoute
 | 
			
		||||
      Object.defineProperty(assignmentPolicies, 'accountIdFromRoute', {
 | 
			
		||||
        get: () => '1',
 | 
			
		||||
        configurable: true,
 | 
			
		||||
      });
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    afterEach(() => {
 | 
			
		||||
      window.axios = originalAxios;
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#getInboxes', () => {
 | 
			
		||||
      assignmentPolicies.getInboxes(123);
 | 
			
		||||
      expect(axiosMock.get).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/assignment_policies/123/inboxes'
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#setInboxPolicy', () => {
 | 
			
		||||
      assignmentPolicies.setInboxPolicy(456, 123);
 | 
			
		||||
      expect(axiosMock.post).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/inboxes/456/assignment_policy',
 | 
			
		||||
        {
 | 
			
		||||
          assignment_policy_id: 123,
 | 
			
		||||
        }
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#getInboxPolicy', () => {
 | 
			
		||||
      assignmentPolicies.getInboxPolicy(456);
 | 
			
		||||
      expect(axiosMock.get).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/inboxes/456/assignment_policy'
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#removeInboxPolicy', () => {
 | 
			
		||||
      assignmentPolicies.removeInboxPolicy(456);
 | 
			
		||||
      expect(axiosMock.delete).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/accounts/1/inboxes/456/assignment_policy'
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
@@ -12,6 +12,7 @@ describe('#InboxesAPI', () => {
 | 
			
		||||
    expect(inboxesAPI).toHaveProperty('getCampaigns');
 | 
			
		||||
    expect(inboxesAPI).toHaveProperty('getAgentBot');
 | 
			
		||||
    expect(inboxesAPI).toHaveProperty('setAgentBot');
 | 
			
		||||
    expect(inboxesAPI).toHaveProperty('syncTemplates');
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  describe('API calls', () => {
 | 
			
		||||
@@ -40,5 +41,12 @@ describe('#InboxesAPI', () => {
 | 
			
		||||
      inboxesAPI.deleteInboxAvatar(2);
 | 
			
		||||
      expect(axiosMock.delete).toHaveBeenCalledWith('/api/v1/inboxes/2/avatar');
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    it('#syncTemplates', () => {
 | 
			
		||||
      inboxesAPI.syncTemplates(2);
 | 
			
		||||
      expect(axiosMock.post).toHaveBeenCalledWith(
 | 
			
		||||
        '/api/v1/inboxes/2/sync_templates'
 | 
			
		||||
      );
 | 
			
		||||
    });
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
 
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user