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/.
|
|
|
|
;;
|
2021-04-10 09:43:04 +02:00
|
|
|
;; Copyright (c) UXBOX Labs SL
|
2020-05-27 10:03:00 +02:00
|
|
|
|
2021-05-28 13:50:42 +02:00
|
|
|
(ns app.services-files-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]
|
|
|
|
[app.http :as http]
|
2021-01-31 20:55:53 +01:00
|
|
|
[app.storage :as sto]
|
2021-05-28 13:50:42 +02:00
|
|
|
[app.test-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
|
2020-02-17 09:49:04 +01:00
|
|
|
: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"
|
|
|
|
:is-shared false}
|
|
|
|
out (th/mutation! 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"
|
|
|
|
:profile-id (:id prof)}
|
2020-12-24 14:32:19 +01:00
|
|
|
out (th/mutation! 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 (deprecated)"
|
2020-12-24 14:32:19 +01:00
|
|
|
(let [data {::th/type :files
|
2020-05-14 13:49:11 +02:00
|
|
|
:project-id proj-id
|
2020-02-17 09:49:04 +01:00
|
|
|
:profile-id (:id prof)}
|
2020-12-24 14:32:19 +01:00
|
|
|
out (th/query! data)]
|
2020-02-17 09:49:04 +01: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-09-07 10:56:42 +02:00
|
|
|
(t/is (= 1 (count (get-in result [0 :data :pages])))))))
|
2020-02-17 09:49:04 +01:00
|
|
|
|
2021-05-07 11:36:34 +02:00
|
|
|
(t/testing "query files"
|
|
|
|
(let [data {::th/type :project-files
|
|
|
|
:project-id proj-id
|
|
|
|
:profile-id (:id prof)}
|
|
|
|
out (th/query! data)]
|
|
|
|
|
|
|
|
;; (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"
|
2020-12-24 14:32:19 +01:00
|
|
|
(let [data {::th/type :file
|
2020-02-17 09:49:04 +01:00
|
|
|
:profile-id (:id prof)
|
|
|
|
:id file-id}
|
2020-12-24 14:32:19 +01:00
|
|
|
out (th/query! 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
|
|
|
|
:profile-id (:id prof)}
|
2020-12-24 14:32:19 +01:00
|
|
|
out (th/mutation! 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"
|
2020-12-24 14:32:19 +01:00
|
|
|
(let [data {::th/type :file
|
2020-02-17 09:49:04 +01:00
|
|
|
:profile-id (:id prof)
|
|
|
|
:id file-id}
|
2020-12-24 14:32:19 +01:00
|
|
|
out (th/query! 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"
|
2020-12-24 14:32:19 +01:00
|
|
|
(let [data {::th/type :files
|
2020-05-14 13:49:11 +02:00
|
|
|
:project-id proj-id
|
2020-02-17 09:49:04 +01:00
|
|
|
:profile-id (:id prof)}
|
2020-12-24 14:32:19 +01:00
|
|
|
out (th/query! 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
|
|
|
|
|
|
|
(t/deftest file-media-gc-task
|
2021-02-25 17:45:39 +01:00
|
|
|
(letfn [(create-file-media-object [{:keys [profile-id file-id]}]
|
|
|
|
(let [mfile {:filename "sample.jpg"
|
2021-05-28 13:50:42 +02:00
|
|
|
:tempfile (th/tempfile "app/test_files/sample.jpg")
|
2021-02-25 17:45:39 +01:00
|
|
|
:content-type "image/jpeg"
|
|
|
|
:size 312043}
|
|
|
|
params {::th/type :upload-file-media-object
|
|
|
|
:profile-id profile-id
|
|
|
|
:file-id file-id
|
|
|
|
:is-local true
|
|
|
|
:name "testfile"
|
|
|
|
:content mfile}
|
|
|
|
out (th/mutation! params)]
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(:result out)))
|
|
|
|
|
|
|
|
(update-file [{:keys [profile-id file-id changes revn] :or {revn 0}}]
|
|
|
|
(let [params {::th/type :update-file
|
|
|
|
:id file-id
|
|
|
|
:session-id (uuid/random)
|
|
|
|
:profile-id profile-id
|
|
|
|
:revn revn
|
|
|
|
:changes changes}
|
|
|
|
out (th/mutation! params)]
|
|
|
|
(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})
|
|
|
|
|
|
|
|
fmo1 (create-file-media-object {:profile-id (:id profile)
|
|
|
|
:file-id (:id file)})
|
|
|
|
fmo2 (create-file-media-object {:profile-id (:id profile)
|
|
|
|
:file-id (:id file)})
|
|
|
|
shid (uuid/random)
|
|
|
|
|
|
|
|
ures (update-file
|
|
|
|
{:file-id (:id file)
|
|
|
|
:profile-id (:id profile)
|
|
|
|
:revn 0
|
|
|
|
:changes
|
|
|
|
[{:type :add-obj
|
|
|
|
:page-id (first (get-in file [:data :pages]))
|
|
|
|
:id shid
|
|
|
|
:parent-id uuid/zero
|
|
|
|
:frame-id uuid/zero
|
|
|
|
:obj {:id shid
|
|
|
|
:name "image"
|
|
|
|
:frame-id uuid/zero
|
|
|
|
:parent-id uuid/zero
|
|
|
|
:type :image
|
|
|
|
:metadata {:id (:id fmo1)}}}]})]
|
|
|
|
|
|
|
|
;; run the task inmediatelly
|
|
|
|
(let [task (:app.tasks.file-media-gc/handler th/*system*)
|
|
|
|
res (task {})]
|
|
|
|
(t/is (= 0 (:processed res))))
|
|
|
|
|
|
|
|
;; make the file ellegible for GC waiting 300ms (configured
|
|
|
|
;; timeout for testing)
|
|
|
|
(th/sleep 300)
|
|
|
|
|
|
|
|
;; run the task again
|
|
|
|
(let [task (:app.tasks.file-media-gc/handler th/*system*)
|
|
|
|
res (task {})]
|
|
|
|
(t/is (= 1 (:processed res))))
|
|
|
|
|
|
|
|
;; retrieve file and check trimmed attribute
|
|
|
|
(let [row (db/exec-one! th/*pool* ["select * from file where id = ?" (:id file)])]
|
|
|
|
(t/is (true? (:has-media-trimmed row))))
|
|
|
|
|
|
|
|
;; check file media objects
|
|
|
|
(let [rows (db/exec! th/*pool* ["select * from file_media_object where file_id = ?" (:id file)])]
|
|
|
|
(t/is (= 1 (count rows))))
|
|
|
|
|
|
|
|
;; The underlying storage objects are still available.
|
|
|
|
(t/is (some? (sto/get-object storage (:media-id fmo2))))
|
|
|
|
(t/is (some? (sto/get-object storage (:thumbnail-id fmo2))))
|
2021-01-31 20:55:53 +01:00
|
|
|
(t/is (some? (sto/get-object storage (:media-id fmo1))))
|
2021-02-25 17:45:39 +01:00
|
|
|
(t/is (some? (sto/get-object storage (:thumbnail-id fmo1))))
|
|
|
|
|
|
|
|
;; but if we pass the touched gc task two of them should disappear
|
|
|
|
(let [task (:app.storage/gc-touched-task th/*system*)
|
|
|
|
res (task {})]
|
|
|
|
(t/is (= 0 (:freeze res)))
|
|
|
|
(t/is (= 2 (:delete res)))
|
|
|
|
|
|
|
|
(t/is (nil? (sto/get-object storage (:media-id fmo2))))
|
|
|
|
(t/is (nil? (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)))))
|
|
|
|
|
|
|
|
)))
|
|
|
|
|
|
|
|
(t/deftest permissions-checks-creating-file
|
|
|
|
(let [profile1 (th/create-profile* 1)
|
|
|
|
profile2 (th/create-profile* 2)
|
|
|
|
|
|
|
|
data {::th/type :create-file
|
|
|
|
:profile-id (:id profile2)
|
|
|
|
:project-id (:default-project-id profile1)
|
|
|
|
:name "foobar"
|
|
|
|
:is-shared false}
|
|
|
|
out (th/mutation! data)
|
|
|
|
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)
|
|
|
|
:profile-id (:id profile2)
|
|
|
|
:name "foobar"}
|
|
|
|
out (th/mutation! data)
|
|
|
|
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
|
|
|
|
:profile-id (:id profile2)
|
|
|
|
:id (:id file)}
|
|
|
|
out (th/mutation! data)
|
|
|
|
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
|
|
|
|
:profile-id (:id profile2)
|
|
|
|
:id (:id file)
|
|
|
|
:is-shared true}
|
|
|
|
out (th/mutation! data)
|
|
|
|
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
|
|
|
|
:profile-id (:id profile2)
|
|
|
|
:file-id (:id file2)
|
|
|
|
:library-id (:id file1)}
|
|
|
|
|
|
|
|
out (th/mutation! data)
|
|
|
|
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
|
|
|
|
:profile-id (:id profile2)
|
|
|
|
:file-id (:id file2)
|
|
|
|
:library-id (:id file1)}
|
|
|
|
|
|
|
|
out (th/mutation! data)
|
|
|
|
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
|
|
|
|