mirror of
https://github.com/penpot/penpot.git
synced 2026-06-02 13:40:30 +00:00
186 lines
5.3 KiB
Clojure
186 lines
5.3 KiB
Clojure
;; This Source Code Form is subject to the terms of the Mozilla Public
|
|
;; License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
;; file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
;;
|
|
;; Copyright (c) KALEIDOS INC Sucursal en España SL
|
|
|
|
(ns app.srepl.helpers
|
|
"A main namespace for server repl."
|
|
(:refer-clojure :exclude [parse-uuid])
|
|
(:require
|
|
[app.binfile.common :as bfc]
|
|
[app.common.data :as d]
|
|
[app.common.files.migrations :as fmg]
|
|
[app.common.files.validate :as cfv]
|
|
[app.common.time :as ct]
|
|
[app.db :as db]
|
|
[app.features.file-snapshots :as fsnap]
|
|
[app.main :as main]))
|
|
|
|
(def ^:dynamic *system* nil)
|
|
|
|
(defn println!
|
|
[& params]
|
|
(locking println
|
|
(apply println params)))
|
|
|
|
(defn get-current-system
|
|
[]
|
|
*system*)
|
|
|
|
(defn parse-uuid
|
|
[v]
|
|
(if (string? v)
|
|
(d/parse-uuid v)
|
|
v))
|
|
|
|
(defn get-file
|
|
"Get the migrated data of one file."
|
|
([id]
|
|
(get-file (or *system* main/system) id))
|
|
([system id]
|
|
(db/run! system bfc/get-file id)))
|
|
|
|
(defn get-raw-file
|
|
"Get the migrated data of one file."
|
|
([id] (get-raw-file (or *system* main/system) id))
|
|
([system id]
|
|
(db/run! system
|
|
(fn [system]
|
|
(bfc/get-file system id :decode? false)))))
|
|
|
|
(defn update-team!
|
|
[system {:keys [id] :as team}]
|
|
(let [conn (db/get-connection system)
|
|
params (-> team
|
|
(update :features db/encode-pgarray conn "text")
|
|
(dissoc :id))]
|
|
(db/update! conn :team
|
|
params
|
|
{:id id})
|
|
team))
|
|
|
|
(def ^:private sql:get-and-lock-team-files
|
|
"SELECT f.id
|
|
FROM file AS f
|
|
JOIN project AS p ON (p.id = f.project_id)
|
|
WHERE p.team_id = ?
|
|
AND p.deleted_at IS NULL
|
|
AND f.deleted_at IS NULL
|
|
FOR UPDATE")
|
|
|
|
(defn get-team
|
|
[conn team-id]
|
|
(-> (db/get conn :team {:id team-id}
|
|
{::db/remove-deleted false
|
|
::db/check-deleted false})
|
|
(update :features db/decode-pgarray #{})))
|
|
|
|
(defn get-and-lock-team-files
|
|
[conn team-id]
|
|
(transduce (map :id) conj []
|
|
(db/plan conn [sql:get-and-lock-team-files team-id])))
|
|
|
|
(defn reset-file-data!
|
|
"Hardcode replace of the data of one file."
|
|
[system id data]
|
|
(db/tx-run! system
|
|
(fn [system]
|
|
(db/update! system :file
|
|
{:data data}
|
|
{:id id}))))
|
|
|
|
|
|
(def ^:private sql:snapshots-with-file
|
|
"WITH files AS (
|
|
SELECT f.id AS file_id,
|
|
(SELECT fc.id
|
|
FROM file_change AS fc
|
|
WHERE fc.label = ?
|
|
AND fc.file_id = f.id
|
|
ORDER BY fc.created_at DESC
|
|
LIMIT 1) AS id
|
|
FROM file AS f
|
|
) SELECT * FROM files
|
|
WHERE file_id = ANY(?)
|
|
AND id IS NOT NULL")
|
|
|
|
(defn search-file-snapshots
|
|
"Get a seq parirs of file-id and snapshot-id for a set of files
|
|
and specified label"
|
|
[conn file-ids label]
|
|
(db/exec! conn [sql:snapshots-with-file label
|
|
(db/create-array conn "uuid" file-ids)]))
|
|
|
|
(defn take-team-snapshot!
|
|
[system team-id label]
|
|
(let [conn (db/get-connection system)]
|
|
(->> (get-and-lock-team-files conn team-id)
|
|
(reduce (fn [result file-id]
|
|
(let [file (bfc/get-file system file-id :realize? true :lock-for-update? true)]
|
|
(fsnap/create! system file
|
|
{:label label
|
|
:created-by "admin"})
|
|
(inc result)))
|
|
0))))
|
|
|
|
(defn restore-team-snapshot!
|
|
[system team-id label]
|
|
(let [conn (db/get-connection system)
|
|
ids (->> (get-and-lock-team-files conn team-id)
|
|
(into #{}))
|
|
|
|
snap (search-file-snapshots conn ids label)
|
|
ids' (into #{} (map :file-id) snap)]
|
|
|
|
(when (not= ids ids')
|
|
(throw (RuntimeException. "no uniform snapshot available")))
|
|
|
|
(reduce (fn [result {:keys [file-id id]}]
|
|
(fsnap/restore! system file-id id)
|
|
(inc result))
|
|
0
|
|
snap)))
|
|
|
|
(defn mark-migrated!
|
|
"A helper that inserts an entry in the file migration table for make
|
|
file migrated for the specified migration label."
|
|
[system file-id label]
|
|
(db/insert! system :file-migration
|
|
{:file-id file-id
|
|
:name label}
|
|
{::db/return-keys false}))
|
|
|
|
(defn process-file!
|
|
[system file-id update-fn
|
|
& {:keys [::snapshot-label ::validate? ::with-libraries?]
|
|
:or {validate? true} :as opts}]
|
|
(let [file (bfc/get-file system file-id
|
|
:lock-for-update? true
|
|
:realize? true)
|
|
|
|
libs (when with-libraries?
|
|
(bfc/get-resolved-file-libraries system file))
|
|
|
|
file' (when file
|
|
(if with-libraries?
|
|
(update-fn file libs opts)
|
|
(update-fn file opts)))]
|
|
|
|
(when (and (some? file')
|
|
(or (fmg/migrated? file)
|
|
(not (identical? file file'))))
|
|
|
|
(when validate?
|
|
(cfv/validate-file-schema! file'))
|
|
|
|
(when (string? snapshot-label)
|
|
(fsnap/create! system file
|
|
{:label snapshot-label
|
|
:deleted-at (ct/in-future {:days 30})
|
|
:created-by "admin"}))
|
|
|
|
(let [file' (update file' :revn inc)]
|
|
(bfc/update-file! system file' opts)
|
|
true))))
|