2021-01-31 17:02:38 +01: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
|
2021-01-31 17:02:38 +01:00
|
|
|
|
2022-11-08 10:40:19 +01:00
|
|
|
(ns backend-tests.storage-test
|
2021-01-31 17:02:38 +01:00
|
|
|
(:require
|
|
|
|
[app.common.exceptions :as ex]
|
2022-02-10 19:50:40 +01:00
|
|
|
[app.common.uuid :as uuid]
|
2021-01-31 17:02:38 +01:00
|
|
|
[app.db :as db]
|
2023-04-24 20:18:14 +02:00
|
|
|
[app.rpc :as-alias rpc]
|
2021-01-31 17:02:38 +01:00
|
|
|
[app.storage :as sto]
|
|
|
|
[app.util.time :as dt]
|
2023-04-24 20:18:14 +02:00
|
|
|
[backend-tests.helpers :as th]
|
2021-01-31 17:02:38 +01:00
|
|
|
[clojure.test :as t]
|
|
|
|
[cuerdas.core :as str]
|
2024-02-14 09:31:59 +01:00
|
|
|
[datoteka.fs :as fs]
|
2022-08-31 13:29:43 +02:00
|
|
|
[datoteka.io :as io]
|
2021-01-31 17:02:38 +01:00
|
|
|
[mockery.core :refer [with-mocks]]))
|
|
|
|
|
|
|
|
(t/use-fixtures :once th/state-init)
|
|
|
|
(t/use-fixtures :each (th/serial
|
|
|
|
th/database-reset
|
|
|
|
th/clean-storage))
|
|
|
|
|
2022-02-10 19:50:40 +01:00
|
|
|
(defn configure-storage-backend
|
|
|
|
"Given storage map, returns a storage configured with the appropriate
|
|
|
|
backend for assets."
|
|
|
|
([storage]
|
2023-02-06 12:27:53 +01:00
|
|
|
(assoc storage ::sto/backend :assets-fs))
|
2022-02-10 19:50:40 +01:00
|
|
|
([storage conn]
|
|
|
|
(-> storage
|
2023-02-06 12:27:53 +01:00
|
|
|
(assoc ::db/pool-or-conn conn)
|
|
|
|
(assoc ::sto/backend :assets-fs))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
(t/deftest put-and-retrieve-object
|
2022-02-10 19:50:40 +01:00
|
|
|
(let [storage (-> (:app.storage/storage th/*system*)
|
|
|
|
(configure-storage-backend))
|
2021-01-31 17:02:38 +01:00
|
|
|
content (sto/content "content")
|
2023-03-03 14:05:26 +01:00
|
|
|
object (sto/put-object! storage {::sto/content content
|
|
|
|
:content-type "text/plain"
|
|
|
|
:other "data"})]
|
2023-02-06 12:27:53 +01:00
|
|
|
|
|
|
|
(t/is (sto/object? object))
|
2023-03-03 14:05:26 +01:00
|
|
|
(t/is (fs/path? (sto/get-object-path storage object)))
|
2023-02-06 12:27:53 +01:00
|
|
|
|
2021-01-31 17:02:38 +01:00
|
|
|
(t/is (nil? (:expired-at object)))
|
2022-06-22 11:42:23 +02:00
|
|
|
(t/is (= :assets-fs (:backend object)))
|
2021-01-31 17:02:38 +01:00
|
|
|
(t/is (= "data" (:other (meta object))))
|
|
|
|
(t/is (= "text/plain" (:content-type (meta object))))
|
2023-03-03 14:05:26 +01:00
|
|
|
(t/is (= "content" (slurp (sto/get-object-data storage object))))
|
2023-11-29 12:53:34 +01:00
|
|
|
(t/is (= "content" (slurp (sto/get-object-path storage object))))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
(t/deftest put-and-retrieve-expired-object
|
2022-02-10 19:50:40 +01:00
|
|
|
(let [storage (-> (:app.storage/storage th/*system*)
|
|
|
|
(configure-storage-backend))
|
2021-01-31 17:02:38 +01:00
|
|
|
content (sto/content "content")
|
2023-03-03 14:05:26 +01:00
|
|
|
object (sto/put-object! storage {::sto/content content
|
|
|
|
::sto/expired-at (dt/in-future {:seconds 1})
|
2023-11-29 12:53:34 +01:00
|
|
|
:content-type "text/plain"})]
|
2023-02-06 12:27:53 +01:00
|
|
|
|
|
|
|
(t/is (sto/object? object))
|
2021-01-31 17:02:38 +01:00
|
|
|
(t/is (dt/instant? (:expired-at object)))
|
|
|
|
(t/is (dt/is-after? (:expired-at object) (dt/now)))
|
2023-03-03 14:05:26 +01:00
|
|
|
(t/is (= object (sto/get-object storage (:id object))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
(th/sleep 1000)
|
2023-03-03 14:05:26 +01:00
|
|
|
(t/is (nil? (sto/get-object storage (:id object))))
|
|
|
|
(t/is (nil? (sto/get-object-data storage object)))
|
|
|
|
(t/is (nil? (sto/get-object-url storage object)))
|
2023-11-29 12:53:34 +01:00
|
|
|
(t/is (nil? (sto/get-object-path storage object)))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
(t/deftest put-and-delete-object
|
2022-02-10 19:50:40 +01:00
|
|
|
(let [storage (-> (:app.storage/storage th/*system*)
|
|
|
|
(configure-storage-backend))
|
2021-01-31 17:02:38 +01:00
|
|
|
content (sto/content "content")
|
2023-03-03 14:05:26 +01:00
|
|
|
object (sto/put-object! storage {::sto/content content
|
|
|
|
:content-type "text/plain"
|
|
|
|
:expired-at (dt/in-future {:seconds 1})})]
|
2023-02-06 12:27:53 +01:00
|
|
|
(t/is (sto/object? object))
|
2023-03-03 14:05:26 +01:00
|
|
|
(t/is (true? (sto/del-object! storage object)))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
;; retrieving the same object should be not nil because the
|
2021-11-15 09:53:10 -05:00
|
|
|
;; deletion is not immediate
|
2023-03-03 14:05:26 +01:00
|
|
|
(t/is (some? (sto/get-object-data storage object)))
|
|
|
|
(t/is (some? (sto/get-object-url storage object)))
|
|
|
|
(t/is (some? (sto/get-object-path storage object)))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
;; But you can't retrieve the object again because in database is
|
|
|
|
;; marked as deleted/expired.
|
2023-11-29 12:53:34 +01:00
|
|
|
(t/is (nil? (sto/get-object storage (:id object))))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
(t/deftest test-deleted-gc-task
|
2022-02-10 19:50:40 +01:00
|
|
|
(let [storage (-> (:app.storage/storage th/*system*)
|
|
|
|
(configure-storage-backend))
|
2022-02-28 17:15:58 +01:00
|
|
|
content1 (sto/content "content1")
|
|
|
|
content2 (sto/content "content2")
|
2023-08-31 14:36:31 +02:00
|
|
|
content3 (sto/content "content3")
|
2023-03-03 14:05:26 +01:00
|
|
|
object1 (sto/put-object! storage {::sto/content content1
|
|
|
|
::sto/expired-at (dt/now)
|
2023-11-29 12:53:34 +01:00
|
|
|
:content-type "text/plain"})
|
2023-03-03 14:05:26 +01:00
|
|
|
object2 (sto/put-object! storage {::sto/content content2
|
|
|
|
::sto/expired-at (dt/in-past {:hours 2})
|
2023-11-29 12:53:34 +01:00
|
|
|
:content-type "text/plain"})
|
2023-08-31 14:36:31 +02:00
|
|
|
object3 (sto/put-object! storage {::sto/content content3
|
|
|
|
::sto/expired-at (dt/in-past {:hours 1})
|
2023-11-29 12:53:34 +01:00
|
|
|
:content-type "text/plain"})]
|
2022-02-28 17:15:58 +01:00
|
|
|
|
2023-08-31 14:36:31 +02:00
|
|
|
|
2021-01-31 17:02:38 +01:00
|
|
|
(th/sleep 200)
|
|
|
|
|
2023-08-31 14:36:31 +02:00
|
|
|
(let [res (th/run-task! :storage-gc-deleted {})]
|
2021-01-31 17:02:38 +01:00
|
|
|
(t/is (= 1 (:deleted res))))
|
|
|
|
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/db-exec-one! ["select count(*) from storage_object;"])]
|
2023-08-31 14:36:31 +02:00
|
|
|
(t/is (= 2 (:count res))))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
2022-02-10 19:50:40 +01:00
|
|
|
(t/deftest test-touched-gc-task-1
|
|
|
|
(let [storage (-> (:app.storage/storage th/*system*)
|
|
|
|
(configure-storage-backend))
|
2021-01-31 17:02:38 +01:00
|
|
|
prof (th/create-profile* 1)
|
|
|
|
proj (th/create-project* 1 {:profile-id (:id prof)
|
|
|
|
:team-id (:default-team-id prof)})
|
2022-02-10 19:50:40 +01:00
|
|
|
|
2021-01-31 17:02:38 +01:00
|
|
|
file (th/create-file* 1 {:profile-id (:id prof)
|
|
|
|
:project-id (:default-project-id prof)
|
|
|
|
:is-shared false})
|
2022-02-10 19:50:40 +01:00
|
|
|
|
2021-01-31 17:02:38 +01:00
|
|
|
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-01-31 17:02:38 +01:00
|
|
|
:size 312043}
|
|
|
|
|
|
|
|
params {::th/type :upload-file-media-object
|
2023-04-24 20:18:14 +02:00
|
|
|
::rpc/profile-id (:id prof)
|
2021-01-31 17:02:38 +01:00
|
|
|
:file-id (:id file)
|
|
|
|
:is-local true
|
|
|
|
:name "testfile"
|
|
|
|
:content mfile}
|
|
|
|
|
2023-04-24 20:18:14 +02:00
|
|
|
out1 (th/command! params)
|
|
|
|
out2 (th/command! params)]
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
(t/is (nil? (:error out1)))
|
|
|
|
(t/is (nil? (:error out2)))
|
|
|
|
|
|
|
|
(let [result-1 (:result out1)
|
|
|
|
result-2 (:result out2)]
|
|
|
|
|
|
|
|
(t/is (uuid? (:id result-1)))
|
|
|
|
(t/is (uuid? (:id result-2)))
|
|
|
|
|
|
|
|
(t/is (uuid? (:media-id result-1)))
|
|
|
|
(t/is (uuid? (:media-id result-2)))
|
|
|
|
|
2022-02-28 17:15:58 +01:00
|
|
|
(t/is (= (:media-id result-1) (:media-id result-2)))
|
|
|
|
|
2023-12-29 15:21:14 +01:00
|
|
|
(th/db-update! :file-media-object
|
|
|
|
{:deleted-at (dt/now)}
|
|
|
|
{:id (:id result-1)})
|
|
|
|
|
|
|
|
;; run the objects gc task for permanent deletion
|
|
|
|
(let [res (th/run-task! :objects-gc {:min-age 0})]
|
|
|
|
(t/is (= 1 (:processed res))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
;; check that we still have all the storage objects
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/db-exec-one! ["select count(*) from storage_object"])]
|
2022-02-28 17:15:58 +01:00
|
|
|
(t/is (= 2 (:count res))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
;; now check if the storage objects are touched
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/db-exec-one! ["select count(*) from storage_object where touched_at is not null"])]
|
2022-02-28 17:15:58 +01:00
|
|
|
(t/is (= 2 (:count res))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
;; run the touched gc task
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/run-task! :storage-gc-touched {})]
|
2022-02-10 19:50:40 +01:00
|
|
|
(t/is (= 2 (:freeze res)))
|
2022-02-28 17:15:58 +01:00
|
|
|
(t/is (= 0 (:delete res))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
;; now check that there are no touched objects
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/db-exec-one! ["select count(*) from storage_object where touched_at is not null"])]
|
2021-01-31 17:02:38 +01:00
|
|
|
(t/is (= 0 (:count res))))
|
|
|
|
|
|
|
|
;; now check that all objects are marked to be deleted
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/db-exec-one! ["select count(*) from storage_object where deleted_at is not null"])]
|
2023-11-29 12:53:34 +01:00
|
|
|
(t/is (= 0 (:count res)))))))
|
2021-01-31 17:02:38 +01:00
|
|
|
|
2022-02-10 19:50:40 +01:00
|
|
|
|
|
|
|
(t/deftest test-touched-gc-task-2
|
|
|
|
(let [storage (-> (:app.storage/storage th/*system*)
|
|
|
|
(configure-storage-backend))
|
|
|
|
prof (th/create-profile* 1 {:is-active true})
|
|
|
|
team-id (:default-team-id prof)
|
|
|
|
proj-id (:default-project-id prof)
|
|
|
|
font-id (uuid/custom 10 1)
|
|
|
|
|
|
|
|
proj (th/create-project* 1 {:profile-id (:id prof)
|
|
|
|
:team-id team-id})
|
|
|
|
|
|
|
|
file (th/create-file* 1 {:profile-id (:id prof)
|
|
|
|
:project-id proj-id
|
|
|
|
:is-shared false})
|
|
|
|
|
2022-11-08 10:40:19 +01:00
|
|
|
ttfdata (-> (io/resource "backend_tests/test_files/font-1.ttf")
|
2022-06-22 11:39:57 +02:00
|
|
|
io/input-stream
|
2022-08-31 13:29:43 +02:00
|
|
|
io/read-as-bytes)
|
2022-02-10 19:50:40 +01:00
|
|
|
|
|
|
|
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"
|
2022-02-10 19:50:40 +01:00
|
|
|
:size 312043}
|
|
|
|
|
|
|
|
params1 {::th/type :upload-file-media-object
|
2023-04-24 20:18:14 +02:00
|
|
|
::rpc/profile-id (:id prof)
|
2022-02-10 19:50:40 +01:00
|
|
|
:file-id (:id file)
|
|
|
|
:is-local true
|
|
|
|
:name "testfile"
|
|
|
|
:content mfile}
|
|
|
|
|
|
|
|
params2 {::th/type :create-font-variant
|
2023-04-24 20:18:14 +02:00
|
|
|
::rpc/profile-id (:id prof)
|
2022-02-10 19:50:40 +01:00
|
|
|
:team-id team-id
|
|
|
|
:font-id font-id
|
|
|
|
:font-family "somefont"
|
|
|
|
:font-weight 400
|
|
|
|
:font-style "normal"
|
|
|
|
:data {"font/ttf" ttfdata}}
|
|
|
|
|
2023-04-24 20:18:14 +02:00
|
|
|
out1 (th/command! params1)
|
|
|
|
out2 (th/command! params2)]
|
2022-02-10 19:50:40 +01:00
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
|
|
|
|
(t/is (nil? (:error out1)))
|
|
|
|
(t/is (nil? (:error out2)))
|
|
|
|
|
|
|
|
;; run the touched gc task
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/run-task! :storage-gc-touched {})]
|
2022-09-26 08:54:12 +02:00
|
|
|
(t/is (= 5 (:freeze res)))
|
2022-02-10 19:50:40 +01:00
|
|
|
(t/is (= 0 (:delete res)))
|
|
|
|
|
|
|
|
(let [result-1 (:result out1)
|
|
|
|
result-2 (:result out2)]
|
|
|
|
|
2023-12-29 15:21:14 +01:00
|
|
|
(th/db-update! :team-font-variant
|
|
|
|
{:deleted-at (dt/now)}
|
|
|
|
{:id (:id result-2)})
|
|
|
|
|
|
|
|
;; run the objects gc task for permanent deletion
|
|
|
|
(let [res (th/run-task! :objects-gc {:min-age 0})]
|
|
|
|
(t/is (= 1 (:processed res))))
|
2022-02-10 19:50:40 +01:00
|
|
|
|
|
|
|
;; revert touched state to all storage objects
|
2023-12-29 15:21:14 +01:00
|
|
|
(th/db-exec-one! ["update storage_object set touched_at=now()"])
|
2022-02-10 19:50:40 +01:00
|
|
|
|
|
|
|
;; Run the task again
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/run-task! :storage-gc-touched {})]
|
2022-02-10 19:50:40 +01:00
|
|
|
(t/is (= 2 (:freeze res)))
|
2022-09-26 08:54:12 +02:00
|
|
|
(t/is (= 3 (:delete res))))
|
2022-02-10 19:50:40 +01:00
|
|
|
|
|
|
|
;; now check that there are no touched objects
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/db-exec-one! ["select count(*) from storage_object where touched_at is not null"])]
|
2022-02-10 19:50:40 +01:00
|
|
|
(t/is (= 0 (:count res))))
|
|
|
|
|
|
|
|
;; now check that all objects are marked to be deleted
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/db-exec-one! ["select count(*) from storage_object where deleted_at is not null"])]
|
2022-09-26 08:54:12 +02:00
|
|
|
(t/is (= 3 (:count res))))))))
|
2022-02-10 19:50:40 +01:00
|
|
|
|
2022-02-28 17:15:58 +01:00
|
|
|
(t/deftest test-touched-gc-task-3
|
2022-02-10 19:50:40 +01:00
|
|
|
(let [storage (-> (:app.storage/storage th/*system*)
|
|
|
|
(configure-storage-backend))
|
2021-01-31 17:02:38 +01:00
|
|
|
prof (th/create-profile* 1)
|
|
|
|
proj (th/create-project* 1 {:profile-id (:id prof)
|
|
|
|
:team-id (:default-team-id prof)})
|
|
|
|
file (th/create-file* 1 {:profile-id (:id prof)
|
|
|
|
:project-id (:default-project-id prof)
|
|
|
|
:is-shared false})
|
|
|
|
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-01-31 17:02:38 +01:00
|
|
|
:size 312043}
|
|
|
|
|
|
|
|
params {::th/type :upload-file-media-object
|
2023-04-24 20:18:14 +02:00
|
|
|
::rpc/profile-id (:id prof)
|
2021-01-31 17:02:38 +01:00
|
|
|
:file-id (:id file)
|
|
|
|
:is-local true
|
|
|
|
:name "testfile"
|
|
|
|
:content mfile}
|
|
|
|
|
2023-04-24 20:18:14 +02:00
|
|
|
out1 (th/command! params)
|
|
|
|
out2 (th/command! params)]
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
(t/is (nil? (:error out1)))
|
|
|
|
(t/is (nil? (:error out2)))
|
|
|
|
|
|
|
|
(let [result-1 (:result out1)
|
|
|
|
result-2 (:result out2)]
|
|
|
|
|
|
|
|
;; now we proceed to manually mark all storage objects touched
|
2023-12-29 15:21:14 +01:00
|
|
|
(th/db-exec! ["update storage_object set touched_at=now()"])
|
2021-01-31 17:02:38 +01:00
|
|
|
|
|
|
|
;; run the touched gc task
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/run-task! "storage-gc-touched" {:min-age 0})]
|
2022-02-28 17:15:58 +01:00
|
|
|
(t/is (= 2 (:freeze res)))
|
2021-01-31 17:02:38 +01:00
|
|
|
(t/is (= 0 (:delete res))))
|
|
|
|
|
|
|
|
;; check that we have all object in the db
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [rows (th/db-exec! ["select * from storage_object"])]
|
|
|
|
(t/is (= 2 (count rows)))))
|
2022-02-28 17:15:58 +01:00
|
|
|
|
2023-11-24 10:40:56 +01:00
|
|
|
;; now we proceed to manually delete all file_media_object
|
2023-12-29 15:21:14 +01:00
|
|
|
(th/db-exec! ["update file_media_object set deleted_at = now()"])
|
|
|
|
|
|
|
|
(let [res (th/run-task! "objects-gc" {:min-age 0})]
|
|
|
|
(t/is (= 2 (:processed res))))
|
2022-02-28 17:15:58 +01:00
|
|
|
|
|
|
|
;; run the touched gc task
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [res (th/run-task! "storage-gc-touched" {:min-age 0})]
|
2022-02-28 17:15:58 +01:00
|
|
|
(t/is (= 0 (:freeze res)))
|
|
|
|
(t/is (= 2 (:delete res))))
|
|
|
|
|
|
|
|
;; check that we have all no objects
|
2023-12-29 15:21:14 +01:00
|
|
|
(let [rows (th/db-exec! ["select * from storage_object where deleted_at is null"])]
|
|
|
|
(t/is (= 0 (count rows))))))
|