Files
logseq/src/main/frontend/config.cljs
hoellen 8997638e5b Custom Sync Server URL (#12459)
* Add custom sync server

Co-authored-by: Tienson Qin <tiensonqin@gmail.com>
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
2026-04-13 13:00:19 +08:00

362 lines
9.7 KiB
Clojure

(ns frontend.config
"App config and fns built on top of configuration"
(:require [clojure.set :as set]
[clojure.string :as string]
[frontend.state :as state]
[frontend.util :as util]
[goog.crypt.Md5]
[logseq.common.config :as common-config]
[logseq.common.path :as path]
[logseq.db.sqlite.util :as sqlite-util]
[shadow.resource :as rc]))
(goog-define DEV-RELEASE false)
(defonce dev-release? DEV-RELEASE)
(defonce dev? ^boolean (or dev-release? goog.DEBUG))
(defonce publishing? common-config/PUBLISHING)
(goog-define REVISION "unknown")
(defonce revision REVISION)
(goog-define ENABLE-FILE-SYNC-PRODUCTION false)
;; this is a feature flag to enable the account tab
;; when it launches (when pro plan launches) it should be removed
(def ENABLE-SETTINGS-ACCOUNT-TAB false)
(if ENABLE-FILE-SYNC-PRODUCTION
(do (def API-DOMAIN "api.logseq.com")
(def COGNITO-IDP "https://cognito-idp.us-east-1.amazonaws.com/")
(def COGNITO-CLIENT-ID "69cs1lgme7p8kbgld8n5kseii6")
(def REGION "us-east-1")
(def USER-POOL-ID "us-east-1_dtagLnju8")
(def IDENTITY-POOL-ID "us-east-1:d6d3b034-1631-402b-b838-b44513e93ee0")
(def OAUTH-DOMAIN "logseq-prod.auth.us-east-1.amazoncognito.com")
(def PUBLISH-API-BASE "https://logseq.io"))
(do (def API-DOMAIN "api-dev.logseq.com")
(def COGNITO-IDP "https://cognito-idp.us-east-2.amazonaws.com/")
(def COGNITO-CLIENT-ID "1qi1uijg8b6ra70nejvbptis0q")
(def REGION "us-east-2")
(def USER-POOL-ID "us-east-2_kAqZcxIeM")
(def IDENTITY-POOL-ID "us-east-2:cc7d2ad3-84d0-4faf-98fe-628f6b52c0a5")
(def OAUTH-DOMAIN "logseq-test2.auth.us-east-2.amazoncognito.com")
(def PUBLISH-API-BASE "https://logseq-publish-staging.logseq.workers.dev")))
;; Enable for local development
;; (def PUBLISH-API-BASE "http://localhost:8787")
(goog-define ENABLE-DB-SYNC-LOCAL false)
(defonce db-sync-local? ENABLE-DB-SYNC-LOCAL)
(defonce default-db-sync-ws-url
(if db-sync-local?
"ws://127.0.0.1:8787/sync/%s"
;; "wss://api-staging.logseq.io/sync/%s"
"wss://api.logseq.io/sync/%s"))
(defonce default-db-sync-http-base
(if db-sync-local?
"http://127.0.0.1:8787"
;; "https://api-staging.logseq.io"
"https://api.logseq.io"
))
(defn get-custom-sync-server-url
"Read the user-configured custom sync server URL from localStorage.
Returns nil when not set or empty."
[]
(when-not util/node-test?
(let [v (.getItem js/localStorage "sync-server-url")]
(when (and (string? v) (not (string/blank? v)))
v))))
(defn set-custom-sync-server-url!
"Persist the custom sync server URL to localStorage. Pass nil or empty string to clear."
[url]
(when-not util/node-test?
(if (or (nil? url) (string/blank? url))
(.removeItem js/localStorage "sync-server-url")
(.setItem js/localStorage "sync-server-url" (string/trim url)))))
(defn valid-sync-server-url?
"Return true when `url` looks like a valid HTTP(S) base URL."
[url]
(and (string? url)
(re-find #"^https?://" url)))
(defn custom-url->ws-url
"Derive a WebSocket sync URL from a custom HTTP base URL. Pure function."
[custom-url]
(let [scheme (if (string/starts-with? custom-url "https") "wss" "ws")
base (-> custom-url
(string/replace #"^https?://" "")
(string/replace #"/+$" ""))]
(str scheme "://" base "/sync/%s")))
(defn custom-url->http-base
"Normalize a custom HTTP base URL by stripping trailing slashes. Pure function."
[custom-url]
(string/replace custom-url #"/+$" ""))
(defn db-sync-ws-url
"Return the WebSocket sync URL. Uses custom server when configured, otherwise the default."
[]
(if-let [custom (get-custom-sync-server-url)]
(custom-url->ws-url custom)
default-db-sync-ws-url))
(defn db-sync-http-base
"Return the HTTP base URL for sync. Uses custom server when configured, otherwise the default."
[]
(if-let [custom (get-custom-sync-server-url)]
(custom-url->http-base custom)
default-db-sync-http-base))
;; Feature flags
;; =============
(goog-define ENABLE-PLUGINS true)
(defonce feature-plugin-system-on? ENABLE-PLUGINS)
;; Desktop only as other platforms requires better understanding of their
;; multi-graph workflows and optimal place for a "global" dir
(def global-config-enabled? util/electron?)
;; User level configuration for whether plugins are enabled
(defonce lsp-enabled?
(and util/plugin-platform?
(not (false? feature-plugin-system-on?))
(state/lsp-enabled?-or-theme)))
(defn plugin-config-enabled?
[]
(and lsp-enabled? (global-config-enabled?)))
;; :TODO: How to do this?
;; (defonce desktop? ^boolean goog.DESKTOP)
;; ============
(def app-name common-config/app-name)
;; FIXME:
(def app-website
(if dev?
"http://localhost:3001"
(util/format "https://%s.com" app-name)))
(def markup-formats
#{:org :md :markdown :asciidoc :adoc :rst})
(def doc-formats
#{:doc :docx :xls :xlsx :ppt :pptx :one :pdf :epub})
(def image-formats
#{:png :jpg :jpeg :bmp :gif :webp :svg :heic})
(def audio-formats
#{:mp3 :ogg :mpeg :wav :m4a :flac :wma :aac})
(def video-formats
#{:mp4 :webm :mov :flv :avi :mkv})
(def media-formats (set/union (common-config/img-formats) audio-formats video-formats))
(def mobile?
"Triggering condition: Mobile phones
*** Warning!!! ***
For UX logic only! Don't use for FS logic
iPad / Android Pad doesn't trigger!
Same as config/mobile?"
(when-not util/node-test?
(util/safe-re-find #"Mobi" js/navigator.userAgent)))
(defn get-hr
[format]
(let [format (or format (keyword (state/get-preferred-format)))]
(case format
:markdown
"---"
"")))
(defn get-bold
[format]
(let [format (or format (keyword (state/get-preferred-format)))]
(case format
:markdown
"**"
"")))
(defn get-italic
[format]
(let [format (or format (keyword (state/get-preferred-format)))]
(case format
:markdown
"*"
"")))
(defn get-underline
[format]
(let [format (or format (keyword (state/get-preferred-format)))]
(case format
:markdown ;; no underline for markdown
""
"")))
(defn get-strike-through
[format]
(let [format (or format (keyword (state/get-preferred-format)))]
(case format
:markdown
"~~"
"")))
(defn get-highlight
[format]
(case format
:markdown
"=="
""))
(defn get-code
[format]
(let [format (or format (keyword (state/get-preferred-format)))]
(case format
:markdown
"`"
"")))
(defn get-empty-link-and-forward-pos
[format]
(case format
:markdown
["[]()" 1]
["" 0]))
(defn link-format
[label link]
(if (not-empty label)
(util/format "[%s](%s)" label link)
link))
(defn with-default-link
[format link]
(case format
:markdown
[(util/format "[](%s)" link)
1]
["" 0]))
(defn with-label-link
[format label link]
(case format
:markdown
[(util/format "[%s](%s)" label link)
(+ 4 (count link) (count label))]
["" 0]))
(defn with-default-label
[format label]
(case format
:markdown
[(util/format "[%s]()" label)
(+ 3 (count label))]
["" 0]))
(defonce demo-repo "Demo")
(defn demo-graph?
"Demo graph or nil graph?"
([]
(demo-graph? (state/get-current-repo)))
([repo-url]
(or (nil? repo-url) (= repo-url demo-repo)
(string/ends-with? repo-url demo-repo))))
(def config-file "config.edn")
(def custom-css-file "custom.css")
(def export-css-file "export.css")
(def custom-js-file "custom.js")
(def config-default-content (rc/inline "templates/config.edn"))
;; NOTE: repo-url is the unique identifier of a repo.
;; - `logseq_db_GraphName` => db based graph, sqlite as backend
;; - Use `""` while writing global files
(defonce db-version-prefix common-config/db-version-prefix)
(defn db-graph-name
[repo-with-prefix]
(string/replace-first repo-with-prefix db-version-prefix ""))
(defn db-based-graph?
([]
(db-based-graph? (state/get-current-repo)))
([s]
(boolean
(and (string? s)
(sqlite-util/db-based-graph? s)))))
(defn get-local-asset-absolute-path
[s]
(str "/" (string/replace s #"^[./]*" "")))
(defn get-local-dir
[repo]
(path/path-join (get-in @state/state [:system/info :home-dir])
"logseq"
"graphs"
(string/replace repo db-version-prefix "")))
(defn get-electron-backup-dir
[repo]
(path/path-join (get-local-dir repo) "backups"))
(defn get-repo-dir
[repo-url]
(when repo-url
(if (util/electron?)
(get-local-dir repo-url)
(str "memory:///"
(string/replace-first repo-url db-version-prefix "")))))
(defn get-repo-config-path
[]
(path/path-join app-name config-file))
(defn get-custom-css-path
([]
(get-custom-css-path (state/get-current-repo)))
([repo]
(if (db-based-graph? repo)
(path/path-join app-name custom-css-file)
(when-let [repo-dir (get-repo-dir repo)]
(path/path-join repo-dir app-name custom-css-file)))))
(defn get-export-css-path
([]
(get-export-css-path (state/get-current-repo)))
([repo]
(when-let [repo-dir (get-repo-dir repo)]
(path/path-join repo-dir app-name export-css-file))))
(defn get-current-repo-assets-root
[]
(when-let [repo-dir (get-repo-dir (state/get-current-repo))]
(path/path-join repo-dir "assets")))
(defn get-repo-assets-root
[repo]
(when-let [repo-dir (get-repo-dir repo)]
(path/path-join repo-dir "assets")))
(defn get-custom-js-path
([]
(get-custom-js-path (state/get-current-repo)))
([repo]
(if (db-based-graph? repo)
(path/path-join app-name custom-js-file)
(when-let [repo-dir (get-repo-dir repo)]
(path/path-join repo-dir app-name custom-js-file)))))