2020-05-27 10:03:00 +02:00
|
|
|
;; 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/.
|
|
|
|
;;
|
2022-09-20 23:23:22 +02:00
|
|
|
;; Copyright (c) KALEIDOS INC
|
2020-05-27 10:03:00 +02:00
|
|
|
|
2022-11-08 10:40:19 +01:00
|
|
|
(ns backend-tests.rpc-file-test
|
2020-02-17 09:49:04 +01:00
|
|
|
(:require
|
2020-08-18 19:26:37 +02:00
|
|
|
[app.common.uuid :as uuid]
|
|
|
|
[app.db :as db]
|
2022-03-23 10:59:20 +01:00
|
|
|
[app.db.sql :as sql]
|
2020-08-18 19:26:37 +02:00
|
|
|
[app.http :as http]
|
2023-01-02 15:49:08 +01:00
|
|
|
[app.rpc :as-alias rpc]
|
2021-01-31 20:55:53 +01:00
|
|
|
[app.storage :as sto]
|
2021-06-07 16:51:09 +02:00
|
|
|
[app.util.time :as dt]
|
2023-01-02 15:49:08 +01:00
|
|
|
[backend-tests.helpers :as th]
|
2021-01-31 20:55:53 +01:00
|
|
|
[clojure.test :as t]
|
|
|
|
[datoteka.core :as fs]))
|
2020-02-17 09:49:04 +01:00
|
|
|
|
|
|
|
(t/use-fixtures :once th/state-init)
|
|
|
|
(t/use-fixtures :each th/database-reset)
|
|
|
|
|
|
|
|
(t/deftest files-crud
|
2021-01-31 20:55:53 +01:00
|
|
|
(let [prof (th/create-profile* 1 {:is-active true})
|
2020-05-27 09:33:09 +02:00
|
|
|
team-id (:default-team-id prof)
|
|
|
|
proj-id (:default-project-id prof)
|
2020-02-17 09:49:04 +01:00
|
|
|
file-id (uuid/next)
|
|
|
|
page-id (uuid/next)]
|
|
|
|
|
|
|
|
(t/testing "create file"
|
2020-12-24 14:32:19 +01:00
|
|
|
(let [data {::th/type :create-file
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id prof)
|
2020-05-14 13:49:11 +02:00
|
|
|
:project-id proj-id
|
2020-02-17 09:49:04 +01:00
|
|
|
:id file-id
|
2020-12-24 14:32:19 +01:00
|
|
|
:name "foobar"
|
2022-07-12 13:52:48 +02:00
|
|
|
:is-shared false
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)]
|
2020-05-14 13:49:11 +02:00
|
|
|
|
2020-02-17 09:49:04 +01:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= (:name data) (:name result)))
|
2020-05-14 13:49:11 +02:00
|
|
|
(t/is (= proj-id (:project-id result))))))
|
2020-02-17 09:49:04 +01:00
|
|
|
|
|
|
|
(t/testing "rename file"
|
2020-12-24 14:32:19 +01:00
|
|
|
(let [data {::th/type :rename-file
|
2020-02-17 09:49:04 +01:00
|
|
|
:id file-id
|
|
|
|
:name "new name"
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id prof)}
|
|
|
|
out (th/command! data)]
|
2020-04-06 23:32:54 +02:00
|
|
|
|
2020-02-17 09:49:04 +01:00
|
|
|
;; (th/print-result! out)
|
2020-04-06 23:32:54 +02:00
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= (:id data) (:id result)))
|
|
|
|
(t/is (= (:name data) (:name result))))))
|
2020-02-17 09:49:04 +01:00
|
|
|
|
2021-05-07 11:36:34 +02:00
|
|
|
(t/testing "query files"
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-project-files
|
|
|
|
::rpc/profile-id (:id prof)
|
|
|
|
:project-id proj-id}
|
|
|
|
out (th/command! data)]
|
2021-05-07 11:36:34 +02:00
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= 1 (count result)))
|
|
|
|
(t/is (= file-id (get-in result [0 :id])))
|
|
|
|
(t/is (= "new name" (get-in result [0 :name]))))))
|
|
|
|
|
2020-02-17 09:49:04 +01:00
|
|
|
(t/testing "query single file without users"
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-file
|
|
|
|
::rpc/profile-id (:id prof)
|
2022-07-12 13:52:48 +02:00
|
|
|
:id file-id
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)]
|
2020-02-17 09:49:04 +01:00
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= file-id (:id result)))
|
|
|
|
(t/is (= "new name" (:name result)))
|
2020-09-07 10:56:42 +02:00
|
|
|
(t/is (= 1 (count (get-in result [:data :pages]))))
|
2020-02-17 09:49:04 +01:00
|
|
|
(t/is (nil? (:users result))))))
|
|
|
|
|
|
|
|
(t/testing "delete file"
|
2020-12-24 14:32:19 +01:00
|
|
|
(let [data {::th/type :delete-file
|
2020-02-17 09:49:04 +01:00
|
|
|
:id file-id
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id prof)}
|
|
|
|
out (th/command! data)]
|
2020-02-17 09:49:04 +01:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(t/is (nil? (:result out)))))
|
|
|
|
|
|
|
|
(t/testing "query single file after delete"
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-file
|
|
|
|
::rpc/profile-id (:id prof)
|
2022-07-12 13:52:48 +02:00
|
|
|
:id file-id
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)]
|
2020-02-17 09:49:04 +01:00
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
|
2020-12-24 14:32:19 +01:00
|
|
|
(let [error (:error out)
|
2020-02-17 09:49:04 +01:00
|
|
|
error-data (ex-data error)]
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (= (:type error-data) :not-found)))))
|
|
|
|
|
|
|
|
(t/testing "query list files after delete"
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-project-files
|
|
|
|
::rpc/profile-id (:id prof)
|
|
|
|
:project-id proj-id}
|
|
|
|
out (th/command! data)]
|
2020-02-17 09:49:04 +01:00
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= 0 (count result))))))
|
|
|
|
))
|
2021-01-31 20:55:53 +01:00
|
|
|
|
2023-04-25 17:14:38 +02:00
|
|
|
(t/deftest file-gc-with-fragments
|
|
|
|
(letfn [(update-file! [& {:keys [profile-id file-id changes revn] :or {revn 0}}]
|
|
|
|
(let [params {::th/type :update-file
|
|
|
|
::rpc/profile-id profile-id
|
|
|
|
:id file-id
|
|
|
|
:session-id (uuid/random)
|
|
|
|
:revn revn
|
|
|
|
:components-v2 true
|
|
|
|
:changes changes}
|
|
|
|
out (th/command! params)]
|
2023-03-18 10:32:26 +01:00
|
|
|
;; (th/print-result! out)
|
2023-04-25 17:14:38 +02:00
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(:result out)))]
|
|
|
|
|
|
|
|
(let [profile (th/create-profile* 1)
|
|
|
|
file (th/create-file* 1 {:profile-id (:id profile)
|
|
|
|
:project-id (:default-project-id profile)
|
|
|
|
:is-shared false})
|
|
|
|
|
|
|
|
page-id (uuid/random)
|
|
|
|
shape-id (uuid/random)]
|
|
|
|
|
|
|
|
;; Preventive file-gc
|
|
|
|
(let [res (th/run-task! "file-gc" {:min-age 0})]
|
|
|
|
(t/is (= 1 (:processed res))))
|
|
|
|
|
|
|
|
;; Check the number of fragments before adding the page
|
|
|
|
(let [rows (th/db-query :file-data-fragment {:file-id (:id file)})]
|
|
|
|
(t/is (= 1 (count rows))))
|
|
|
|
|
|
|
|
;; Add page
|
|
|
|
(update-file!
|
|
|
|
:file-id (:id file)
|
|
|
|
:profile-id (:id profile)
|
|
|
|
:revn 0
|
|
|
|
:changes
|
|
|
|
[{:type :add-page
|
|
|
|
:name "test"
|
|
|
|
:id page-id}])
|
|
|
|
|
|
|
|
;; Check the number of fragments
|
|
|
|
(let [rows (th/db-query :file-data-fragment {:file-id (:id file)})]
|
|
|
|
(t/is (= 2 (count rows))))
|
|
|
|
|
|
|
|
;; Check the number of fragments
|
|
|
|
(let [rows (th/db-query :file-data-fragment {:file-id (:id file)})]
|
|
|
|
(t/is (= 2 (count rows))))
|
|
|
|
|
|
|
|
;; The file-gc should remove unused fragments
|
|
|
|
(let [res (th/run-task! "file-gc" {:min-age 0})]
|
|
|
|
(t/is (= 1 (:processed res))))
|
|
|
|
|
|
|
|
|
|
|
|
;; Add shape to page that should add a new fragment
|
|
|
|
(update-file!
|
|
|
|
:file-id (:id file)
|
|
|
|
:profile-id (:id profile)
|
|
|
|
:revn 0
|
|
|
|
:changes
|
|
|
|
[{:type :add-obj
|
|
|
|
:page-id page-id
|
|
|
|
:id shape-id
|
|
|
|
:parent-id uuid/zero
|
|
|
|
:frame-id uuid/zero
|
|
|
|
:components-v2 true
|
|
|
|
:obj {:id shape-id
|
|
|
|
:name "image"
|
|
|
|
:frame-id uuid/zero
|
|
|
|
:parent-id uuid/zero
|
|
|
|
:type :rect}}])
|
|
|
|
|
|
|
|
;; Check the number of fragments
|
|
|
|
(let [rows (th/db-query :file-data-fragment {:file-id (:id file)})]
|
|
|
|
(t/is (= 3 (count rows))))
|
|
|
|
|
|
|
|
;; The file-gc should remove unused fragments
|
|
|
|
(let [res (th/run-task! "file-gc" {:min-age 0})]
|
|
|
|
(t/is (= 1 (:processed res))))
|
|
|
|
|
|
|
|
;; Check the number of fragments; should be 3 because changes
|
|
|
|
;; are also holding pointers to fragments;
|
|
|
|
(let [rows (th/db-query :file-data-fragment {:file-id (:id file)})]
|
|
|
|
(t/is (= 3 (count rows))))
|
|
|
|
|
|
|
|
;; Lets proceed to delete all changes
|
|
|
|
(th/db-delete! :file-change {:file-id (:id file)})
|
|
|
|
(th/db-update! :file
|
|
|
|
{:has-media-trimmed false}
|
|
|
|
{:id (:id file)})
|
|
|
|
|
|
|
|
|
|
|
|
;; The file-gc should remove fragments related to changes
|
|
|
|
;; snapshots previously deleted.
|
|
|
|
(let [res (th/run-task! "file-gc" {:min-age 0})]
|
|
|
|
(t/is (= 1 (:processed res))))
|
|
|
|
|
|
|
|
;; Check the number of fragments;
|
|
|
|
(let [rows (th/db-query :file-data-fragment {:file-id (:id file)})]
|
|
|
|
(t/is (= 2 (count rows))))
|
|
|
|
|
|
|
|
)))
|
|
|
|
|
|
|
|
|
|
|
|
(t/deftest file-gc-task-with-thumbnails
|
|
|
|
(letfn [(add-file-media-object [& {:keys [profile-id file-id]}]
|
2021-02-25 17:45:39 +01:00
|
|
|
(let [mfile {:filename "sample.jpg"
|
2022-11-08 10:40:19 +01:00
|
|
|
:path (th/tempfile "backend_tests/test_files/sample.jpg")
|
2022-03-04 18:00:16 +01:00
|
|
|
:mtype "image/jpeg"
|
2021-02-25 17:45:39 +01:00
|
|
|
:size 312043}
|
|
|
|
params {::th/type :upload-file-media-object
|
2023-04-24 20:18:14 +02:00
|
|
|
::rpc/profile-id profile-id
|
2021-02-25 17:45:39 +01:00
|
|
|
:file-id file-id
|
|
|
|
:is-local true
|
|
|
|
:name "testfile"
|
|
|
|
:content mfile}
|
2023-04-24 20:18:14 +02:00
|
|
|
out (th/command! params)]
|
2022-03-23 10:59:20 +01:00
|
|
|
|
|
|
|
;; (th/print-result! out)
|
2021-02-25 17:45:39 +01:00
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(:result out)))
|
|
|
|
|
2023-04-25 17:14:38 +02:00
|
|
|
(update-file! [& {:keys [profile-id file-id changes revn] :or {revn 0}}]
|
2021-02-25 17:45:39 +01:00
|
|
|
(let [params {::th/type :update-file
|
2023-01-02 17:31:47 +01:00
|
|
|
::rpc/profile-id profile-id
|
2021-02-25 17:45:39 +01:00
|
|
|
:id file-id
|
|
|
|
:session-id (uuid/random)
|
|
|
|
:revn revn
|
2022-07-12 13:52:48 +02:00
|
|
|
:components-v2 true
|
2021-02-25 17:45:39 +01:00
|
|
|
:changes changes}
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! params)]
|
2023-07-04 13:36:14 +02:00
|
|
|
;; (th/print-result! out)
|
2021-02-25 17:45:39 +01:00
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(:result out)))]
|
|
|
|
|
|
|
|
(let [storage (:app.storage/storage th/*system*)
|
|
|
|
|
|
|
|
profile (th/create-profile* 1)
|
|
|
|
file (th/create-file* 1 {:profile-id (:id profile)
|
|
|
|
:project-id (:default-project-id profile)
|
|
|
|
:is-shared false})
|
|
|
|
|
2023-04-25 17:14:38 +02:00
|
|
|
fmo1 (add-file-media-object :profile-id (:id profile) :file-id (:id file))
|
|
|
|
fmo2 (add-file-media-object :profile-id (:id profile) :file-id (:id file))
|
2021-02-25 17:45:39 +01:00
|
|
|
shid (uuid/random)
|
|
|
|
|
2023-04-25 17:14:38 +02:00
|
|
|
page-id (first (get-in file [:data :pages]))]
|
|
|
|
|
|
|
|
|
|
|
|
;; Update file inserting a new image object
|
|
|
|
(update-file!
|
|
|
|
:file-id (:id file)
|
|
|
|
:profile-id (:id profile)
|
|
|
|
:revn 0
|
|
|
|
:changes
|
|
|
|
[{:type :add-obj
|
|
|
|
:page-id page-id
|
|
|
|
:id shid
|
2023-07-04 13:36:14 +02:00
|
|
|
:parent-id uuid/zero
|
2023-04-25 17:14:38 +02:00
|
|
|
:frame-id uuid/zero
|
|
|
|
:components-v2 true
|
|
|
|
:obj {:id shid
|
|
|
|
:name "image"
|
|
|
|
:frame-id uuid/zero
|
|
|
|
:parent-id uuid/zero
|
|
|
|
:type :image
|
2023-07-04 13:36:14 +02:00
|
|
|
:metadata {:id (:id fmo1) :width 200 :height 200 :mtype "image/jpeg"}}}])
|
2021-02-25 17:45:39 +01:00
|
|
|
|
:wrench: Fix typos in source code
Found via `codespell -q 3 -S *.po,./frontend/yarn.lock -L childs,clen,fpr,inflight,ody,ot,ro,te,trys,ue`
2022-10-02 14:00:19 -04:00
|
|
|
;; Check that reference storage objects on filemediaobjects
|
2022-02-28 17:15:58 +01:00
|
|
|
;; are the same because of deduplication feature.
|
|
|
|
(t/is (= (:media-id fmo1) (:media-id fmo2)))
|
|
|
|
(t/is (= (:thumbnail-id fmo1) (:thumbnail-id fmo2)))
|
|
|
|
|
|
|
|
;; If we launch gc-touched-task, we should have 2 items to
|
|
|
|
;; freeze because of the deduplication (we have uploaded 2 times
|
2023-04-25 17:14:38 +02:00
|
|
|
;; the same files).
|
|
|
|
|
2022-02-10 19:50:40 +01:00
|
|
|
(let [task (:app.storage/gc-touched-task th/*system*)
|
2022-08-12 08:34:23 +02:00
|
|
|
res (task {:min-age (dt/duration 0)})]
|
2022-02-28 17:15:58 +01:00
|
|
|
(t/is (= 2 (:freeze res)))
|
2022-02-10 19:50:40 +01:00
|
|
|
(t/is (= 0 (:delete res))))
|
|
|
|
|
2022-08-12 08:34:23 +02:00
|
|
|
;; run the file-gc task immediately without forced min-age
|
2023-04-25 17:14:38 +02:00
|
|
|
(let [res (th/run-task! "file-gc")]
|
2021-02-25 17:45:39 +01:00
|
|
|
(t/is (= 0 (:processed res))))
|
|
|
|
|
|
|
|
;; run the task again
|
2023-04-25 17:14:38 +02:00
|
|
|
(let [res (th/run-task! "file-gc" {:min-age 0})]
|
2021-02-25 17:45:39 +01:00
|
|
|
(t/is (= 1 (:processed res))))
|
|
|
|
|
|
|
|
;; retrieve file and check trimmed attribute
|
2023-04-25 17:14:38 +02:00
|
|
|
(let [row (th/db-get :file {:id (:id file)})]
|
2021-02-25 17:45:39 +01:00
|
|
|
(t/is (true? (:has-media-trimmed row))))
|
|
|
|
|
|
|
|
;; check file media objects
|
2023-04-25 17:14:38 +02:00
|
|
|
(let [rows (th/db-exec! ["select * from file_media_object where file_id = ?" (:id file)])]
|
2021-02-25 17:45:39 +01:00
|
|
|
(t/is (= 1 (count rows))))
|
|
|
|
|
|
|
|
;; The underlying storage objects are still available.
|
2023-03-03 14:05:26 +01:00
|
|
|
(t/is (some? (sto/get-object storage (:media-id fmo2))))
|
|
|
|
(t/is (some? (sto/get-object storage (:thumbnail-id fmo2))))
|
|
|
|
(t/is (some? (sto/get-object storage (:media-id fmo1))))
|
|
|
|
(t/is (some? (sto/get-object storage (:thumbnail-id fmo1))))
|
2021-02-25 17:45:39 +01:00
|
|
|
|
2022-08-12 08:34:23 +02:00
|
|
|
;; proceed to remove usage of the file
|
2023-04-25 17:14:38 +02:00
|
|
|
(update-file!
|
|
|
|
:file-id (:id file)
|
|
|
|
:profile-id (:id profile)
|
|
|
|
:revn 0
|
|
|
|
:changes [{:type :del-obj
|
|
|
|
:page-id (first (get-in file [:data :pages]))
|
|
|
|
:id shid}])
|
2022-08-12 08:34:23 +02:00
|
|
|
|
:wrench: Fix typos in source code
Found via `codespell -q 3 -S *.po,./frontend/yarn.lock -L childs,clen,fpr,inflight,ody,ot,ro,te,trys,ue`
2022-10-02 14:00:19 -04:00
|
|
|
;; Now, we have deleted the usage of pointers to the
|
|
|
|
;; file-media-objects, if we paste file-gc, they should be marked
|
2022-08-12 08:34:23 +02:00
|
|
|
;; as deleted.
|
|
|
|
(let [task (:app.tasks.file-gc/handler th/*system*)
|
|
|
|
res (task {:min-age (dt/duration 0)})]
|
|
|
|
(t/is (= 1 (:processed res))))
|
|
|
|
|
|
|
|
;; Now that file-gc have deleted the file-media-object usage,
|
|
|
|
;; lets execute the touched-gc task, we should see that two of
|
|
|
|
;; them are marked to be deleted.
|
2021-02-25 17:45:39 +01:00
|
|
|
(let [task (:app.storage/gc-touched-task th/*system*)
|
2022-08-12 08:34:23 +02:00
|
|
|
res (task {:min-age (dt/duration 0)})]
|
|
|
|
(t/is (= 0 (:freeze res)))
|
|
|
|
(t/is (= 2 (:delete res))))
|
2021-02-25 17:45:39 +01:00
|
|
|
|
2022-02-10 19:50:40 +01:00
|
|
|
;; Finally, check that some of the objects that are marked as
|
|
|
|
;; deleted we are unable to retrieve them using standard storage
|
|
|
|
;; public api.
|
2023-03-03 14:05:26 +01:00
|
|
|
(t/is (nil? (sto/get-object storage (:media-id fmo2))))
|
|
|
|
(t/is (nil? (sto/get-object storage (:thumbnail-id fmo2))))
|
|
|
|
(t/is (nil? (sto/get-object storage (:media-id fmo1))))
|
|
|
|
(t/is (nil? (sto/get-object storage (:thumbnail-id fmo1))))
|
2021-02-25 17:45:39 +01:00
|
|
|
)))
|
|
|
|
|
|
|
|
(t/deftest permissions-checks-creating-file
|
|
|
|
(let [profile1 (th/create-profile* 1)
|
|
|
|
profile2 (th/create-profile* 2)
|
|
|
|
|
|
|
|
data {::th/type :create-file
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id profile2)
|
2021-02-25 17:45:39 +01:00
|
|
|
:project-id (:default-project-id profile1)
|
|
|
|
:name "foobar"
|
2022-07-12 13:52:48 +02:00
|
|
|
:is-shared false
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)
|
2021-02-25 17:45:39 +01:00
|
|
|
error (:error out)]
|
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :not-found))))
|
|
|
|
|
|
|
|
(t/deftest permissions-checks-rename-file
|
|
|
|
(let [profile1 (th/create-profile* 1)
|
|
|
|
profile2 (th/create-profile* 2)
|
|
|
|
|
|
|
|
file (th/create-file* 1 {:project-id (:default-project-id profile1)
|
|
|
|
:profile-id (:id profile1)})
|
|
|
|
data {::th/type :rename-file
|
|
|
|
:id (:id file)
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id profile2)
|
2021-02-25 17:45:39 +01:00
|
|
|
:name "foobar"}
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)
|
2021-02-25 17:45:39 +01:00
|
|
|
error (:error out)]
|
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :not-found))))
|
|
|
|
|
|
|
|
(t/deftest permissions-checks-delete-file
|
|
|
|
(let [profile1 (th/create-profile* 1)
|
|
|
|
profile2 (th/create-profile* 2)
|
|
|
|
|
|
|
|
file (th/create-file* 1 {:project-id (:default-project-id profile1)
|
|
|
|
:profile-id (:id profile1)})
|
|
|
|
data {::th/type :delete-file
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id profile2)
|
2021-02-25 17:45:39 +01:00
|
|
|
:id (:id file)}
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)
|
2021-02-25 17:45:39 +01:00
|
|
|
error (:error out)]
|
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :not-found))))
|
|
|
|
|
|
|
|
(t/deftest permissions-checks-set-file-shared
|
|
|
|
(let [profile1 (th/create-profile* 1)
|
|
|
|
profile2 (th/create-profile* 2)
|
|
|
|
file (th/create-file* 1 {:project-id (:default-project-id profile1)
|
|
|
|
:profile-id (:id profile1)})
|
|
|
|
data {::th/type :set-file-shared
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id profile2)
|
2021-02-25 17:45:39 +01:00
|
|
|
:id (:id file)
|
|
|
|
:is-shared true}
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)
|
2021-02-25 17:45:39 +01:00
|
|
|
error (:error out)]
|
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :not-found))))
|
|
|
|
|
|
|
|
(t/deftest permissions-checks-link-to-library-1
|
|
|
|
(let [profile1 (th/create-profile* 1)
|
|
|
|
profile2 (th/create-profile* 2)
|
|
|
|
file1 (th/create-file* 1 {:project-id (:default-project-id profile1)
|
|
|
|
:profile-id (:id profile1)
|
|
|
|
:is-shared true})
|
|
|
|
file2 (th/create-file* 2 {:project-id (:default-project-id profile1)
|
|
|
|
:profile-id (:id profile1)})
|
|
|
|
|
|
|
|
data {::th/type :link-file-to-library
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id profile2)
|
2021-02-25 17:45:39 +01:00
|
|
|
:file-id (:id file2)
|
|
|
|
:library-id (:id file1)}
|
|
|
|
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)
|
2021-02-25 17:45:39 +01:00
|
|
|
error (:error out)]
|
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :not-found))))
|
|
|
|
|
|
|
|
(t/deftest permissions-checks-link-to-library-2
|
|
|
|
(let [profile1 (th/create-profile* 1)
|
|
|
|
profile2 (th/create-profile* 2)
|
|
|
|
file1 (th/create-file* 1 {:project-id (:default-project-id profile1)
|
|
|
|
:profile-id (:id profile1)
|
|
|
|
:is-shared true})
|
|
|
|
|
|
|
|
file2 (th/create-file* 2 {:project-id (:default-project-id profile2)
|
|
|
|
:profile-id (:id profile2)})
|
|
|
|
|
|
|
|
data {::th/type :link-file-to-library
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id profile2)
|
2021-02-25 17:45:39 +01:00
|
|
|
:file-id (:id file2)
|
|
|
|
:library-id (:id file1)}
|
|
|
|
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)
|
2021-02-25 17:45:39 +01:00
|
|
|
error (:error out)]
|
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :not-found))))
|
2021-01-31 20:55:53 +01:00
|
|
|
|
2022-03-23 10:59:20 +01:00
|
|
|
(t/deftest deletion
|
2021-06-07 16:51:09 +02:00
|
|
|
(let [task (:app.tasks.objects-gc/handler th/*system*)
|
|
|
|
profile1 (th/create-profile* 1)
|
|
|
|
file (th/create-file* 1 {:project-id (:default-project-id profile1)
|
|
|
|
:profile-id (:id profile1)})]
|
|
|
|
;; file is not deleted because it does not meet all
|
|
|
|
;; conditions to be deleted.
|
2022-08-12 08:34:23 +02:00
|
|
|
(let [result (task {:min-age (dt/duration 0)})]
|
|
|
|
(t/is (= 0 (:processed result))))
|
2021-06-07 16:51:09 +02:00
|
|
|
|
|
|
|
;; query the list of files
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-project-files
|
|
|
|
::rpc/profile-id (:id profile1)
|
|
|
|
:project-id (:default-project-id profile1)}
|
|
|
|
out (th/command! data)]
|
2021-06-07 16:51:09 +02:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= 1 (count result)))))
|
|
|
|
|
|
|
|
;; Request file to be deleted
|
|
|
|
(let [params {::th/type :delete-file
|
|
|
|
:id (:id file)
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id profile1)}
|
|
|
|
out (th/command! params)]
|
2021-06-07 16:51:09 +02:00
|
|
|
(t/is (nil? (:error out))))
|
|
|
|
|
|
|
|
;; query the list of files after soft deletion
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-project-files
|
|
|
|
::rpc/profile-id (:id profile1)
|
|
|
|
:project-id (:default-project-id profile1)}
|
|
|
|
out (th/command! data)]
|
2021-06-07 16:51:09 +02:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= 0 (count result)))))
|
|
|
|
|
|
|
|
;; run permanent deletion (should be noop)
|
2022-08-12 08:34:23 +02:00
|
|
|
(let [result (task {:min-age (dt/duration {:minutes 1})})]
|
|
|
|
(t/is (= 0 (:processed result))))
|
2021-06-07 16:51:09 +02:00
|
|
|
|
|
|
|
;; query the list of file libraries of a after hard deletion
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-file-libraries
|
|
|
|
::rpc/profile-id (:id profile1)
|
|
|
|
:file-id (:id file)}
|
|
|
|
out (th/command! data)]
|
2021-06-07 16:51:09 +02:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= 0 (count result)))))
|
|
|
|
|
|
|
|
;; run permanent deletion
|
2022-08-12 08:34:23 +02:00
|
|
|
(let [result (task {:min-age (dt/duration 0)})]
|
|
|
|
(t/is (= 1 (:processed result))))
|
2021-06-07 16:51:09 +02:00
|
|
|
|
|
|
|
;; query the list of file libraries of a after hard deletion
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-file-libraries
|
|
|
|
::rpc/profile-id (:id profile1)
|
|
|
|
:file-id (:id file)}
|
|
|
|
out (th/command! data)]
|
2021-06-07 16:51:09 +02:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(let [error (:error out)
|
|
|
|
error-data (ex-data error)]
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (= (:type error-data) :not-found))))
|
|
|
|
))
|
2022-02-08 13:25:25 +01:00
|
|
|
|
|
|
|
|
2022-03-30 00:11:43 +02:00
|
|
|
(t/deftest object-thumbnails-ops
|
2022-03-23 10:59:20 +01:00
|
|
|
(let [prof (th/create-profile* 1 {:is-active true})
|
|
|
|
file (th/create-file* 1 {:profile-id (:id prof)
|
|
|
|
:project-id (:default-project-id prof)
|
|
|
|
:is-shared false})
|
2022-03-30 00:11:43 +02:00
|
|
|
page-id (get-in file [:data :pages 0])
|
|
|
|
frame1-id (uuid/next)
|
|
|
|
shape1-id (uuid/next)
|
|
|
|
frame2-id (uuid/next)
|
|
|
|
shape2-id (uuid/next)
|
|
|
|
|
|
|
|
changes [{:type :add-obj
|
|
|
|
:page-id page-id
|
|
|
|
:id frame1-id
|
|
|
|
:parent-id uuid/zero
|
|
|
|
:frame-id uuid/zero
|
|
|
|
:obj {:id frame1-id
|
|
|
|
:use-for-thumbnail? true
|
|
|
|
:name "test-frame1"
|
|
|
|
:type :frame}}
|
|
|
|
{:type :add-obj
|
|
|
|
:page-id page-id
|
|
|
|
:id shape1-id
|
|
|
|
:parent-id frame1-id
|
|
|
|
:frame-id frame1-id
|
|
|
|
:obj {:id shape1-id
|
|
|
|
:name "test-shape1"
|
|
|
|
:type :rect}}
|
|
|
|
{:type :add-obj
|
|
|
|
:page-id page-id
|
|
|
|
:id frame2-id
|
|
|
|
:parent-id uuid/zero
|
|
|
|
:frame-id uuid/zero
|
|
|
|
:obj {:id frame2-id
|
|
|
|
:name "test-frame2"
|
|
|
|
:type :frame}}
|
|
|
|
{:type :add-obj
|
|
|
|
:page-id page-id
|
|
|
|
:id shape2-id
|
|
|
|
:parent-id frame2-id
|
|
|
|
:frame-id frame2-id
|
|
|
|
:obj {:id shape2-id
|
|
|
|
:name "test-shape2"
|
|
|
|
:type :rect}}]]
|
|
|
|
;; Update the file
|
|
|
|
(th/update-file* {:file-id (:id file)
|
|
|
|
:profile-id (:id prof)
|
|
|
|
:revn 0
|
2022-07-12 13:52:48 +02:00
|
|
|
:components-v2 true
|
2022-03-30 00:11:43 +02:00
|
|
|
:changes changes})
|
|
|
|
|
|
|
|
(t/testing "RPC page query (rendering purposes)"
|
|
|
|
|
|
|
|
;; Query :page RPC method without passing page-id
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-page
|
|
|
|
::rpc/profile-id (:id prof)
|
2022-07-12 13:52:48 +02:00
|
|
|
:file-id (:id file)
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [error result] :as out} (th/command! data)]
|
2022-02-08 13:25:25 +01:00
|
|
|
|
2022-03-30 00:11:43 +02:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (map? result))
|
|
|
|
(t/is (contains? result :objects))
|
|
|
|
(t/is (contains? (:objects result) frame1-id))
|
|
|
|
(t/is (contains? (:objects result) shape1-id))
|
|
|
|
(t/is (contains? (:objects result) frame2-id))
|
|
|
|
(t/is (contains? (:objects result) shape2-id))
|
2022-11-01 09:46:54 +01:00
|
|
|
(t/is (contains? (:objects result) uuid/zero))
|
|
|
|
)
|
2022-03-30 00:11:43 +02:00
|
|
|
|
|
|
|
;; Query :page RPC method with page-id
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-page
|
|
|
|
::rpc/profile-id (:id prof)
|
2022-03-30 00:11:43 +02:00
|
|
|
:file-id (:id file)
|
2022-07-12 13:52:48 +02:00
|
|
|
:page-id page-id
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [error result] :as out} (th/command! data)]
|
2022-03-30 00:11:43 +02:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (map? result))
|
|
|
|
(t/is (contains? result :objects))
|
|
|
|
(t/is (contains? (:objects result) frame1-id))
|
|
|
|
(t/is (contains? (:objects result) shape1-id))
|
|
|
|
(t/is (contains? (:objects result) frame2-id))
|
|
|
|
(t/is (contains? (:objects result) shape2-id))
|
|
|
|
(t/is (contains? (:objects result) uuid/zero)))
|
|
|
|
|
|
|
|
;; Query :page RPC method with page-id and object-id
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-page
|
|
|
|
::rpc/profile-id (:id prof)
|
2022-03-30 00:11:43 +02:00
|
|
|
:file-id (:id file)
|
|
|
|
:page-id page-id
|
2022-07-12 13:52:48 +02:00
|
|
|
:object-id frame1-id
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [error result] :as out} (th/command! data)]
|
2022-03-30 00:11:43 +02:00
|
|
|
;; (th/print-result! out)
|
2022-11-01 09:46:54 +01:00
|
|
|
(t/is (nil? error))
|
2022-03-30 00:11:43 +02:00
|
|
|
(t/is (map? result))
|
|
|
|
(t/is (contains? result :objects))
|
|
|
|
(t/is (contains? (:objects result) frame1-id))
|
|
|
|
(t/is (contains? (:objects result) shape1-id))
|
|
|
|
(t/is (not (contains? (:objects result) uuid/zero)))
|
|
|
|
(t/is (not (contains? (:objects result) frame2-id)))
|
|
|
|
(t/is (not (contains? (:objects result) shape2-id))))
|
|
|
|
|
|
|
|
;; Query :page RPC method with wrong params
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-page
|
|
|
|
::rpc/profile-id (:id prof)
|
2022-03-30 00:11:43 +02:00
|
|
|
:file-id (:id file)
|
2022-07-12 13:52:48 +02:00
|
|
|
:object-id frame1-id
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
out (th/command! data)]
|
2022-09-22 17:47:05 +02:00
|
|
|
|
2023-03-18 10:32:26 +01:00
|
|
|
;; (th/print-result! out)
|
2022-09-22 17:47:05 +02:00
|
|
|
(t/is (not (th/success? out)))
|
|
|
|
(let [{:keys [type code]} (-> out :error ex-data)]
|
|
|
|
(t/is (= :validation type))
|
2023-03-18 10:32:26 +01:00
|
|
|
(t/is (= :params-validation code))))
|
2022-11-01 09:46:54 +01:00
|
|
|
|
|
|
|
)
|
2022-02-08 13:25:25 +01:00
|
|
|
|
2022-03-30 00:11:43 +02:00
|
|
|
(t/testing "RPC :file-data-for-thumbnail"
|
|
|
|
;; Insert a thumbnail data for the frame-id
|
|
|
|
(let [data {::th/type :upsert-file-object-thumbnail
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id prof)
|
2022-03-30 00:11:43 +02:00
|
|
|
:file-id (:id file)
|
2022-05-09 12:35:34 +02:00
|
|
|
:object-id (str page-id frame1-id)
|
2022-03-30 00:11:43 +02:00
|
|
|
:data "random-data-1"}
|
2022-03-23 10:59:20 +01:00
|
|
|
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [error result] :as out} (th/command! data)]
|
2022-03-30 00:11:43 +02:00
|
|
|
(t/is (nil? error))
|
|
|
|
(t/is (nil? result)))
|
|
|
|
|
|
|
|
;; Check the result
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-file-data-for-thumbnail
|
|
|
|
::rpc/profile-id (:id prof)
|
2022-07-12 13:52:48 +02:00
|
|
|
:file-id (:id file)
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [error result] :as out} (th/command! data)]
|
2022-03-30 00:11:43 +02:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (map? result))
|
|
|
|
(t/is (contains? result :page))
|
|
|
|
(t/is (contains? result :revn))
|
|
|
|
(t/is (contains? result :file-id))
|
|
|
|
|
|
|
|
(t/is (= (:id file) (:file-id result)))
|
|
|
|
(t/is (= "random-data-1" (get-in result [:page :objects frame1-id :thumbnail])))
|
|
|
|
(t/is (= [] (get-in result [:page :objects frame1-id :shapes]))))
|
|
|
|
|
|
|
|
;; Delete thumbnail data
|
|
|
|
(let [data {::th/type :upsert-file-object-thumbnail
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id prof)
|
2022-03-30 00:11:43 +02:00
|
|
|
:file-id (:id file)
|
2022-05-09 12:35:34 +02:00
|
|
|
:object-id (str page-id frame1-id)
|
2022-03-30 00:11:43 +02:00
|
|
|
:data nil}
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [error result] :as out} (th/command! data)]
|
2022-12-21 11:53:56 +01:00
|
|
|
;; (th/print-result! out)
|
2022-03-30 00:11:43 +02:00
|
|
|
(t/is (nil? error))
|
|
|
|
(t/is (nil? result)))
|
|
|
|
|
|
|
|
;; Check the result
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [data {::th/type :get-file-data-for-thumbnail
|
|
|
|
::rpc/profile-id (:id prof)
|
2022-07-12 13:52:48 +02:00
|
|
|
:file-id (:id file)
|
|
|
|
:components-v2 true}
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [error result] :as out} (th/command! data)]
|
2022-03-30 00:11:43 +02:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (map? result))
|
|
|
|
(t/is (contains? result :page))
|
|
|
|
(t/is (contains? result :revn))
|
|
|
|
(t/is (contains? result :file-id))
|
|
|
|
(t/is (= (:id file) (:file-id result)))
|
|
|
|
(t/is (nil? (get-in result [:page :objects frame1-id :thumbnail])))
|
|
|
|
(t/is (not= [] (get-in result [:page :objects frame1-id :shapes])))))
|
|
|
|
|
|
|
|
(t/testing "TASK :file-gc"
|
|
|
|
|
|
|
|
;; insert object snapshot for known frame
|
|
|
|
(let [data {::th/type :upsert-file-object-thumbnail
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id prof)
|
2022-03-30 00:11:43 +02:00
|
|
|
:file-id (:id file)
|
2022-05-09 12:35:34 +02:00
|
|
|
:object-id (str page-id frame1-id)
|
2022-03-30 00:11:43 +02:00
|
|
|
:data "new-data"}
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [error result] :as out} (th/command! data)]
|
2022-03-30 00:11:43 +02:00
|
|
|
(t/is (nil? error))
|
|
|
|
(t/is (nil? result)))
|
|
|
|
|
|
|
|
;; Wait to file be ellegible for GC
|
|
|
|
(th/sleep 300)
|
|
|
|
|
|
|
|
;; run the task again
|
|
|
|
(let [task (:app.tasks.file-gc/handler th/*system*)
|
2022-08-12 08:34:23 +02:00
|
|
|
res (task {:min-age (dt/duration 0)})]
|
2022-03-30 00:11:43 +02:00
|
|
|
(t/is (= 1 (:processed res))))
|
|
|
|
|
|
|
|
;; check that object thumbnails are still here
|
|
|
|
(let [res (th/db-exec! ["select * from file_object_thumbnail"])]
|
|
|
|
(t/is (= 1 (count res)))
|
|
|
|
(t/is (= "new-data" (get-in res [0 :data]))))
|
|
|
|
|
|
|
|
;; insert object snapshot for for unknown frame
|
|
|
|
(let [data {::th/type :upsert-file-object-thumbnail
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id prof)
|
2022-03-30 00:11:43 +02:00
|
|
|
:file-id (:id file)
|
2022-05-09 12:35:34 +02:00
|
|
|
:object-id (str page-id (uuid/next))
|
2022-03-30 00:11:43 +02:00
|
|
|
:data "new-data-2"}
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [error result] :as out} (th/command! data)]
|
2022-03-30 00:11:43 +02:00
|
|
|
(t/is (nil? error))
|
|
|
|
(t/is (nil? result)))
|
|
|
|
|
|
|
|
;; Mark file as modified
|
|
|
|
(th/db-exec! ["update file set has_media_trimmed=false where id=?" (:id file)])
|
|
|
|
|
|
|
|
;; check that we have all object thumbnails
|
|
|
|
(let [res (th/db-exec! ["select * from file_object_thumbnail"])]
|
|
|
|
(t/is (= 2 (count res))))
|
|
|
|
|
|
|
|
;; run the task again
|
|
|
|
(let [task (:app.tasks.file-gc/handler th/*system*)
|
2022-08-12 08:34:23 +02:00
|
|
|
res (task {:min-age (dt/duration 0)})]
|
2022-03-30 00:11:43 +02:00
|
|
|
(t/is (= 1 (:processed res))))
|
2022-02-08 13:25:25 +01:00
|
|
|
|
2022-03-30 00:11:43 +02:00
|
|
|
;; check that the unknown frame thumbnail is deleted
|
|
|
|
(let [res (th/db-exec! ["select * from file_object_thumbnail"])]
|
|
|
|
(t/is (= 1 (count res)))
|
2023-01-02 22:56:24 +01:00
|
|
|
(t/is (= "new-data" (get-in res [0 :data])))))
|
|
|
|
|
|
|
|
))
|
2022-03-23 10:59:20 +01:00
|
|
|
|
|
|
|
|
|
|
|
(t/deftest file-thumbnail-ops
|
|
|
|
(let [prof (th/create-profile* 1 {:is-active true})
|
|
|
|
file (th/create-file* 1 {:profile-id (:id prof)
|
|
|
|
:project-id (:default-project-id prof)
|
|
|
|
:revn 2
|
|
|
|
:is-shared false})
|
2023-01-02 15:49:08 +01:00
|
|
|
data {::th/type :get-file-thumbnail
|
|
|
|
::rpc/profile-id (:id prof)
|
2022-03-23 10:59:20 +01:00
|
|
|
:file-id (:id file)}]
|
|
|
|
|
|
|
|
(t/testing "query a thumbnail with single revn"
|
|
|
|
|
|
|
|
;; insert an entry on the database with a test value for the thumbnail of this frame
|
|
|
|
(th/db-insert! :file-thumbnail
|
|
|
|
{:file-id (:file-id data)
|
|
|
|
:revn 1
|
|
|
|
:data "testvalue1"})
|
|
|
|
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [{:keys [result error] :as out} (th/command! data)]
|
2022-03-23 10:59:20 +01:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? error))
|
|
|
|
(t/is (= 4 (count result)))
|
|
|
|
(t/is (= "testvalue1" (:data result)))
|
|
|
|
(t/is (= 1 (:revn result)))))
|
|
|
|
|
|
|
|
(t/testing "query thumbnail with two revisions"
|
|
|
|
;; insert an entry on the database with a test value for the thumbnail of this frame
|
|
|
|
(th/db-insert! :file-thumbnail
|
|
|
|
{:file-id (:file-id data)
|
|
|
|
:revn 2
|
|
|
|
:data "testvalue2"})
|
|
|
|
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [{:keys [result error] :as out} (th/command! data)]
|
2022-03-23 10:59:20 +01:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? error))
|
|
|
|
(t/is (= 4 (count result)))
|
|
|
|
(t/is (= "testvalue2" (:data result)))
|
|
|
|
(t/is (= 2 (:revn result))))
|
|
|
|
|
|
|
|
;; Then query the specific revn
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [{:keys [result error] :as out} (th/command! (assoc data :revn 1))]
|
2022-03-23 10:59:20 +01:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? error))
|
|
|
|
(t/is (= 4 (count result)))
|
|
|
|
(t/is (= "testvalue1" (:data result)))
|
|
|
|
(t/is (= 1 (:revn result)))))
|
|
|
|
|
|
|
|
(t/testing "upsert file-thumbnail"
|
|
|
|
(let [data {::th/type :upsert-file-thumbnail
|
2023-01-02 15:49:08 +01:00
|
|
|
::rpc/profile-id (:id prof)
|
2022-03-23 10:59:20 +01:00
|
|
|
:file-id (:id file)
|
|
|
|
:data "foobar"
|
|
|
|
:props {:baz 1}
|
|
|
|
:revn 2}
|
2023-01-02 15:49:08 +01:00
|
|
|
{:keys [result error] :as out} (th/command! data)]
|
2022-03-23 10:59:20 +01:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? error))
|
|
|
|
(t/is (nil? result))))
|
|
|
|
|
|
|
|
(t/testing "query last result"
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [{:keys [result error] :as out} (th/command! data)]
|
2022-03-23 10:59:20 +01:00
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? error))
|
|
|
|
(t/is (= 4 (count result)))
|
|
|
|
(t/is (= "foobar" (:data result)))
|
|
|
|
(t/is (= {:baz 1} (:props result)))
|
|
|
|
(t/is (= 2 (:revn result)))))
|
|
|
|
|
|
|
|
(t/testing "gc task"
|
|
|
|
;; make the file eligible for GC waiting 300ms (configured
|
|
|
|
;; timeout for testing)
|
|
|
|
(th/sleep 300)
|
|
|
|
|
|
|
|
;; run the task again
|
|
|
|
(let [task (:app.tasks.file-gc/handler th/*system*)
|
2022-08-12 08:34:23 +02:00
|
|
|
res (task {:min-age (dt/duration 0)})]
|
2022-03-23 10:59:20 +01:00
|
|
|
(t/is (= 1 (:processed res))))
|
|
|
|
|
|
|
|
;; Then query the specific revn
|
2023-01-02 15:49:08 +01:00
|
|
|
(let [{:keys [result error] :as out} (th/command! (assoc data :revn 1))]
|
2022-09-21 14:20:26 +02:00
|
|
|
(t/is (th/ex-of-type? error :not-found))
|
|
|
|
(t/is (th/ex-of-code? error :file-thumbnail-not-found))))
|
2022-03-23 10:59:20 +01:00
|
|
|
))
|
|
|
|
|
|
|
|
|