Data & Privacy

Overview

Because WaDesk is self-hosted, your data lives entirely on your own server — nothing leaves your systems except the calls you make to WhatsApp, your payment gateway, and any AI provider you set up. Inside the app, three things keep that data safe: every secret is encrypted, each workspace is kept separate so customers can't see each other's data, and sensitive actions are recorded in an audit trail. A GDPR/CCPA cookie-consent banner covers the public-facing side.

Encryption at Rest

Provider credentials and other secrets are encrypted with strong, industry-standard encryption (AES-256) before they're ever saved to the database. The following sensitive data is encrypted:

DataProtection
WhatsApp provider credentials (Cloud API app secret, access token, and webhook verify token; Twilio SID and token; Unofficial API device info)Encrypted
Per-workspace AI provider API keys (OpenAI, Anthropic, Gemini)Encrypted
Admin / platform AI keysEncrypted
2FA authenticator secret & recovery codesEncrypted
Payment-gateway credentialsEncrypted
Integration tokens (Shopify, WooCommerce, HubSpot, Google)Encrypted
Account passwordsbcrypt one-way hash (can never be read back)

Decryption fails gracefully: if a secret can't be unlocked (for example after the encryption key is changed), the app simply asks you to re-enter that key instead of breaking. Nothing crashes.

Important: Everything above is protected by your app's encryption key (APP_KEY). Set a strong, unique key when you install, keep it out of source control, and back it up safely — if you lose it, every encrypted credential becomes unrecoverable, and changing it invalidates all stored secrets and 2FA setups. Never reuse the key across different environments.

Per-Workspace Data Isolation

The workspace is the top-level boundary between customers. A user can belong to several workspaces and switch between them, and each one holds completely separate data — contacts, conversations, campaigns, flows, templates, devices, AI keys, and provider settings.

Every piece of data is tagged with the workspace it belongs to, and the app only ever reads data for the workspace you're currently in — so one customer's data can never leak into another's. Within a workspace, roles (owner, admin, agent, viewer) further limit what each member can see and do.

Note: Platform staff work in a separate admin area and aren't put through the workspace gate, because workspaces are a customer concept. Customer data is reached through the workspace, not by staff browsing the database directly.

Audit Log

Sensitive actions are recorded in a tamper-resistant, append-only audit trail. Each entry captures:

RecordedWhat it captures
LevelWhether it was a platform-wide (admin) action or a workspace action
WhoThe signed-in user who performed the action
WorkspaceThe workspace the action belongs to
ActionThe event name (e.g. billing.payment.captured, admin.security.policy_updated)
What was affectedThe specific record the action changed
DetailsA label plus any extra context — with secrets stripped out, see below
OriginThe IP address and browser the request came from
Resultsuccess, warning, or failure

A separate security log captures security-tab events — turning 2FA on or off, failed 2FA attempts, signing devices out, and password changes — with the same who / workspace / IP / browser context. Neither log can ever block the action it's recording: if writing the log fails, the action still goes through. The Admin Security dashboard presents these entries as a risk queue with key figures (open risks, blocked login attempts, 2FA coverage).

Secret Scrubbing

As an extra safeguard, before any audit entry is saved, its details are scanned and any value that looks like a secret — anything labelled password, secret, token, api_key, credential, private, access_key, client_secret, webhook, signing, or bearer — is replaced with [REDACTED]. The label is kept so you can still see what changed, just never the secret value itself. This is a backstop to make sure a secret can never accidentally end up in an audit entry. A few fields that only hold field names (not the secrets) are left as-is.

Audit Retention & Pruning

Old audit entries are removed automatically once they pass the retention window. The default window is 365 days, and you can change it under Admin → Security. If you'd rather run the cleanup by hand, these commands are available:

php artisan audit:prune              # use the configured retention period
php artisan audit:prune --days=180   # one-off: use 180 days instead
php artisan audit:prune --dry-run    # report what would be deleted, delete nothing

Setting the retention to 0 turns pruning off entirely — useful where compliance requires keeping records permanently. The cleanup deletes in small batches so it can't overwhelm a large database.

Important: Choose a retention period that matches your compliance and legal-hold requirements before going live. Pruning is permanent — run the --dry-run option first to see how much would be deleted, and make sure your database backups capture any audit records you might need beyond the retention window.

The public site shows a GDPR/CCPA cookie-consent banner. It has two parts: a compact banner (a bottom bar, top bar, or centered box, depending on the admin setting) with Decline / Customize / Accept All, and a detailed preferences box with three categories — Strictly Necessary (always on), Analytics, and Marketing.

The visitor's choice is saved in a cookie named wadesk_cookie_consent. On every page, only the trackers for the categories the visitor allowed are loaded — so analytics and marketing scripts never run until consent is given. Admins set the banner text, style, policy links, and behaviour from the privacy settings page, and the banner respects the browser's Do Not Track signal when that option is turned on.

Note: Set your Privacy Policy and Cookies Policy URLs in the admin privacy settings so the consent banner can link to them — required for a compliant disclosure.
WaDesk Documentation