mirror of
https://github.com/lingble/chatwoot.git
synced 2025-11-20 13:05:16 +00:00
Merge branch 'develop' into pr/12259
This commit is contained in:
@@ -1,6 +1,7 @@
|
|||||||
version: 2.1
|
version: 2.1
|
||||||
orbs:
|
orbs:
|
||||||
node: circleci/node@6.1.0
|
node: circleci/node@6.1.0
|
||||||
|
qlty-orb: qltysh/qlty-orb@0.0
|
||||||
|
|
||||||
defaults: &defaults
|
defaults: &defaults
|
||||||
working_directory: ~/build
|
working_directory: ~/build
|
||||||
@@ -89,14 +90,6 @@ jobs:
|
|||||||
command: |
|
command: |
|
||||||
source ~/.rvm/scripts/rvm
|
source ~/.rvm/scripts/rvm
|
||||||
bundle install
|
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
|
# Swagger verification
|
||||||
- run:
|
- 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'."
|
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
|
exit 1
|
||||||
fi
|
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
|
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
|
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:
|
- run:
|
||||||
name: Database Setup and Configure Environment Variables
|
name: Database Setup and Configure Environment Variables
|
||||||
command: |
|
command: |
|
||||||
@@ -149,17 +143,11 @@ jobs:
|
|||||||
command: pnpm run eslint
|
command: pnpm run eslint
|
||||||
|
|
||||||
- run:
|
- run:
|
||||||
name: Run frontend tests
|
name: Run frontend tests (with coverage)
|
||||||
command: |
|
command: |
|
||||||
mkdir -p ~/build/coverage/frontend
|
mkdir -p ~/build/coverage/frontend
|
||||||
~/tmp/cc-test-reporter before-build
|
|
||||||
pnpm run test:coverage
|
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 backend tests
|
||||||
- run:
|
- run:
|
||||||
name: Run backend tests
|
name: Run backend tests
|
||||||
@@ -167,18 +155,18 @@ jobs:
|
|||||||
mkdir -p ~/tmp/test-results/rspec
|
mkdir -p ~/tmp/test-results/rspec
|
||||||
mkdir -p ~/tmp/test-artifacts
|
mkdir -p ~/tmp/test-artifacts
|
||||||
mkdir -p ~/build/coverage/backend
|
mkdir -p ~/build/coverage/backend
|
||||||
~/tmp/cc-test-reporter before-build
|
|
||||||
TESTFILES=$(circleci tests glob "spec/**/*_spec.rb" | circleci tests split --split-by=timings)
|
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 \
|
--format RspecJunitFormatter \
|
||||||
--out ~/tmp/test-results/rspec.xml \
|
--out ~/tmp/test-results/rspec.xml \
|
||||||
-- ${TESTFILES}
|
-- ${TESTFILES}
|
||||||
no_output_timeout: 30m
|
no_output_timeout: 30m
|
||||||
|
|
||||||
- run:
|
# Qlty coverage publish
|
||||||
name: Code Climate Test Coverage (Backend)
|
- qlty-orb/coverage_publish:
|
||||||
command: |
|
files: |
|
||||||
~/tmp/cc-test-reporter format-coverage -t simplecov -o "~/build/coverage/backend/codeclimate.$CIRCLE_NODE_INDEX.json"
|
coverage/coverage.json
|
||||||
|
coverage/lcov.info
|
||||||
|
|
||||||
- run:
|
- run:
|
||||||
name: List coverage directory contents
|
name: List coverage directory contents
|
||||||
@@ -189,3 +177,7 @@ jobs:
|
|||||||
root: ~/build
|
root: ~/build
|
||||||
paths:
|
paths:
|
||||||
- coverage
|
- 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
|
WORKDIR /workspace
|
||||||
RUN chown vscode:vscode /workspace
|
RUN chown vscode:vscode /workspace
|
||||||
|
|
||||||
# set up node js and pnpm in single layer
|
# set up node js, pnpm and claude code in single layer
|
||||||
RUN npm install -g pnpm@${PNPM_VERSION} \
|
RUN npm install -g pnpm@${PNPM_VERSION} @anthropic-ai/claude-code \
|
||||||
&& npm cache clean --force
|
&& npm cache clean --force
|
||||||
|
|
||||||
# Switch to vscode user
|
# Switch to vscode user
|
||||||
|
|||||||
@@ -4,17 +4,26 @@
|
|||||||
"dockerComposeFile": "docker-compose.yml",
|
"dockerComposeFile": "docker-compose.yml",
|
||||||
|
|
||||||
"settings": {
|
"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.
|
// Add the IDs of extensions you want installed when the container is created.
|
||||||
"extensions": [
|
"extensions": [
|
||||||
"rebornix.Ruby",
|
"Shopify.ruby-lsp",
|
||||||
"misogi.ruby-rubocop",
|
"misogi.ruby-rubocop",
|
||||||
"wingrunr21.vscode-ruby",
|
|
||||||
"davidpallinder.rails-test-runner",
|
"davidpallinder.rails-test-runner",
|
||||||
"eamodio.gitlens",
|
|
||||||
"github.copilot",
|
"github.copilot",
|
||||||
"mrmlnc.vscode-duplicate"
|
"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 '/SMTP_ADDRESS/ s/=.*/=localhost/' .env
|
||||||
sed -i -e "/FRONTEND_URL/ s/=.*/=https:\/\/$CODESPACE_NAME-3000.app.github.dev/" .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
|
# codespaces make the ports public
|
||||||
gh codespace ports visibility 3000:public 3036:public 8025:public -c $CODESPACE_NAME
|
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:
|
on:
|
||||||
pull_request:
|
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:
|
jobs:
|
||||||
deployment_check:
|
deployment_check:
|
||||||
name: Check Deployment
|
name: Check Deployment
|
||||||
|
|||||||
@@ -5,6 +5,11 @@ on:
|
|||||||
branches:
|
branches:
|
||||||
- develop
|
- 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:
|
jobs:
|
||||||
log_lines_check:
|
log_lines_check:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|||||||
5
.github/workflows/size-limit.yml
vendored
5
.github/workflows/size-limit.yml
vendored
@@ -5,6 +5,11 @@ on:
|
|||||||
branches:
|
branches:
|
||||||
- develop
|
- 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:
|
jobs:
|
||||||
test:
|
test:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
|
|||||||
5
.gitignore
vendored
5
.gitignore
vendored
@@ -94,3 +94,8 @@ yarn-debug.log*
|
|||||||
.vscode
|
.vscode
|
||||||
.claude/settings.local.json
|
.claude/settings.local.json
|
||||||
.cursor
|
.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
|
Enabled: false
|
||||||
|
|
||||||
Layout/TrailingEmptyLines:
|
Layout/TrailingEmptyLines:
|
||||||
Enabled: false
|
Enabled: true
|
||||||
|
|
||||||
Style/SafeNavigationChainLength:
|
Style/SafeNavigationChainLength:
|
||||||
Enabled: false
|
Enabled: false
|
||||||
|
|||||||
19
AGENTS.md
19
AGENTS.md
@@ -55,4 +55,21 @@
|
|||||||
|
|
||||||
## Ruby Best Practices
|
## Ruby Best Practices
|
||||||
|
|
||||||
- Use compact `module/class` definitions; avoid nested styles
|
- 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
|
# super fast record imports in bulk
|
||||||
gem 'activerecord-import'
|
gem 'activerecord-import'
|
||||||
|
|
||||||
|
gem 'searchkick'
|
||||||
|
gem 'opensearch-ruby'
|
||||||
|
gem 'faraday_middleware-aws-sigv4'
|
||||||
|
|
||||||
##--- gems for server & infra configuration ---##
|
##--- gems for server & infra configuration ---##
|
||||||
gem 'dotenv-rails', '>= 3.0.0'
|
gem 'dotenv-rails', '>= 3.0.0'
|
||||||
gem 'foreman'
|
gem 'foreman'
|
||||||
@@ -77,6 +81,7 @@ gem 'devise_token_auth', '>= 1.2.3'
|
|||||||
# authorization
|
# authorization
|
||||||
gem 'jwt'
|
gem 'jwt'
|
||||||
gem 'pundit'
|
gem 'pundit'
|
||||||
|
|
||||||
# super admin
|
# super admin
|
||||||
gem 'administrate', '>= 0.20.1'
|
gem 'administrate', '>= 0.20.1'
|
||||||
gem 'administrate-field-active_storage', '>= 1.0.3'
|
gem 'administrate-field-active_storage', '>= 1.0.3'
|
||||||
@@ -89,7 +94,7 @@ gem 'wisper', '2.0.0'
|
|||||||
##--- gems for channels ---##
|
##--- gems for channels ---##
|
||||||
gem 'facebook-messenger'
|
gem 'facebook-messenger'
|
||||||
gem 'line-bot-api'
|
gem 'line-bot-api'
|
||||||
gem 'twilio-ruby', '~> 5.66'
|
gem 'twilio-ruby'
|
||||||
# twitty will handle subscription of twitter account events
|
# twitty will handle subscription of twitter account events
|
||||||
# gem 'twitty', git: 'https://github.com/chatwoot/twitty'
|
# gem 'twitty', git: 'https://github.com/chatwoot/twitty'
|
||||||
gem 'twitty', '~> 0.1.5'
|
gem 'twitty', '~> 0.1.5'
|
||||||
@@ -108,7 +113,7 @@ gem 'google-cloud-translate-v3', '>= 0.7.0'
|
|||||||
##-- apm and error monitoring ---#
|
##-- apm and error monitoring ---#
|
||||||
# loaded only when environment variables are set.
|
# loaded only when environment variables are set.
|
||||||
# ref application.rb
|
# ref application.rb
|
||||||
gem 'ddtrace', require: false
|
gem 'datadog', '~> 2.0', require: false
|
||||||
gem 'elastic-apm', require: false
|
gem 'elastic-apm', require: false
|
||||||
gem 'newrelic_rpm', require: false
|
gem 'newrelic_rpm', require: false
|
||||||
gem 'newrelic-sidekiq-metrics', '>= 1.6.2', 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'
|
gem 'sidekiq', '>= 7.3.1'
|
||||||
# We want cron jobs
|
# We want cron jobs
|
||||||
gem 'sidekiq-cron', '>= 1.12.0'
|
gem 'sidekiq-cron', '>= 1.12.0'
|
||||||
|
# for sidekiq healthcheck
|
||||||
|
gem 'sidekiq_alive'
|
||||||
|
|
||||||
##-- Push notification service --##
|
##-- Push notification service --##
|
||||||
gem 'fcm'
|
gem 'fcm'
|
||||||
@@ -165,6 +172,7 @@ gem 'audited', '~> 5.4', '>= 5.4.1'
|
|||||||
|
|
||||||
# need for google auth
|
# need for google auth
|
||||||
gem 'omniauth', '>= 2.1.2'
|
gem 'omniauth', '>= 2.1.2'
|
||||||
|
gem 'omniauth-saml'
|
||||||
gem 'omniauth-google-oauth2', '>= 1.1.3'
|
gem 'omniauth-google-oauth2', '>= 1.1.3'
|
||||||
gem 'omniauth-rails_csrf_protection', '~> 1.0', '>= 1.0.2'
|
gem 'omniauth-rails_csrf_protection', '~> 1.0', '>= 1.0.2'
|
||||||
|
|
||||||
@@ -177,6 +185,10 @@ gem 'reverse_markdown'
|
|||||||
|
|
||||||
gem 'iso-639'
|
gem 'iso-639'
|
||||||
gem 'ruby-openai'
|
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'
|
gem 'shopify_api'
|
||||||
|
|
||||||
@@ -206,6 +218,8 @@ group :development do
|
|||||||
gem 'stackprof'
|
gem 'stackprof'
|
||||||
# Should install the associated chrome extension to view query logs
|
# Should install the associated chrome extension to view query logs
|
||||||
gem 'meta_request', '>= 0.8.3'
|
gem 'meta_request', '>= 0.8.3'
|
||||||
|
|
||||||
|
gem 'tidewave'
|
||||||
end
|
end
|
||||||
|
|
||||||
group :test do
|
group :test do
|
||||||
@@ -215,6 +229,7 @@ group :test do
|
|||||||
gem 'webmock'
|
gem 'webmock'
|
||||||
# test profiling
|
# test profiling
|
||||||
gem 'test-prof'
|
gem 'test-prof'
|
||||||
|
gem 'simplecov_json_formatter', require: false
|
||||||
end
|
end
|
||||||
|
|
||||||
group :development, :test do
|
group :development, :test do
|
||||||
@@ -239,7 +254,7 @@ group :development, :test do
|
|||||||
gem 'rubocop-factory_bot', require: false
|
gem 'rubocop-factory_bot', require: false
|
||||||
gem 'seed_dump'
|
gem 'seed_dump'
|
||||||
gem 'shoulda-matchers'
|
gem 'shoulda-matchers'
|
||||||
gem 'simplecov', '0.17.1', require: false
|
gem 'simplecov', '>= 0.21', require: false
|
||||||
gem 'spring'
|
gem 'spring'
|
||||||
gem 'spring-watcher-listen'
|
gem 'spring-watcher-listen'
|
||||||
end
|
end
|
||||||
|
|||||||
292
Gemfile.lock
292
Gemfile.lock
@@ -25,35 +25,35 @@ GIT
|
|||||||
GEM
|
GEM
|
||||||
remote: https://rubygems.org/
|
remote: https://rubygems.org/
|
||||||
specs:
|
specs:
|
||||||
actioncable (7.1.5.1)
|
actioncable (7.1.5.2)
|
||||||
actionpack (= 7.1.5.1)
|
actionpack (= 7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
nio4r (~> 2.0)
|
nio4r (~> 2.0)
|
||||||
websocket-driver (>= 0.6.1)
|
websocket-driver (>= 0.6.1)
|
||||||
zeitwerk (~> 2.6)
|
zeitwerk (~> 2.6)
|
||||||
actionmailbox (7.1.5.1)
|
actionmailbox (7.1.5.2)
|
||||||
actionpack (= 7.1.5.1)
|
actionpack (= 7.1.5.2)
|
||||||
activejob (= 7.1.5.1)
|
activejob (= 7.1.5.2)
|
||||||
activerecord (= 7.1.5.1)
|
activerecord (= 7.1.5.2)
|
||||||
activestorage (= 7.1.5.1)
|
activestorage (= 7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
mail (>= 2.7.1)
|
mail (>= 2.7.1)
|
||||||
net-imap
|
net-imap
|
||||||
net-pop
|
net-pop
|
||||||
net-smtp
|
net-smtp
|
||||||
actionmailer (7.1.5.1)
|
actionmailer (7.1.5.2)
|
||||||
actionpack (= 7.1.5.1)
|
actionpack (= 7.1.5.2)
|
||||||
actionview (= 7.1.5.1)
|
actionview (= 7.1.5.2)
|
||||||
activejob (= 7.1.5.1)
|
activejob (= 7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
mail (~> 2.5, >= 2.5.4)
|
mail (~> 2.5, >= 2.5.4)
|
||||||
net-imap
|
net-imap
|
||||||
net-pop
|
net-pop
|
||||||
net-smtp
|
net-smtp
|
||||||
rails-dom-testing (~> 2.2)
|
rails-dom-testing (~> 2.2)
|
||||||
actionpack (7.1.5.1)
|
actionpack (7.1.5.2)
|
||||||
actionview (= 7.1.5.1)
|
actionview (= 7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
nokogiri (>= 1.8.5)
|
nokogiri (>= 1.8.5)
|
||||||
racc
|
racc
|
||||||
rack (>= 2.2.4)
|
rack (>= 2.2.4)
|
||||||
@@ -61,38 +61,38 @@ GEM
|
|||||||
rack-test (>= 0.6.3)
|
rack-test (>= 0.6.3)
|
||||||
rails-dom-testing (~> 2.2)
|
rails-dom-testing (~> 2.2)
|
||||||
rails-html-sanitizer (~> 1.6)
|
rails-html-sanitizer (~> 1.6)
|
||||||
actiontext (7.1.5.1)
|
actiontext (7.1.5.2)
|
||||||
actionpack (= 7.1.5.1)
|
actionpack (= 7.1.5.2)
|
||||||
activerecord (= 7.1.5.1)
|
activerecord (= 7.1.5.2)
|
||||||
activestorage (= 7.1.5.1)
|
activestorage (= 7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
globalid (>= 0.6.0)
|
globalid (>= 0.6.0)
|
||||||
nokogiri (>= 1.8.5)
|
nokogiri (>= 1.8.5)
|
||||||
actionview (7.1.5.1)
|
actionview (7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
builder (~> 3.1)
|
builder (~> 3.1)
|
||||||
erubi (~> 1.11)
|
erubi (~> 1.11)
|
||||||
rails-dom-testing (~> 2.2)
|
rails-dom-testing (~> 2.2)
|
||||||
rails-html-sanitizer (~> 1.6)
|
rails-html-sanitizer (~> 1.6)
|
||||||
active_record_query_trace (1.8)
|
active_record_query_trace (1.8)
|
||||||
activejob (7.1.5.1)
|
activejob (7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
globalid (>= 0.3.6)
|
globalid (>= 0.3.6)
|
||||||
activemodel (7.1.5.1)
|
activemodel (7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
activerecord (7.1.5.1)
|
activerecord (7.1.5.2)
|
||||||
activemodel (= 7.1.5.1)
|
activemodel (= 7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
timeout (>= 0.4.0)
|
timeout (>= 0.4.0)
|
||||||
activerecord-import (2.1.0)
|
activerecord-import (2.1.0)
|
||||||
activerecord (>= 4.2)
|
activerecord (>= 4.2)
|
||||||
activestorage (7.1.5.1)
|
activestorage (7.1.5.2)
|
||||||
actionpack (= 7.1.5.1)
|
actionpack (= 7.1.5.2)
|
||||||
activejob (= 7.1.5.1)
|
activejob (= 7.1.5.2)
|
||||||
activerecord (= 7.1.5.1)
|
activerecord (= 7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
marcel (~> 1.0)
|
marcel (~> 1.0)
|
||||||
activesupport (7.1.5.1)
|
activesupport (7.1.5.2)
|
||||||
base64
|
base64
|
||||||
benchmark (>= 0.3)
|
benchmark (>= 0.3)
|
||||||
bigdecimal
|
bigdecimal
|
||||||
@@ -126,6 +126,8 @@ GEM
|
|||||||
jbuilder (~> 2)
|
jbuilder (~> 2)
|
||||||
rails (>= 4.2, < 7.2)
|
rails (>= 4.2, < 7.2)
|
||||||
selectize-rails (~> 0.6)
|
selectize-rails (~> 0.6)
|
||||||
|
ai-agents (0.4.3)
|
||||||
|
ruby_llm (~> 1.3)
|
||||||
annotate (3.2.0)
|
annotate (3.2.0)
|
||||||
activerecord (>= 3.2, < 8.0)
|
activerecord (>= 3.2, < 8.0)
|
||||||
rake (>= 10.4, < 14.0)
|
rake (>= 10.4, < 14.0)
|
||||||
@@ -153,10 +155,10 @@ GEM
|
|||||||
barnes (0.0.9)
|
barnes (0.0.9)
|
||||||
multi_json (~> 1)
|
multi_json (~> 1)
|
||||||
statsd-ruby (~> 1.1)
|
statsd-ruby (~> 1.1)
|
||||||
base64 (0.2.0)
|
base64 (0.3.0)
|
||||||
bcrypt (3.1.20)
|
bcrypt (3.1.20)
|
||||||
benchmark (0.4.0)
|
benchmark (0.4.1)
|
||||||
bigdecimal (3.1.9)
|
bigdecimal (3.2.2)
|
||||||
bindex (0.8.1)
|
bindex (0.8.1)
|
||||||
bootsnap (1.16.0)
|
bootsnap (1.16.0)
|
||||||
msgpack (~> 1.2)
|
msgpack (~> 1.2)
|
||||||
@@ -172,6 +174,8 @@ GEM
|
|||||||
bundler (>= 1.2.0, < 3)
|
bundler (>= 1.2.0, < 3)
|
||||||
thor (~> 1.0)
|
thor (~> 1.0)
|
||||||
byebug (11.1.3)
|
byebug (11.1.3)
|
||||||
|
childprocess (5.1.0)
|
||||||
|
logger (~> 1.5)
|
||||||
climate_control (1.2.0)
|
climate_control (1.2.0)
|
||||||
coderay (1.1.3)
|
coderay (1.1.3)
|
||||||
commonmarker (0.23.10)
|
commonmarker (0.23.10)
|
||||||
@@ -190,10 +194,14 @@ GEM
|
|||||||
activerecord (>= 5.a)
|
activerecord (>= 5.a)
|
||||||
database_cleaner-core (~> 2.0.0)
|
database_cleaner-core (~> 2.0.0)
|
||||||
database_cleaner-core (2.0.1)
|
database_cleaner-core (2.0.1)
|
||||||
date (3.4.1)
|
datadog (2.19.0)
|
||||||
ddtrace (0.48.0)
|
datadog-ruby_core_source (~> 3.4, >= 3.4.1)
|
||||||
ffi (~> 1.0)
|
libdatadog (~> 18.1.0.1.0)
|
||||||
|
libddwaf (~> 1.24.1.0.3)
|
||||||
|
logger
|
||||||
msgpack
|
msgpack
|
||||||
|
datadog-ruby_core_source (3.4.1)
|
||||||
|
date (3.4.1)
|
||||||
debug (1.8.0)
|
debug (1.8.0)
|
||||||
irb (>= 1.5.0)
|
irb (>= 1.5.0)
|
||||||
reline (>= 0.3.1)
|
reline (>= 0.3.1)
|
||||||
@@ -211,7 +219,7 @@ GEM
|
|||||||
diff-lcs (1.5.1)
|
diff-lcs (1.5.1)
|
||||||
digest-crc (0.6.5)
|
digest-crc (0.6.5)
|
||||||
rake (>= 12.0.0, < 14.0.0)
|
rake (>= 12.0.0, < 14.0.0)
|
||||||
docile (1.4.0)
|
docile (1.4.1)
|
||||||
domain_name (0.5.20190701)
|
domain_name (0.5.20190701)
|
||||||
unf (>= 0.0.5, < 1.0.0)
|
unf (>= 0.0.5, < 1.0.0)
|
||||||
dotenv (3.1.2)
|
dotenv (3.1.2)
|
||||||
@@ -222,6 +230,35 @@ GEM
|
|||||||
addressable (~> 2.8)
|
addressable (~> 2.8)
|
||||||
drb (2.2.3)
|
drb (2.2.3)
|
||||||
dry-cli (1.1.0)
|
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)
|
ecma-re-validator (0.4.0)
|
||||||
regexp_parser (~> 2.2)
|
regexp_parser (~> 2.2)
|
||||||
elastic-apm (4.6.2)
|
elastic-apm (4.6.2)
|
||||||
@@ -244,8 +281,10 @@ GEM
|
|||||||
railties (>= 5.0.0)
|
railties (>= 5.0.0)
|
||||||
faker (3.2.0)
|
faker (3.2.0)
|
||||||
i18n (>= 1.8.11, < 2)
|
i18n (>= 1.8.11, < 2)
|
||||||
faraday (2.9.0)
|
faraday (2.13.1)
|
||||||
faraday-net_http (>= 2.0, < 3.2)
|
faraday-net_http (>= 2.0, < 3.5)
|
||||||
|
json
|
||||||
|
logger
|
||||||
faraday-follow_redirects (0.3.0)
|
faraday-follow_redirects (0.3.0)
|
||||||
faraday (>= 1, < 3)
|
faraday (>= 1, < 3)
|
||||||
faraday-mashify (0.1.1)
|
faraday-mashify (0.1.1)
|
||||||
@@ -253,13 +292,23 @@ GEM
|
|||||||
hashie
|
hashie
|
||||||
faraday-multipart (1.0.4)
|
faraday-multipart (1.0.4)
|
||||||
multipart-post (~> 2)
|
multipart-post (~> 2)
|
||||||
faraday-net_http (3.1.0)
|
faraday-net_http (3.4.0)
|
||||||
net-http
|
net-http (>= 0.5.0)
|
||||||
faraday-net_http_persistent (2.1.0)
|
faraday-net_http_persistent (2.1.0)
|
||||||
faraday (~> 2.5)
|
faraday (~> 2.5)
|
||||||
net-http-persistent (~> 4.0)
|
net-http-persistent (~> 4.0)
|
||||||
faraday-retry (2.2.1)
|
faraday-retry (2.2.1)
|
||||||
faraday (~> 2.0)
|
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)
|
fcm (1.0.8)
|
||||||
faraday (>= 1.0.0, < 3.0)
|
faraday (>= 1.0.0, < 3.0)
|
||||||
googleauth (~> 1)
|
googleauth (~> 1)
|
||||||
@@ -355,6 +404,7 @@ GEM
|
|||||||
grpc (1.72.0-x86_64-linux)
|
grpc (1.72.0-x86_64-linux)
|
||||||
google-protobuf (>= 3.25, < 5.0)
|
google-protobuf (>= 3.25, < 5.0)
|
||||||
googleapis-common-protos-types (~> 1.0)
|
googleapis-common-protos-types (~> 1.0)
|
||||||
|
gserver (0.0.1)
|
||||||
haikunator (1.1.1)
|
haikunator (1.1.1)
|
||||||
hairtrigger (1.0.0)
|
hairtrigger (1.0.0)
|
||||||
activerecord (>= 6.0, < 8)
|
activerecord (>= 6.0, < 8)
|
||||||
@@ -397,7 +447,7 @@ GEM
|
|||||||
rails-dom-testing (>= 1, < 3)
|
rails-dom-testing (>= 1, < 3)
|
||||||
railties (>= 4.2.0)
|
railties (>= 4.2.0)
|
||||||
thor (>= 0.14, < 2.0)
|
thor (>= 0.14, < 2.0)
|
||||||
json (2.12.0)
|
json (2.13.2)
|
||||||
json_refs (0.1.8)
|
json_refs (0.1.8)
|
||||||
hana
|
hana
|
||||||
json_schemer (0.2.24)
|
json_schemer (0.2.24)
|
||||||
@@ -412,7 +462,7 @@ GEM
|
|||||||
judoscale-sidekiq (1.8.2)
|
judoscale-sidekiq (1.8.2)
|
||||||
judoscale-ruby (= 1.8.2)
|
judoscale-ruby (= 1.8.2)
|
||||||
sidekiq (>= 5.0)
|
sidekiq (>= 5.0)
|
||||||
jwt (2.8.1)
|
jwt (2.10.1)
|
||||||
base64
|
base64
|
||||||
kaminari (1.2.2)
|
kaminari (1.2.2)
|
||||||
activesupport (>= 4.1.0)
|
activesupport (>= 4.1.0)
|
||||||
@@ -433,10 +483,22 @@ GEM
|
|||||||
json (>= 1.8)
|
json (>= 1.8)
|
||||||
rexml
|
rexml
|
||||||
language_server-protocol (3.17.0.5)
|
language_server-protocol (3.17.0.5)
|
||||||
launchy (2.5.2)
|
launchy (3.1.1)
|
||||||
addressable (~> 2.8)
|
addressable (~> 2.8)
|
||||||
letter_opener (1.8.1)
|
childprocess (~> 5.0)
|
||||||
launchy (>= 2.2, < 3)
|
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)
|
line-bot-api (1.28.0)
|
||||||
lint_roller (1.1.0)
|
lint_roller (1.1.0)
|
||||||
liquid (5.4.0)
|
liquid (5.4.0)
|
||||||
@@ -471,7 +533,7 @@ GEM
|
|||||||
mime-types-data (3.2023.0218.1)
|
mime-types-data (3.2023.0218.1)
|
||||||
mini_magick (4.12.0)
|
mini_magick (4.12.0)
|
||||||
mini_mime (1.1.5)
|
mini_mime (1.1.5)
|
||||||
mini_portile2 (2.8.8)
|
mini_portile2 (2.8.9)
|
||||||
minitest (5.25.5)
|
minitest (5.25.5)
|
||||||
mock_redis (0.36.0)
|
mock_redis (0.36.0)
|
||||||
ruby2_keywords
|
ruby2_keywords
|
||||||
@@ -482,7 +544,7 @@ GEM
|
|||||||
mutex_m (0.3.0)
|
mutex_m (0.3.0)
|
||||||
neighbor (0.2.3)
|
neighbor (0.2.3)
|
||||||
activerecord (>= 5.2)
|
activerecord (>= 5.2)
|
||||||
net-http (0.4.1)
|
net-http (0.6.0)
|
||||||
uri
|
uri
|
||||||
net-http-persistent (4.0.2)
|
net-http-persistent (4.0.2)
|
||||||
connection_pool (~> 2.2)
|
connection_pool (~> 2.2)
|
||||||
@@ -502,14 +564,14 @@ GEM
|
|||||||
newrelic_rpm (9.6.0)
|
newrelic_rpm (9.6.0)
|
||||||
base64
|
base64
|
||||||
nio4r (2.7.3)
|
nio4r (2.7.3)
|
||||||
nokogiri (1.18.8)
|
nokogiri (1.18.9)
|
||||||
mini_portile2 (~> 2.8.2)
|
mini_portile2 (~> 2.8.2)
|
||||||
racc (~> 1.4)
|
racc (~> 1.4)
|
||||||
nokogiri (1.18.8-arm64-darwin)
|
nokogiri (1.18.9-arm64-darwin)
|
||||||
racc (~> 1.4)
|
racc (~> 1.4)
|
||||||
nokogiri (1.18.8-x86_64-darwin)
|
nokogiri (1.18.9-x86_64-darwin)
|
||||||
racc (~> 1.4)
|
racc (~> 1.4)
|
||||||
nokogiri (1.18.8-x86_64-linux-gnu)
|
nokogiri (1.18.9-x86_64-linux-gnu)
|
||||||
racc (~> 1.4)
|
racc (~> 1.4)
|
||||||
oauth (1.1.0)
|
oauth (1.1.0)
|
||||||
oauth-tty (~> 1.0, >= 1.0.1)
|
oauth-tty (~> 1.0, >= 1.0.1)
|
||||||
@@ -527,8 +589,9 @@ GEM
|
|||||||
oj (3.16.10)
|
oj (3.16.10)
|
||||||
bigdecimal (>= 3.0)
|
bigdecimal (>= 3.0)
|
||||||
ostruct (>= 0.2)
|
ostruct (>= 0.2)
|
||||||
omniauth (2.1.2)
|
omniauth (2.1.3)
|
||||||
hashie (>= 3.4.6)
|
hashie (>= 3.4.6)
|
||||||
|
logger
|
||||||
rack (>= 2.2.3)
|
rack (>= 2.2.3)
|
||||||
rack-protection
|
rack-protection
|
||||||
omniauth-google-oauth2 (1.1.3)
|
omniauth-google-oauth2 (1.1.3)
|
||||||
@@ -542,6 +605,12 @@ GEM
|
|||||||
omniauth-rails_csrf_protection (1.0.2)
|
omniauth-rails_csrf_protection (1.0.2)
|
||||||
actionpack (>= 4.2)
|
actionpack (>= 4.2)
|
||||||
omniauth (~> 2.0)
|
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)
|
openssl (3.2.0)
|
||||||
orm_adapter (0.5.0)
|
orm_adapter (0.5.0)
|
||||||
os (1.1.4)
|
os (1.1.4)
|
||||||
@@ -563,14 +632,14 @@ GEM
|
|||||||
method_source (~> 1.0)
|
method_source (~> 1.0)
|
||||||
pry-rails (0.3.9)
|
pry-rails (0.3.9)
|
||||||
pry (>= 0.10.4)
|
pry (>= 0.10.4)
|
||||||
public_suffix (6.0.0)
|
public_suffix (6.0.2)
|
||||||
puma (6.4.3)
|
puma (6.4.3)
|
||||||
nio4r (~> 2.0)
|
nio4r (~> 2.0)
|
||||||
pundit (2.3.0)
|
pundit (2.3.0)
|
||||||
activesupport (>= 3.0.0)
|
activesupport (>= 3.0.0)
|
||||||
raabro (1.4.0)
|
raabro (1.4.0)
|
||||||
racc (1.8.1)
|
racc (1.8.1)
|
||||||
rack (2.2.15)
|
rack (3.2.0)
|
||||||
rack-attack (6.7.0)
|
rack-attack (6.7.0)
|
||||||
rack (>= 1.0, < 4)
|
rack (>= 1.0, < 4)
|
||||||
rack-contrib (2.5.0)
|
rack-contrib (2.5.0)
|
||||||
@@ -579,33 +648,34 @@ GEM
|
|||||||
rack (>= 2.0.0)
|
rack (>= 2.0.0)
|
||||||
rack-mini-profiler (3.2.0)
|
rack-mini-profiler (3.2.0)
|
||||||
rack (>= 1.2.0)
|
rack (>= 1.2.0)
|
||||||
rack-protection (3.2.0)
|
rack-protection (4.1.1)
|
||||||
base64 (>= 0.1.0)
|
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-proxy (0.7.7)
|
||||||
rack
|
rack
|
||||||
rack-session (1.0.2)
|
rack-session (2.1.1)
|
||||||
rack (< 3)
|
base64 (>= 0.1.0)
|
||||||
|
rack (>= 3.0.0)
|
||||||
rack-test (2.1.0)
|
rack-test (2.1.0)
|
||||||
rack (>= 1.3)
|
rack (>= 1.3)
|
||||||
rack-timeout (0.6.3)
|
rack-timeout (0.6.3)
|
||||||
rackup (1.0.1)
|
rackup (2.2.1)
|
||||||
rack (< 3)
|
rack (>= 3)
|
||||||
webrick
|
rails (7.1.5.2)
|
||||||
rails (7.1.5.1)
|
actioncable (= 7.1.5.2)
|
||||||
actioncable (= 7.1.5.1)
|
actionmailbox (= 7.1.5.2)
|
||||||
actionmailbox (= 7.1.5.1)
|
actionmailer (= 7.1.5.2)
|
||||||
actionmailer (= 7.1.5.1)
|
actionpack (= 7.1.5.2)
|
||||||
actionpack (= 7.1.5.1)
|
actiontext (= 7.1.5.2)
|
||||||
actiontext (= 7.1.5.1)
|
actionview (= 7.1.5.2)
|
||||||
actionview (= 7.1.5.1)
|
activejob (= 7.1.5.2)
|
||||||
activejob (= 7.1.5.1)
|
activemodel (= 7.1.5.2)
|
||||||
activemodel (= 7.1.5.1)
|
activerecord (= 7.1.5.2)
|
||||||
activerecord (= 7.1.5.1)
|
activestorage (= 7.1.5.2)
|
||||||
activestorage (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
|
||||||
bundler (>= 1.15.0)
|
bundler (>= 1.15.0)
|
||||||
railties (= 7.1.5.1)
|
railties (= 7.1.5.2)
|
||||||
rails-dom-testing (2.2.0)
|
rails-dom-testing (2.2.0)
|
||||||
activesupport (>= 5.0.0)
|
activesupport (>= 5.0.0)
|
||||||
minitest
|
minitest
|
||||||
@@ -613,9 +683,9 @@ GEM
|
|||||||
rails-html-sanitizer (1.6.1)
|
rails-html-sanitizer (1.6.1)
|
||||||
loofah (~> 2.21)
|
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)
|
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)
|
railties (7.1.5.2)
|
||||||
actionpack (= 7.1.5.1)
|
actionpack (= 7.1.5.2)
|
||||||
activesupport (= 7.1.5.1)
|
activesupport (= 7.1.5.2)
|
||||||
irb
|
irb
|
||||||
rackup (>= 1.0.0)
|
rackup (>= 1.0.0)
|
||||||
rake (>= 12.2)
|
rake (>= 12.2)
|
||||||
@@ -707,12 +777,25 @@ GEM
|
|||||||
faraday (>= 1)
|
faraday (>= 1)
|
||||||
faraday-multipart (>= 1)
|
faraday-multipart (>= 1)
|
||||||
ruby-progressbar (1.13.0)
|
ruby-progressbar (1.13.0)
|
||||||
|
ruby-saml (1.18.1)
|
||||||
|
nokogiri (>= 1.13.10)
|
||||||
|
rexml
|
||||||
ruby-vips (2.1.4)
|
ruby-vips (2.1.4)
|
||||||
ffi (~> 1.12)
|
ffi (~> 1.12)
|
||||||
ruby2_keywords (0.0.5)
|
ruby2_keywords (0.0.5)
|
||||||
ruby2ruby (2.5.0)
|
ruby2ruby (2.5.0)
|
||||||
ruby_parser (~> 3.1)
|
ruby_parser (~> 3.1)
|
||||||
sexp_processor (~> 4.6)
|
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)
|
ruby_parser (3.20.0)
|
||||||
sexp_processor (~> 4.16)
|
sexp_processor (~> 4.16)
|
||||||
sass (3.7.4)
|
sass (3.7.4)
|
||||||
@@ -732,6 +815,9 @@ GEM
|
|||||||
parser
|
parser
|
||||||
scss_lint (0.60.0)
|
scss_lint (0.60.0)
|
||||||
sass (~> 3.5, >= 3.5.5)
|
sass (~> 3.5, >= 3.5.5)
|
||||||
|
searchkick (5.5.2)
|
||||||
|
activemodel (>= 7.1)
|
||||||
|
hashie
|
||||||
securerandom (0.4.1)
|
securerandom (0.4.1)
|
||||||
seed_dump (3.3.1)
|
seed_dump (3.3.1)
|
||||||
activerecord (>= 4)
|
activerecord (>= 4)
|
||||||
@@ -770,16 +856,20 @@ GEM
|
|||||||
fugit (~> 1.8)
|
fugit (~> 1.8)
|
||||||
globalid (>= 1.0.1)
|
globalid (>= 1.0.1)
|
||||||
sidekiq (>= 6)
|
sidekiq (>= 6)
|
||||||
|
sidekiq_alive (2.5.0)
|
||||||
|
gserver (~> 0.0.1)
|
||||||
|
sidekiq (>= 5, < 9)
|
||||||
signet (0.17.0)
|
signet (0.17.0)
|
||||||
addressable (~> 2.8)
|
addressable (~> 2.8)
|
||||||
faraday (>= 0.17.5, < 3.a)
|
faraday (>= 0.17.5, < 3.a)
|
||||||
jwt (>= 1.5, < 3.0)
|
jwt (>= 1.5, < 3.0)
|
||||||
multi_json (~> 1.10)
|
multi_json (~> 1.10)
|
||||||
simplecov (0.17.1)
|
simplecov (0.22.0)
|
||||||
docile (~> 1.1)
|
docile (~> 1.1)
|
||||||
json (>= 1.8, < 3)
|
simplecov-html (~> 0.11)
|
||||||
simplecov-html (~> 0.10.0)
|
simplecov_json_formatter (~> 0.1)
|
||||||
simplecov-html (0.10.2)
|
simplecov-html (0.13.2)
|
||||||
|
simplecov_json_formatter (0.1.4)
|
||||||
slack-ruby-client (2.5.2)
|
slack-ruby-client (2.5.2)
|
||||||
faraday (>= 2.0)
|
faraday (>= 2.0)
|
||||||
faraday-mashify
|
faraday-mashify
|
||||||
@@ -808,14 +898,18 @@ GEM
|
|||||||
stripe (8.5.0)
|
stripe (8.5.0)
|
||||||
telephone_number (1.4.20)
|
telephone_number (1.4.20)
|
||||||
test-prof (1.2.1)
|
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)
|
tilt (2.3.0)
|
||||||
time_diff (0.3.0)
|
time_diff (0.3.0)
|
||||||
activesupport
|
activesupport
|
||||||
i18n
|
i18n
|
||||||
timeout (0.4.3)
|
timeout (0.4.3)
|
||||||
trailblazer-option (0.1.2)
|
trailblazer-option (0.1.2)
|
||||||
twilio-ruby (5.77.0)
|
twilio-ruby (7.6.0)
|
||||||
faraday (>= 0.9, < 3.0)
|
faraday (>= 0.9, < 3.0)
|
||||||
jwt (>= 1.5, < 3.0)
|
jwt (>= 1.5, < 3.0)
|
||||||
nokogiri (>= 1.6, < 2.0)
|
nokogiri (>= 1.6, < 2.0)
|
||||||
@@ -862,7 +956,6 @@ GEM
|
|||||||
addressable (>= 2.8.0)
|
addressable (>= 2.8.0)
|
||||||
crack (>= 0.3.2)
|
crack (>= 0.3.2)
|
||||||
hashdiff (>= 0.4.0, < 2.0.0)
|
hashdiff (>= 0.4.0, < 2.0.0)
|
||||||
webrick (1.9.1)
|
|
||||||
websocket-driver (0.7.7)
|
websocket-driver (0.7.7)
|
||||||
base64
|
base64
|
||||||
websocket-extensions (>= 0.1.0)
|
websocket-extensions (>= 0.1.0)
|
||||||
@@ -891,6 +984,7 @@ DEPENDENCIES
|
|||||||
administrate (>= 0.20.1)
|
administrate (>= 0.20.1)
|
||||||
administrate-field-active_storage (>= 1.0.3)
|
administrate-field-active_storage (>= 1.0.3)
|
||||||
administrate-field-belongs_to_search (>= 0.9.0)
|
administrate-field-belongs_to_search (>= 0.9.0)
|
||||||
|
ai-agents (>= 0.4.3)
|
||||||
annotate
|
annotate
|
||||||
attr_extras
|
attr_extras
|
||||||
audited (~> 5.4, >= 5.4.1)
|
audited (~> 5.4, >= 5.4.1)
|
||||||
@@ -907,7 +1001,7 @@ DEPENDENCIES
|
|||||||
commonmarker
|
commonmarker
|
||||||
csv-safe
|
csv-safe
|
||||||
database_cleaner
|
database_cleaner
|
||||||
ddtrace
|
datadog (~> 2.0)
|
||||||
debug (~> 1.8)
|
debug (~> 1.8)
|
||||||
devise (>= 4.9.4)
|
devise (>= 4.9.4)
|
||||||
devise-secure_password!
|
devise-secure_password!
|
||||||
@@ -919,6 +1013,7 @@ DEPENDENCIES
|
|||||||
facebook-messenger
|
facebook-messenger
|
||||||
factory_bot_rails (>= 6.4.3)
|
factory_bot_rails (>= 6.4.3)
|
||||||
faker
|
faker
|
||||||
|
faraday_middleware-aws-sigv4
|
||||||
fcm
|
fcm
|
||||||
flag_shih_tzu
|
flag_shih_tzu
|
||||||
foreman
|
foreman
|
||||||
@@ -959,6 +1054,8 @@ DEPENDENCIES
|
|||||||
omniauth-google-oauth2 (>= 1.1.3)
|
omniauth-google-oauth2 (>= 1.1.3)
|
||||||
omniauth-oauth2
|
omniauth-oauth2
|
||||||
omniauth-rails_csrf_protection (~> 1.0, >= 1.0.2)
|
omniauth-rails_csrf_protection (~> 1.0, >= 1.0.2)
|
||||||
|
omniauth-saml
|
||||||
|
opensearch-ruby
|
||||||
pg
|
pg
|
||||||
pg_search
|
pg_search
|
||||||
pgvector
|
pgvector
|
||||||
@@ -984,8 +1081,10 @@ DEPENDENCIES
|
|||||||
rubocop-rails
|
rubocop-rails
|
||||||
rubocop-rspec
|
rubocop-rspec
|
||||||
ruby-openai
|
ruby-openai
|
||||||
|
ruby_llm-schema
|
||||||
scout_apm
|
scout_apm
|
||||||
scss_lint
|
scss_lint
|
||||||
|
searchkick
|
||||||
seed_dump
|
seed_dump
|
||||||
sentry-rails (>= 5.19.0)
|
sentry-rails (>= 5.19.0)
|
||||||
sentry-ruby
|
sentry-ruby
|
||||||
@@ -994,7 +1093,9 @@ DEPENDENCIES
|
|||||||
shoulda-matchers
|
shoulda-matchers
|
||||||
sidekiq (>= 7.3.1)
|
sidekiq (>= 7.3.1)
|
||||||
sidekiq-cron (>= 1.12.0)
|
sidekiq-cron (>= 1.12.0)
|
||||||
simplecov (= 0.17.1)
|
sidekiq_alive
|
||||||
|
simplecov (>= 0.21)
|
||||||
|
simplecov_json_formatter
|
||||||
slack-ruby-client (~> 2.5.2)
|
slack-ruby-client (~> 2.5.2)
|
||||||
spring
|
spring
|
||||||
spring-watcher-listen
|
spring-watcher-listen
|
||||||
@@ -1003,8 +1104,9 @@ DEPENDENCIES
|
|||||||
stripe
|
stripe
|
||||||
telephone_number
|
telephone_number
|
||||||
test-prof
|
test-prof
|
||||||
|
tidewave
|
||||||
time_diff
|
time_diff
|
||||||
twilio-ruby (~> 5.66)
|
twilio-ruby
|
||||||
twitty (~> 0.1.5)
|
twitty (~> 0.1.5)
|
||||||
tzinfo-data
|
tzinfo-data
|
||||||
uglifier
|
uglifier
|
||||||
|
|||||||
9
Makefile
9
Makefile
@@ -41,8 +41,15 @@ run:
|
|||||||
|
|
||||||
force_run:
|
force_run:
|
||||||
rm -f ./.overmind.sock
|
rm -f ./.overmind.sock
|
||||||
|
rm -f tmp/pids/*.pid
|
||||||
overmind start -f Procfile.dev
|
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:
|
debug:
|
||||||
overmind connect backend
|
overmind connect backend
|
||||||
|
|
||||||
@@ -52,4 +59,4 @@ debug_worker:
|
|||||||
docker:
|
docker:
|
||||||
docker build -t $(APP_NAME) -f ./docker/Dockerfile .
|
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.
|
# for example lib/tasks/capistrano.rake, and they will automatically be available to Rake.
|
||||||
|
|
||||||
require_relative 'config/application'
|
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
|
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.
|
# We don't want to update the name of the identified original contact.
|
||||||
|
|
||||||
class ContactIdentifyAction
|
class ContactIdentifyAction
|
||||||
|
include UrlHelper
|
||||||
pattr_initialize [:contact!, :params!, { retain_original_contact_name: false, discard_invalid_attrs: false }]
|
pattr_initialize [:contact!, :params!, { retain_original_contact_name: false, discard_invalid_attrs: false }]
|
||||||
|
|
||||||
def perform
|
def perform
|
||||||
@@ -104,7 +105,14 @@ class ContactIdentifyAction
|
|||||||
# TODO: replace reject { |_k, v| v.blank? } with compact_blank when rails is upgraded
|
# TODO: replace reject { |_k, v| v.blank? } with compact_blank when rails is upgraded
|
||||||
@contact.discard_invalid_attrs if discard_invalid_attrs
|
@contact.discard_invalid_attrs if discard_invalid_attrs
|
||||||
@contact.save!
|
@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
|
end
|
||||||
|
|
||||||
def merge_contact(base_contact, merge_contact)
|
def merge_contact(base_contact, merge_contact)
|
||||||
|
|||||||
@@ -10,7 +10,8 @@ function toggleSecretField(e) {
|
|||||||
if (!textElement) return;
|
if (!textElement) return;
|
||||||
|
|
||||||
if (textElement.dataset.secretMasked === 'false') {
|
if (textElement.dataset.secretMasked === 'false') {
|
||||||
textElement.textContent = '•'.repeat(10);
|
const maskedLength = secretField.dataset.secretText?.length || 10;
|
||||||
|
textElement.textContent = '•'.repeat(maskedLength);
|
||||||
textElement.dataset.secretMasked = 'true';
|
textElement.dataset.secretMasked = 'true';
|
||||||
toggler.querySelector('svg use').setAttribute('xlink:href', '#eye-show');
|
toggler.querySelector('svg use').setAttribute('xlink:href', '#eye-show');
|
||||||
|
|
||||||
@@ -32,3 +33,13 @@ function copySecretField(e) {
|
|||||||
|
|
||||||
navigator.clipboard.writeText(secretField.dataset.secretText);
|
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 {
|
.cell-data__secret-field {
|
||||||
align-items: center;
|
align-items: center;
|
||||||
|
color: $hint-grey;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|
||||||
span {
|
span {
|
||||||
flex: 1;
|
flex: 0 0 auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
button {
|
[data-secret-toggler],
|
||||||
margin-left: 5px;
|
[data-secret-copier] {
|
||||||
|
background: transparent;
|
||||||
|
border: 0;
|
||||||
|
color: inherit;
|
||||||
|
margin-left: 0.5rem;
|
||||||
|
padding: 0;
|
||||||
|
|
||||||
svg {
|
svg {
|
||||||
fill: currentColor;
|
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
|
end
|
||||||
|
|
||||||
def scope_for_resolutions_count
|
def scope_for_resolutions_count
|
||||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(
|
scope.reporting_events.where(
|
||||||
name: :conversation_resolved,
|
name: :conversation_resolved,
|
||||||
conversations: { status: :resolved }, created_at: range
|
account_id: account.id,
|
||||||
).distinct
|
created_at: range
|
||||||
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def scope_for_bot_resolutions_count
|
def scope_for_bot_resolutions_count
|
||||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(
|
scope.reporting_events.where(
|
||||||
name: :conversation_bot_resolved,
|
name: :conversation_bot_resolved,
|
||||||
conversations: { status: :resolved }, created_at: range
|
account_id: account.id,
|
||||||
).distinct
|
created_at: range
|
||||||
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def scope_for_bot_handoffs_count
|
def scope_for_bot_handoffs_count
|
||||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(
|
scope.reporting_events.joins(:conversation).select(:conversation_id).where(
|
||||||
name: :conversation_bot_handoff,
|
name: :conversation_bot_handoff,
|
||||||
|
account_id: account.id,
|
||||||
created_at: range
|
created_at: range
|
||||||
).distinct
|
).distinct
|
||||||
end
|
end
|
||||||
|
|
||||||
def grouped_count
|
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(
|
@grouped_values = object_scope.group_by_period(
|
||||||
group_by,
|
group_by,
|
||||||
:created_at,
|
: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
|
end
|
||||||
|
|
||||||
def facebook_pages
|
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
|
end
|
||||||
|
|
||||||
def set_instagram_id(page_access_token, facebook_channel)
|
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
|
def campaign_params
|
||||||
params.require(:campaign).permit(:title, :description, :message, :enabled, :trigger_only_during_business_hours, :inbox_id, :sender_id,
|
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
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ class Api::V1::Accounts::ContactsController < Api::V1::Accounts::BaseController
|
|||||||
before_action :check_authorization
|
before_action :check_authorization
|
||||||
before_action :set_current_page, only: [:index, :active, :search, :filter]
|
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 :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
|
def index
|
||||||
@contacts_count = resolved_contacts.count
|
@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
|
contacts = Current.account.contacts.where(id: ::OnlineStatusTracker
|
||||||
.get_available_contact_ids(Current.account.id))
|
.get_available_contact_ids(Current.account.id))
|
||||||
@contacts_count = contacts.count
|
@contacts_count = contacts.count
|
||||||
@contacts = contacts.page(@current_page)
|
@contacts = fetch_contacts(contacts)
|
||||||
end
|
end
|
||||||
|
|
||||||
def show; end
|
def show; end
|
||||||
@@ -122,7 +122,7 @@ class Api::V1::Accounts::ContactsController < Api::V1::Accounts::BaseController
|
|||||||
def resolved_contacts
|
def resolved_contacts
|
||||||
return @resolved_contacts if @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 = @resolved_contacts.tagged_with(params[:labels], any: true) if params[:labels].present?
|
||||||
@resolved_contacts
|
@resolved_contacts
|
||||||
|
|||||||
@@ -124,6 +124,12 @@ class Api::V1::Accounts::ConversationsController < Api::V1::Accounts::BaseContro
|
|||||||
@conversation.save!
|
@conversation.save!
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def destroy
|
||||||
|
authorize @conversation, :destroy?
|
||||||
|
::DeleteObjectJob.perform_later(@conversation, Current.user, request.ip)
|
||||||
|
head :ok
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def permitted_update_params
|
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
|
include GoogleConcern
|
||||||
before_action :check_authorization
|
|
||||||
|
|
||||||
def create
|
def create
|
||||||
email = params[:authorization][:email]
|
|
||||||
redirect_url = google_client.auth_code.authorize_url(
|
redirect_url = google_client.auth_code.authorize_url(
|
||||||
{
|
{
|
||||||
redirect_uri: "#{base_url}/google/callback",
|
redirect_uri: "#{base_url}/google/callback",
|
||||||
scope: 'email profile https://mail.google.com/',
|
scope: scope,
|
||||||
response_type: 'code',
|
response_type: 'code',
|
||||||
prompt: 'consent', # the oauth flow does not return a refresh token, this is supposed to fix it
|
prompt: 'consent', # the oauth flow does not return a refresh token, this is supposed to fix it
|
||||||
access_type: 'offline', # the default is 'online'
|
access_type: 'offline', # the default is 'online'
|
||||||
|
state: state,
|
||||||
client_id: GlobalConfigService.load('GOOGLE_OAUTH_CLIENT_ID', nil)
|
client_id: GlobalConfigService.load('GOOGLE_OAUTH_CLIENT_ID', nil)
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
if redirect_url
|
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 }
|
render json: { success: true, url: redirect_url }
|
||||||
else
|
else
|
||||||
render json: { success: false }, status: :unprocessable_entity
|
render json: { success: false }, status: :unprocessable_entity
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def check_authorization
|
|
||||||
raise Pundit::NotAuthorizedError unless Current.account_user.administrator?
|
|
||||||
end
|
|
||||||
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') }
|
render status: :ok, json: { message: I18n.t('messages.inbox_deletetion_response') }
|
||||||
end
|
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
|
private
|
||||||
|
|
||||||
def fetch_inbox
|
def fetch_inbox
|
||||||
@@ -81,11 +90,15 @@ class Api::V1::Accounts::InboxesController < Api::V1::Accounts::BaseController
|
|||||||
end
|
end
|
||||||
|
|
||||||
def create_channel
|
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))
|
account_channels_method.create!(permitted_params(channel_type_from_params::EDITABLE_ATTRS)[:channel].except(:type))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def allowed_channel_types
|
||||||
|
%w[web_widget api email line telegram whatsapp sms]
|
||||||
|
end
|
||||||
|
|
||||||
def update_inbox_working_hours
|
def update_inbox_working_hours
|
||||||
@inbox.update_working_hours(params.permit(working_hours: Inbox::OFFISABLE_ATTRS)[:working_hours]) if params[:working_hours]
|
@inbox.update_working_hours(params.permit(working_hours: Inbox::OFFISABLE_ATTRS)[:working_hours]) if params[:working_hours]
|
||||||
end
|
end
|
||||||
@@ -170,6 +183,18 @@ class Api::V1::Accounts::InboxesController < Api::V1::Accounts::BaseController
|
|||||||
[]
|
[]
|
||||||
end
|
end
|
||||||
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
|
end
|
||||||
|
|
||||||
Api::V1::Accounts::InboxesController.prepend_mod_with('Api::V1::Accounts::InboxesController')
|
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 InstagramConcern
|
||||||
include Instagram::IntegrationHelper
|
include Instagram::IntegrationHelper
|
||||||
before_action :check_authorization
|
|
||||||
|
|
||||||
def create
|
def create
|
||||||
# https://developers.facebook.com/docs/instagram-platform/instagram-api-with-instagram-login/business-login#step-1--get-authorization
|
# 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
|
render json: { success: false }, status: :unprocessable_entity
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def check_authorization
|
|
||||||
raise Pundit::NotAuthorizedError unless Current.account_user.administrator?
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
class Api::V1::Accounts::Integrations::LinearController < Api::V1::Accounts::BaseController
|
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]
|
before_action :fetch_hook, only: [:destroy]
|
||||||
|
|
||||||
def destroy
|
def destroy
|
||||||
|
revoke_linear_token
|
||||||
@hook.destroy!
|
@hook.destroy!
|
||||||
head :ok
|
head :ok
|
||||||
end
|
end
|
||||||
@@ -27,10 +28,16 @@ class Api::V1::Accounts::Integrations::LinearController < Api::V1::Accounts::Bas
|
|||||||
end
|
end
|
||||||
|
|
||||||
def create_issue
|
def create_issue
|
||||||
issue = linear_processor_service.create_issue(permitted_params)
|
issue = linear_processor_service.create_issue(permitted_params, Current.user)
|
||||||
if issue[:error]
|
if issue[:error]
|
||||||
render json: { error: issue[:error] }, status: :unprocessable_entity
|
render json: { error: issue[:error] }, status: :unprocessable_entity
|
||||||
else
|
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
|
render json: issue[:data], status: :ok
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -38,21 +45,34 @@ class Api::V1::Accounts::Integrations::LinearController < Api::V1::Accounts::Bas
|
|||||||
def link_issue
|
def link_issue
|
||||||
issue_id = permitted_params[:issue_id]
|
issue_id = permitted_params[:issue_id]
|
||||||
title = permitted_params[:title]
|
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]
|
if issue[:error]
|
||||||
render json: { error: issue[:error] }, status: :unprocessable_entity
|
render json: { error: issue[:error] }, status: :unprocessable_entity
|
||||||
else
|
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
|
render json: issue[:data], status: :ok
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def unlink_issue
|
def unlink_issue
|
||||||
link_id = permitted_params[:link_id]
|
link_id = permitted_params[:link_id]
|
||||||
|
issue_id = permitted_params[:issue_id]
|
||||||
issue = linear_processor_service.unlink_issue(link_id)
|
issue = linear_processor_service.unlink_issue(link_id)
|
||||||
|
|
||||||
if issue[:error]
|
if issue[:error]
|
||||||
render json: { error: issue[:error] }, status: :unprocessable_entity
|
render json: { error: issue[:error] }, status: :unprocessable_entity
|
||||||
else
|
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
|
render json: issue[:data], status: :ok
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -101,4 +121,15 @@ class Api::V1::Accounts::Integrations::LinearController < Api::V1::Accounts::Bas
|
|||||||
def fetch_hook
|
def fetch_hook
|
||||||
@hook = Integrations::Hook.where(account: Current.account).find_by(app_id: 'linear')
|
@hook = Integrations::Hook.where(account: Current.account).find_by(app_id: 'linear')
|
||||||
end
|
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
|
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
|
include MicrosoftConcern
|
||||||
before_action :check_authorization
|
|
||||||
|
|
||||||
def create
|
def create
|
||||||
email = params[:authorization][:email]
|
|
||||||
redirect_url = microsoft_client.auth_code.authorize_url(
|
redirect_url = microsoft_client.auth_code.authorize_url(
|
||||||
{
|
{
|
||||||
redirect_uri: "#{base_url}/microsoft/callback",
|
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'
|
prompt: 'consent'
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
if redirect_url
|
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 }
|
render json: { success: true, url: redirect_url }
|
||||||
else
|
else
|
||||||
render json: { success: false }, status: :unprocessable_entity
|
render json: { success: false }, status: :unprocessable_entity
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
|
||||||
|
|
||||||
def check_authorization
|
|
||||||
raise Pundit::NotAuthorizedError unless Current.account_user.administrator?
|
|
||||||
end
|
|
||||||
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.update!(portal_params.merge(live_chat_widget_params)) if params[:portal].present?
|
||||||
# @portal.custom_domain = parsed_custom_domain
|
# @portal.custom_domain = parsed_custom_domain
|
||||||
process_attached_logo if params[:blob_id].present?
|
process_attached_logo if params[:blob_id].present?
|
||||||
rescue StandardError => e
|
rescue ActiveRecord::RecordInvalid => e
|
||||||
Rails.logger.error e
|
render_record_invalid(e)
|
||||||
render json: { error: @portal.errors.messages }.to_json, status: :unprocessable_entity
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -47,6 +46,20 @@ class Api::V1::Accounts::PortalsController < Api::V1::Accounts::BaseController
|
|||||||
head :ok
|
head :ok
|
||||||
end
|
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
|
def process_attached_logo
|
||||||
blob_id = params[:blob_id]
|
blob_id = params[:blob_id]
|
||||||
blob = ActiveStorage::Blob.find_by(id: blob_id)
|
blob = ActiveStorage::Blob.find_by(id: blob_id)
|
||||||
@@ -60,12 +73,12 @@ class Api::V1::Accounts::PortalsController < Api::V1::Accounts::BaseController
|
|||||||
end
|
end
|
||||||
|
|
||||||
def permitted_params
|
def permitted_params
|
||||||
params.permit(:id)
|
params.permit(:id, :email)
|
||||||
end
|
end
|
||||||
|
|
||||||
def portal_params
|
def portal_params
|
||||||
params.require(:portal).permit(
|
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: [] }] }
|
:name, :page_title, :slug, :archived, { config: [:default_locale, { allowed_locales: [] }] }
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
@@ -88,4 +101,10 @@ class Api::V1::Accounts::PortalsController < Api::V1::Accounts::BaseController
|
|||||||
domain = URI.parse(@portal.custom_domain)
|
domain = URI.parse(@portal.custom_domain)
|
||||||
domain.is_a?(URI::HTTP) ? domain.host : @portal.custom_domain
|
domain.is_a?(URI::HTTP) ? domain.host : @portal.custom_domain
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def valid_email?(email)
|
||||||
|
ValidEmail2::Address.new(email).valid?
|
||||||
|
end
|
||||||
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
|
end
|
||||||
|
|
||||||
def settings_params
|
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
|
end
|
||||||
|
|
||||||
def check_signup_enabled
|
def check_signup_enabled
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ class Api::V1::Widget::ConfigsController < Api::V1::Widget::BaseController
|
|||||||
private
|
private
|
||||||
|
|
||||||
def set_global_config
|
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
|
||||||
|
|
||||||
def set_contact
|
def set_contact
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
class Api::V2::Accounts::SummaryReportsController < Api::V1::Accounts::BaseController
|
class Api::V2::Accounts::SummaryReportsController < Api::V1::Accounts::BaseController
|
||||||
before_action :check_authorization
|
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
|
def agent
|
||||||
render_report_with(V2::Reports::AgentSummaryBuilder)
|
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)
|
render_report_with(V2::Reports::InboxSummaryBuilder)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def label
|
||||||
|
render_report_with(V2::Reports::LabelSummaryBuilder)
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def check_authorization
|
def check_authorization
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ module GoogleConcern
|
|||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def base_url
|
def scope
|
||||||
ENV.fetch('FRONTEND_URL', 'http://localhost:3000')
|
'email profile https://mail.google.com/'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ module MicrosoftConcern
|
|||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def base_url
|
def scope
|
||||||
ENV.fetch('FRONTEND_URL', 'http://localhost:3000')
|
'offline_access https://outlook.office.com/IMAP.AccessAsUser.All https://outlook.office.com/SMTP.Send openid profile email'
|
||||||
end
|
end
|
||||||
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
|
private
|
||||||
|
|
||||||
def switch_locale(&)
|
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]
|
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
|
locale ||= locale_from_custom_domain
|
||||||
|
|
||||||
# if locale is not set in account, let's use DEFAULT_LOCALE env variable
|
# if locale is not set in account, let's use DEFAULT_LOCALE env variable
|
||||||
locale ||= ENV.fetch('DEFAULT_LOCALE', nil)
|
locale ||= ENV.fetch('DEFAULT_LOCALE', nil)
|
||||||
|
|
||||||
set_locale(locale, &)
|
set_locale(locale, &)
|
||||||
end
|
end
|
||||||
|
|
||||||
def switch_locale_using_account_locale(&)
|
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, &)
|
set_locale(locale, &)
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -32,6 +43,12 @@ module SwitchLocale
|
|||||||
@portal.default_locale
|
@portal.default_locale
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def locale_from_user
|
||||||
|
return unless @user
|
||||||
|
|
||||||
|
@user.ui_settings&.dig('locale')
|
||||||
|
end
|
||||||
|
|
||||||
def set_locale(locale, &)
|
def set_locale(locale, &)
|
||||||
safe_locale = validate_and_get_locale(locale)
|
safe_locale = validate_and_get_locale(locale)
|
||||||
# Ensure locale won't bleed into other requests
|
# Ensure locale won't bleed into other requests
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ class DashboardController < ActionController::Base
|
|||||||
private
|
private
|
||||||
|
|
||||||
def ensure_html_format
|
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
|
end
|
||||||
|
|
||||||
def set_global_config
|
def set_global_config
|
||||||
@@ -66,7 +66,9 @@ class DashboardController < ActionController::Base
|
|||||||
ENABLE_ACCOUNT_SIGNUP: GlobalConfigService.load('ENABLE_ACCOUNT_SIGNUP', 'false'),
|
ENABLE_ACCOUNT_SIGNUP: GlobalConfigService.load('ENABLE_ACCOUNT_SIGNUP', 'false'),
|
||||||
FB_APP_ID: GlobalConfigService.load('FB_APP_ID', ''),
|
FB_APP_ID: GlobalConfigService.load('FB_APP_ID', ''),
|
||||||
INSTAGRAM_APP_ID: GlobalConfigService.load('INSTAGRAM_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?,
|
IS_ENTERPRISE: ChatwootApp.enterprise?,
|
||||||
AZURE_APP_ID: GlobalConfigService.load('AZURE_APP_ID', ''),
|
AZURE_APP_ID: GlobalConfigService.load('AZURE_APP_ID', ''),
|
||||||
GIT_SHA: GIT_HASH
|
GIT_SHA: GIT_HASH
|
||||||
|
|||||||
@@ -47,10 +47,8 @@ class DeviseOverrides::OmniauthCallbacksController < DeviseTokenAuth::OmniauthCa
|
|||||||
end
|
end
|
||||||
|
|
||||||
def get_resource_from_auth_hash # rubocop:disable Naming/AccessorMethodName
|
def get_resource_from_auth_hash # rubocop:disable Naming/AccessorMethodName
|
||||||
# find the user with their email instead of UID and token
|
email = auth_hash.dig('info', 'email')
|
||||||
@resource = resource_class.where(
|
@resource = resource_class.from_email(email)
|
||||||
email: auth_hash['info']['email']
|
|
||||||
).first
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def validate_signup_email_is_business_domain?
|
def validate_signup_email_is_business_domain?
|
||||||
@@ -75,3 +73,5 @@ class DeviseOverrides::OmniauthCallbacksController < DeviseTokenAuth::OmniauthCa
|
|||||||
'user'
|
'user'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
DeviseOverrides::OmniauthCallbacksController.prepend_mod_with('DeviseOverrides::OmniauthCallbacksController')
|
||||||
|
|||||||
@@ -44,3 +44,5 @@ class DeviseOverrides::PasswordsController < Devise::PasswordsController
|
|||||||
}, status: status
|
}, status: status
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
DeviseOverrides::PasswordsController.prepend_mod_with('DeviseOverrides::PasswordsController')
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ class MicrosoftController < ApplicationController
|
|||||||
after_action :set_version_header
|
after_action :set_version_header
|
||||||
|
|
||||||
def identity_association
|
def identity_association
|
||||||
microsoft_indentity
|
microsoft_identity
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
@@ -11,7 +11,7 @@ class MicrosoftController < ApplicationController
|
|||||||
response.headers['Content-Length'] = { associatedApplications: [{ applicationId: @identity_json }] }.to_json.length
|
response.headers['Content-Length'] = { associatedApplications: [{ applicationId: @identity_json }] }.to_json.length
|
||||||
end
|
end
|
||||||
|
|
||||||
def microsoft_indentity
|
def microsoft_identity
|
||||||
@identity_json = GlobalConfigService.load('AZURE_APP_ID', nil)
|
@identity_json = GlobalConfigService.load('AZURE_APP_ID', nil)
|
||||||
end
|
end
|
||||||
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
|
handle_response
|
||||||
::Redis::Alfred.delete(cache_key)
|
|
||||||
rescue StandardError => e
|
rescue StandardError => e
|
||||||
ChatwootExceptionTracker.new(e).capture_exception
|
ChatwootExceptionTracker.new(e).capture_exception
|
||||||
redirect_to '/'
|
redirect_to '/'
|
||||||
@@ -64,13 +63,10 @@ class OauthCallbackController < ApplicationController
|
|||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
end
|
end
|
||||||
|
|
||||||
def cache_key
|
|
||||||
"#{provider_name}::#{users_data['email'].downcase}"
|
|
||||||
end
|
|
||||||
|
|
||||||
def create_channel_with_inbox
|
def create_channel_with_inbox
|
||||||
ActiveRecord::Base.transaction do
|
ActiveRecord::Base.transaction do
|
||||||
channel_email = Channel::Email.create!(email: users_data['email'], account: account)
|
channel_email = Channel::Email.create!(email: users_data['email'], account: account)
|
||||||
|
|
||||||
account.inboxes.create!(
|
account.inboxes.create!(
|
||||||
account: account,
|
account: account,
|
||||||
channel: channel_email,
|
channel: channel_email,
|
||||||
@@ -85,12 +81,17 @@ class OauthCallbackController < ApplicationController
|
|||||||
decoded_token[0]
|
decoded_token[0]
|
||||||
end
|
end
|
||||||
|
|
||||||
def account_id
|
def account_from_signed_id
|
||||||
::Redis::Alfred.get(cache_key)
|
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
|
end
|
||||||
|
|
||||||
def account
|
def account
|
||||||
@account ||= Account.find(account_id)
|
@account ||= account_from_signed_id
|
||||||
end
|
end
|
||||||
|
|
||||||
# Fallback name, for when name field is missing from users_data
|
# Fallback name, for when name field is missing from users_data
|
||||||
|
|||||||
@@ -1,4 +1,11 @@
|
|||||||
class Platform::Api::V1::AccountsController < PlatformController
|
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 show; end
|
||||||
|
|
||||||
def create
|
def create
|
||||||
|
|||||||
@@ -7,13 +7,19 @@ class Public::Api::V1::Portals::ArticlesController < Public::Api::V1::Portals::B
|
|||||||
|
|
||||||
def index
|
def index
|
||||||
@articles = @portal.articles.published.includes(:category, :author)
|
@articles = @portal.articles.published.includes(:category, :author)
|
||||||
|
|
||||||
|
@articles = @articles.where(locale: permitted_params[:locale]) if permitted_params[:locale].present?
|
||||||
|
|
||||||
@articles_count = @articles.count
|
@articles_count = @articles.count
|
||||||
|
|
||||||
search_articles
|
search_articles
|
||||||
order_by_sort_param
|
order_by_sort_param
|
||||||
limit_results
|
limit_results
|
||||||
end
|
end
|
||||||
|
|
||||||
def show; end
|
def show
|
||||||
|
@og_image_url = helpers.set_og_image_url(@portal.name, @article.title)
|
||||||
|
end
|
||||||
|
|
||||||
def tracking_pixel
|
def tracking_pixel
|
||||||
@article = @portal.articles.find_by(slug: permitted_params[:article_slug])
|
@article = @portal.articles.find_by(slug: permitted_params[:article_slug])
|
||||||
|
|||||||
@@ -58,6 +58,6 @@ class Public::Api::V1::Portals::BaseController < PublicController
|
|||||||
end
|
end
|
||||||
|
|
||||||
def set_global_config
|
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
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -8,7 +8,9 @@ class Public::Api::V1::Portals::CategoriesController < Public::Api::V1::Portals:
|
|||||||
@categories = @portal.categories.order(position: :asc)
|
@categories = @portal.categories.order(position: :asc)
|
||||||
end
|
end
|
||||||
|
|
||||||
def show; end
|
def show
|
||||||
|
@og_image_url = helpers.set_og_image_url(@portal.name, @category.name)
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
|||||||
@@ -4,7 +4,9 @@ class Public::Api::V1::PortalsController < Public::Api::V1::Portals::BaseControl
|
|||||||
before_action :redirect_to_portal_with_locale, only: [:show]
|
before_action :redirect_to_portal_with_locale, only: [:show]
|
||||||
layout 'portal'
|
layout 'portal'
|
||||||
|
|
||||||
def show; end
|
def show
|
||||||
|
@og_image_url = helpers.set_og_image_url('', @portal.header_text)
|
||||||
|
end
|
||||||
|
|
||||||
def sitemap
|
def sitemap
|
||||||
@help_center_url = @portal.custom_domain || ChatwootApp.help_center_root
|
@help_center_url = @portal.custom_domain || ChatwootApp.help_center_root
|
||||||
|
|||||||
@@ -17,7 +17,12 @@ class SlackUploadsController < ApplicationController
|
|||||||
end
|
end
|
||||||
|
|
||||||
def blob_url
|
def blob_url
|
||||||
url_for(@blob.representation(resize_to_fill: [250, nil]))
|
# 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
|
end
|
||||||
|
|
||||||
def avatar_url
|
def avatar_url
|
||||||
|
|||||||
@@ -2,6 +2,13 @@ class SuperAdmin::AccountUsersController < SuperAdmin::ApplicationController
|
|||||||
# Overwrite any of the RESTful controller actions to implement custom behavior
|
# 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.
|
# 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
|
def create
|
||||||
resource = resource_class.new(resource_params)
|
resource = resource_class.new(resource_params)
|
||||||
authorize_resource(resource)
|
authorize_resource(resource)
|
||||||
|
|||||||
@@ -39,7 +39,10 @@ class SuperAdmin::AppConfigsController < SuperAdmin::ApplicationController
|
|||||||
'email' => ['MAILER_INBOUND_EMAIL_DOMAIN'],
|
'email' => ['MAILER_INBOUND_EMAIL_DOMAIN'],
|
||||||
'linear' => %w[LINEAR_CLIENT_ID LINEAR_CLIENT_SECRET],
|
'linear' => %w[LINEAR_CLIENT_ID LINEAR_CLIENT_SECRET],
|
||||||
'slack' => %w[SLACK_CLIENT_ID SLACK_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])
|
@allowed_configs = mapping.fetch(@config, %w[ENABLE_ACCOUNT_SIGNUP FIREBASE_PROJECT_ID FIREBASE_CREDENTIALS])
|
||||||
|
|||||||
@@ -7,8 +7,9 @@
|
|||||||
class SuperAdmin::ApplicationController < Administrate::ApplicationController
|
class SuperAdmin::ApplicationController < Administrate::ApplicationController
|
||||||
include ActionView::Helpers::TagHelper
|
include ActionView::Helpers::TagHelper
|
||||||
include ActionView::Context
|
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
|
# authenticiation done via devise : SuperAdmin Model
|
||||||
before_action :authenticate_super_admin!
|
before_action :authenticate_super_admin!
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,6 @@ class Survey::ResponsesController < ActionController::Base
|
|||||||
private
|
private
|
||||||
|
|
||||||
def set_global_config
|
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
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -27,7 +27,11 @@ class Twilio::CallbackController < ApplicationController
|
|||||||
*Array.new(10) { |i| :"MediaUrl#{i}" },
|
*Array.new(10) { |i| :"MediaUrl#{i}" },
|
||||||
*Array.new(10) { |i| :"MediaContentType#{i}" },
|
*Array.new(10) { |i| :"MediaContentType#{i}" },
|
||||||
:MessagingServiceSid,
|
:MessagingServiceSid,
|
||||||
:NumMedia
|
:NumMedia,
|
||||||
|
:Latitude,
|
||||||
|
:Longitude,
|
||||||
|
:MessageType,
|
||||||
|
:ProfileName
|
||||||
)
|
)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -4,7 +4,16 @@ class Webhooks::InstagramController < ActionController::API
|
|||||||
def events
|
def events
|
||||||
Rails.logger.info('Instagram webhook received events')
|
Rails.logger.info('Instagram webhook received events')
|
||||||
if params['object'].casecmp('instagram').zero?
|
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
|
render json: :ok
|
||||||
else
|
else
|
||||||
Rails.logger.warn("Message is not received from the instagram webhook event: #{params['object']}")
|
Rails.logger.warn("Message is not received from the instagram webhook event: #{params['object']}")
|
||||||
@@ -14,6 +23,16 @@ class Webhooks::InstagramController < ActionController::API
|
|||||||
|
|
||||||
private
|
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)
|
def valid_token?(token)
|
||||||
# Validates against both IG_VERIFY_TOKEN (Instagram channel via Facebook page) and
|
# Validates against both IG_VERIFY_TOKEN (Instagram channel via Facebook page) and
|
||||||
# INSTAGRAM_VERIFY_TOKEN (Instagram channel via direct Instagram login)
|
# INSTAGRAM_VERIFY_TOKEN (Instagram channel via direct Instagram login)
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ class WidgetsController < ActionController::Base
|
|||||||
private
|
private
|
||||||
|
|
||||||
def set_global_config
|
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
|
end
|
||||||
|
|
||||||
def set_web_widget
|
def set_web_widget
|
||||||
|
|||||||
@@ -15,7 +15,13 @@ class NotificationFinder
|
|||||||
end
|
end
|
||||||
|
|
||||||
def unread_count
|
def unread_count
|
||||||
@notifications.where(read_at: nil).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
|
end
|
||||||
|
|
||||||
def count
|
def count
|
||||||
@@ -27,7 +33,7 @@ class NotificationFinder
|
|||||||
def set_up
|
def set_up
|
||||||
find_all_notifications
|
find_all_notifications
|
||||||
filter_snoozed_notifications
|
filter_snoozed_notifications
|
||||||
fitler_read_notifications
|
filter_read_notifications
|
||||||
end
|
end
|
||||||
|
|
||||||
def find_all_notifications
|
def find_all_notifications
|
||||||
@@ -38,7 +44,7 @@ class NotificationFinder
|
|||||||
@notifications = @notifications.where(snoozed_until: nil) unless type_included?('snoozed')
|
@notifications = @notifications.where(snoozed_until: nil) unless type_included?('snoozed')
|
||||||
end
|
end
|
||||||
|
|
||||||
def fitler_read_notifications
|
def filter_read_notifications
|
||||||
@notifications = @notifications.where(read_at: nil) unless type_included?('read')
|
@notifications = @notifications.where(read_at: nil) unless type_included?('read')
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -36,9 +36,13 @@ module Api::V2::Accounts::ReportsHelper
|
|||||||
end
|
end
|
||||||
|
|
||||||
def generate_labels_report
|
def generate_labels_report
|
||||||
Current.account.labels.map do |label|
|
reports = V2::Reports::LabelSummaryBuilder.new(
|
||||||
label_report = report_builder({ type: :label, id: label.id }).short_summary
|
account: Current.account,
|
||||||
[label.title] + generate_readable_report_metrics(label_report)
|
params: build_params({})
|
||||||
|
).build
|
||||||
|
|
||||||
|
reports.map do |report|
|
||||||
|
[report[:name]] + generate_readable_report_metrics(report)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,13 @@
|
|||||||
module MessageFormatHelper
|
module MessageFormatHelper
|
||||||
include RegexHelper
|
|
||||||
|
|
||||||
def transform_user_mention_content(message_content)
|
def transform_user_mention_content(message_content)
|
||||||
# attachment message without content, message_content is nil
|
# 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
|
end
|
||||||
|
|
||||||
def render_message_content(message_content)
|
def render_message_content(message_content)
|
||||||
|
|||||||
@@ -1,4 +1,22 @@
|
|||||||
module PortalHelper
|
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)
|
def generate_portal_bg_color(portal_color, theme)
|
||||||
base_color = theme == 'dark' ? 'black' : 'white'
|
base_color = theme == 'dark' ? 'black' : 'white'
|
||||||
"color-mix(in srgb, #{portal_color} 20%, #{base_color})"
|
"color-mix(in srgb, #{portal_color} 20%, #{base_color})"
|
||||||
@@ -57,6 +75,17 @@ module PortalHelper
|
|||||||
end
|
end
|
||||||
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)
|
def render_category_content(content)
|
||||||
ChatwootMarkdownRenderer.new(content).render_markdown_to_plain_text
|
ChatwootMarkdownRenderer.new(content).render_markdown_to_plain_text
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -53,13 +53,13 @@ module ReportHelper
|
|||||||
end
|
end
|
||||||
|
|
||||||
def resolutions
|
def resolutions
|
||||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(account_id: account.id, name: :conversation_resolved,
|
scope.reporting_events.where(account_id: account.id, name: :conversation_resolved,
|
||||||
conversations: { status: :resolved }, created_at: range).distinct
|
created_at: range)
|
||||||
end
|
end
|
||||||
|
|
||||||
def bot_resolutions
|
def bot_resolutions
|
||||||
scope.reporting_events.joins(:conversation).select(:conversation_id).where(account_id: account.id, name: :conversation_bot_resolved,
|
scope.reporting_events.where(account_id: account.id, name: :conversation_bot_resolved,
|
||||||
conversations: { status: :resolved }, created_at: range).distinct
|
created_at: range)
|
||||||
end
|
end
|
||||||
|
|
||||||
def bot_handoffs
|
def bot_handoffs
|
||||||
|
|||||||
@@ -18,12 +18,25 @@ module ReportingEventHelper
|
|||||||
end
|
end
|
||||||
|
|
||||||
def last_non_human_activity(conversation)
|
def last_non_human_activity(conversation)
|
||||||
# check if a handoff event already exists
|
# Try to get either a handoff or reopened event first
|
||||||
handoff_event = ReportingEvent.where(conversation_id: conversation.id, name: 'conversation_bot_handoff').last
|
# 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,
|
return event.event_end_time if event&.event_end_time
|
||||||
# otherwise it's when the conversation was created
|
|
||||||
handoff_event&.event_end_time || conversation.created_at
|
# 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
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
# TODO: Move this values to features.yml itself
|
# TODO: Move this values to features.yml itself
|
||||||
# No need to replicate the same values in two places
|
# No need to replicate the same values in two places
|
||||||
|
|
||||||
|
# ------- Premium Features ------- #
|
||||||
captain:
|
captain:
|
||||||
name: 'Captain'
|
name: 'Captain'
|
||||||
description: 'Enable AI-powered conversations with your customers.'
|
description: 'Enable AI-powered conversations with your customers.'
|
||||||
@@ -32,6 +34,15 @@ disable_branding:
|
|||||||
enabled: <%= (ChatwootHub.pricing_plan != 'community') %>
|
enabled: <%= (ChatwootHub.pricing_plan != 'community') %>
|
||||||
icon: 'icon-sailbot-fill'
|
icon: 'icon-sailbot-fill'
|
||||||
enterprise: true
|
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:
|
live_chat:
|
||||||
name: 'Live Chat'
|
name: 'Live Chat'
|
||||||
description: 'Improve your customer experience using a live chat on your website.'
|
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.'
|
description: 'Manage your email customer interactions from Chatwoot.'
|
||||||
enabled: true
|
enabled: true
|
||||||
icon: 'icon-mail-send-fill'
|
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:
|
messenger:
|
||||||
name: 'Messenger'
|
name: 'Messenger'
|
||||||
description: 'Stay connected with your customers on Facebook & Instagram.'
|
description: 'Stay connected with your customers on Facebook & Instagram.'
|
||||||
@@ -69,34 +86,46 @@ line:
|
|||||||
description: 'Manage your Line customer interactions from Chatwoot.'
|
description: 'Manage your Line customer interactions from Chatwoot.'
|
||||||
enabled: true
|
enabled: true
|
||||||
icon: 'icon-line-line'
|
icon: 'icon-line-line'
|
||||||
sms:
|
|
||||||
name: 'SMS'
|
# ------- OAuth & Authentication ------- #
|
||||||
description: 'Manage your SMS customer interactions from Chatwoot.'
|
google:
|
||||||
|
name: 'Google'
|
||||||
|
description: 'Configuration for setting up Google OAuth Integration'
|
||||||
enabled: true
|
enabled: true
|
||||||
icon: 'icon-message-line'
|
icon: 'icon-google'
|
||||||
help_center:
|
config_key: 'google'
|
||||||
name: 'Help Center'
|
|
||||||
description: 'Allow agents to create help center articles and publish them in a portal.'
|
|
||||||
enabled: true
|
|
||||||
icon: 'icon-book-2-line'
|
|
||||||
microsoft:
|
microsoft:
|
||||||
name: 'Microsoft'
|
name: 'Microsoft'
|
||||||
description: 'Configuration for setting up Microsoft Email'
|
description: 'Configuration for setting up Microsoft Email'
|
||||||
enabled: true
|
enabled: true
|
||||||
icon: 'icon-microsoft'
|
icon: 'icon-microsoft'
|
||||||
config_key: 'microsoft'
|
config_key: 'microsoft'
|
||||||
|
|
||||||
|
# ------- Third-party Integrations ------- #
|
||||||
linear:
|
linear:
|
||||||
name: 'Linear'
|
name: 'Linear'
|
||||||
description: 'Configuration for setting up Linear Integration'
|
description: 'Configuration for setting up Linear Integration'
|
||||||
enabled: true
|
enabled: true
|
||||||
icon: 'icon-linear'
|
icon: 'icon-linear'
|
||||||
config_key: 'linear'
|
config_key: 'linear'
|
||||||
|
notion:
|
||||||
|
name: 'Notion'
|
||||||
|
description: 'Configuration for setting up Notion Integration'
|
||||||
|
enabled: true
|
||||||
|
icon: 'icon-notion'
|
||||||
|
config_key: 'notion'
|
||||||
slack:
|
slack:
|
||||||
name: 'Slack'
|
name: 'Slack'
|
||||||
description: 'Configuration for setting up Slack Integration'
|
description: 'Configuration for setting up Slack Integration'
|
||||||
enabled: true
|
enabled: true
|
||||||
icon: 'icon-slack'
|
icon: 'icon-slack'
|
||||||
config_key: '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:
|
shopify:
|
||||||
name: 'Shopify'
|
name: 'Shopify'
|
||||||
description: 'Configuration for setting up Shopify Integration'
|
description: 'Configuration for setting up Shopify Integration'
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
module SuperAdmin::FeaturesHelper
|
module SuperAdmin::FeaturesHelper
|
||||||
def self.available_features
|
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
|
end
|
||||||
|
|
||||||
def self.plan_details
|
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>
|
<script>
|
||||||
import { mapGetters } from 'vuex';
|
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 LoadingState from './components/widgets/LoadingState.vue';
|
||||||
import NetworkNotification from './components/NetworkNotification.vue';
|
import NetworkNotification from './components/NetworkNotification.vue';
|
||||||
import UpdateBanner from './components/app/UpdateBanner.vue';
|
import UpdateBanner from './components/app/UpdateBanner.vue';
|
||||||
@@ -19,6 +19,7 @@ import {
|
|||||||
verifyServiceWorkerExistence,
|
verifyServiceWorkerExistence,
|
||||||
} from './helper/pushHelper';
|
} from './helper/pushHelper';
|
||||||
import ReconnectService from 'dashboard/helper/ReconnectService';
|
import ReconnectService from 'dashboard/helper/ReconnectService';
|
||||||
|
import { useUISettings } from 'dashboard/composables/useUISettings';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'App',
|
name: 'App',
|
||||||
@@ -38,12 +39,14 @@ export default {
|
|||||||
const { accountId } = useAccount();
|
const { accountId } = useAccount();
|
||||||
// Use the font size composable (it automatically sets up the watcher)
|
// Use the font size composable (it automatically sets up the watcher)
|
||||||
const { currentFontSize } = useFontSize();
|
const { currentFontSize } = useFontSize();
|
||||||
|
const { uiSettings } = useUISettings();
|
||||||
|
|
||||||
return {
|
return {
|
||||||
router,
|
router,
|
||||||
store,
|
store,
|
||||||
currentAccountId: accountId,
|
currentAccountId: accountId,
|
||||||
currentFontSize,
|
currentFontSize,
|
||||||
|
uiSettings,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
@@ -88,7 +91,10 @@ export default {
|
|||||||
mounted() {
|
mounted() {
|
||||||
this.initializeColorTheme();
|
this.initializeColorTheme();
|
||||||
this.listenToThemeChanges();
|
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() {
|
unmounted() {
|
||||||
if (this.reconnectService) {
|
if (this.reconnectService) {
|
||||||
@@ -114,7 +120,8 @@ export default {
|
|||||||
const { locale, latest_chatwoot_version: latestChatwootVersion } =
|
const { locale, latest_chatwoot_version: latestChatwootVersion } =
|
||||||
this.getAccount(this.currentAccountId);
|
this.getAccount(this.currentAccountId);
|
||||||
const { pubsub_token: pubsubToken } = this.currentUser || {};
|
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;
|
this.latestChatwootVersion = latestChatwootVersion;
|
||||||
vueActionCable.init(this.store, pubsubToken);
|
vueActionCable.init(this.store, pubsubToken);
|
||||||
this.reconnectService = new ReconnectService(this.store, this.router);
|
this.reconnectService = new ReconnectService(this.store, this.router);
|
||||||
@@ -136,8 +143,7 @@ export default {
|
|||||||
<div
|
<div
|
||||||
v-if="!authUIFlags.isFetching && !accountUIFlags.isFetchingItem"
|
v-if="!authUIFlags.isFetching && !accountUIFlags.isFetchingItem"
|
||||||
id="app"
|
id="app"
|
||||||
class="flex-grow-0 w-full h-full min-h-0 app-wrapper"
|
class="flex flex-col w-full h-screen min-h-0"
|
||||||
:class="{ 'app-rtl--wrapper': isRTL }"
|
|
||||||
:dir="isRTL ? 'rtl' : 'ltr'"
|
:dir="isRTL ? 'rtl' : 'ltr'"
|
||||||
>
|
>
|
||||||
<UpdateBanner :latest-chatwoot-version="latestChatwootVersion" />
|
<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;
|
return false;
|
||||||
},
|
},
|
||||||
profileUpdate({
|
profileUpdate({ displayName, avatar, ...profileAttributes }) {
|
||||||
password,
|
|
||||||
password_confirmation,
|
|
||||||
displayName,
|
|
||||||
avatar,
|
|
||||||
...profileAttributes
|
|
||||||
}) {
|
|
||||||
const formData = new FormData();
|
const formData = new FormData();
|
||||||
Object.keys(profileAttributes).forEach(key => {
|
Object.keys(profileAttributes).forEach(key => {
|
||||||
const hasValue = profileAttributes[key] === undefined;
|
const hasValue = profileAttributes[key] === undefined;
|
||||||
@@ -53,16 +47,22 @@ export default {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
formData.append('profile[display_name]', displayName || '');
|
formData.append('profile[display_name]', displayName || '');
|
||||||
if (password && password_confirmation) {
|
|
||||||
formData.append('profile[password]', password);
|
|
||||||
formData.append('profile[password_confirmation]', password_confirmation);
|
|
||||||
}
|
|
||||||
if (avatar) {
|
if (avatar) {
|
||||||
formData.append('profile[avatar]', avatar);
|
formData.append('profile[avatar]', avatar);
|
||||||
}
|
}
|
||||||
return axios.put(endPoints('profileUpdate').url, formData);
|
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 }) {
|
updateUISettings({ uiSettings }) {
|
||||||
return axios.put(endPoints('profileUpdate').url, {
|
return axios.put(endPoints('profileUpdate').url, {
|
||||||
profile: { ui_settings: uiSettings },
|
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);
|
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
|
// eslint-disable-next-line default-param-last
|
||||||
filter(page = 1, sortAttr = 'name', queryPayload) {
|
filter(page = 1, sortAttr = 'name', queryPayload) {
|
||||||
let requestURL = `${this.url}/filter?${buildContactParams(page, sortAttr)}`;
|
let requestURL = `${this.url}/filter?${buildContactParams(page, sortAttr)}`;
|
||||||
|
|||||||
@@ -51,6 +51,7 @@ const endPoints = {
|
|||||||
resendConfirmation: {
|
resendConfirmation: {
|
||||||
url: '/api/v1/profile/resend_confirmation',
|
url: '/api/v1/profile/resend_confirmation',
|
||||||
},
|
},
|
||||||
|
|
||||||
resetAccessToken: {
|
resetAccessToken: {
|
||||||
url: '/api/v1/profile/reset_access_token',
|
url: '/api/v1/profile/reset_access_token',
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -21,6 +21,14 @@ class PortalsAPI extends ApiClient {
|
|||||||
deleteLogo(portalSlug) {
|
deleteLogo(portalSlug) {
|
||||||
return axios.delete(`${this.url}/${portalSlug}/logo`);
|
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;
|
export default PortalsAPI;
|
||||||
|
|||||||
@@ -134,13 +134,13 @@ class ConversationApi extends ApiClient {
|
|||||||
return axios.get(`${this.url}/${conversationId}/attachments`);
|
return axios.get(`${this.url}/${conversationId}/attachments`);
|
||||||
}
|
}
|
||||||
|
|
||||||
requestCopilot(conversationId, body) {
|
|
||||||
return axios.post(`${this.url}/${conversationId}/copilot`, body);
|
|
||||||
}
|
|
||||||
|
|
||||||
getInboxAssistant(conversationId) {
|
getInboxAssistant(conversationId) {
|
||||||
return axios.get(`${this.url}/${conversationId}/inbox_assistant`);
|
return axios.get(`${this.url}/${conversationId}/inbox_assistant`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
delete(conversationId) {
|
||||||
|
return axios.delete(`${this.url}/${conversationId}`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default new ConversationApi();
|
export default new ConversationApi();
|
||||||
|
|||||||
@@ -28,6 +28,10 @@ class Inboxes extends CacheEnabledApiClient {
|
|||||||
agent_bot: botId,
|
agent_bot: botId,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
syncTemplates(inboxId) {
|
||||||
|
return axios.post(`${this.url}/${inboxId}/sync_templates`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default new Inboxes();
|
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`, {
|
return axios.post(`${this.url}/unlink_issue`, {
|
||||||
link_id: linkId,
|
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('getCampaigns');
|
||||||
expect(inboxesAPI).toHaveProperty('getAgentBot');
|
expect(inboxesAPI).toHaveProperty('getAgentBot');
|
||||||
expect(inboxesAPI).toHaveProperty('setAgentBot');
|
expect(inboxesAPI).toHaveProperty('setAgentBot');
|
||||||
|
expect(inboxesAPI).toHaveProperty('syncTemplates');
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('API calls', () => {
|
describe('API calls', () => {
|
||||||
@@ -40,5 +41,12 @@ describe('#InboxesAPI', () => {
|
|||||||
inboxesAPI.deleteInboxAvatar(2);
|
inboxesAPI.deleteInboxAvatar(2);
|
||||||
expect(axiosMock.delete).toHaveBeenCalledWith('/api/v1/inboxes/2/avatar');
|
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