2020-02-04 10:05:51 -05: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 02:43:04 -05:00
|
|
|
;; Copyright (c) UXBOX Labs SL
|
2020-02-04 10:05:51 -05:00
|
|
|
|
2021-05-28 06:50:42 -05:00
|
|
|
(ns app.services-profile-test
|
2020-02-04 10:05:51 -05:00
|
|
|
(:require
|
2021-05-28 06:50:42 -05:00
|
|
|
[app.db :as db]
|
|
|
|
[app.rpc.mutations.profile :as profile]
|
|
|
|
[app.test-helpers :as th]
|
2020-02-04 10:05:51 -05:00
|
|
|
[clojure.java.io :as io]
|
2021-05-28 06:50:42 -05:00
|
|
|
[clojure.test :as t]
|
2020-02-04 10:05:51 -05:00
|
|
|
[cuerdas.core :as str]
|
|
|
|
[datoteka.core :as fs]
|
2021-05-28 06:50:42 -05:00
|
|
|
[mockery.core :refer [with-mocks]]))
|
2020-02-04 10:05:51 -05:00
|
|
|
|
2021-02-11 11:57:41 -05:00
|
|
|
;; TODO: profile deletion with teams
|
|
|
|
;; TODO: profile deletion with owner teams
|
|
|
|
|
2020-02-04 10:05:51 -05:00
|
|
|
(t/use-fixtures :once th/state-init)
|
|
|
|
(t/use-fixtures :each th/database-reset)
|
|
|
|
|
2021-01-31 11:02:10 -05:00
|
|
|
;; Test with wrong credentials
|
|
|
|
(t/deftest profile-login-failed-1
|
|
|
|
(let [profile (th/create-profile* 1)
|
|
|
|
data {::th/type :login
|
|
|
|
:email "profile1.test@nodomain.com"
|
|
|
|
:password "foobar"
|
|
|
|
:scope "foobar"}
|
|
|
|
out (th/mutation! data)]
|
|
|
|
|
|
|
|
#_(th/print-result! out)
|
|
|
|
(let [error (:error out)]
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :validation))
|
|
|
|
(t/is (th/ex-of-code? error :wrong-credentials)))))
|
|
|
|
|
|
|
|
;; Test with good credentials but profile not activated.
|
|
|
|
(t/deftest profile-login-failed-2
|
|
|
|
(let [profile (th/create-profile* 1)
|
|
|
|
data {::th/type :login
|
|
|
|
:email "profile1.test@nodomain.com"
|
|
|
|
:password "123123"
|
|
|
|
:scope "foobar"}
|
|
|
|
out (th/mutation! data)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(let [error (:error out)]
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :validation))
|
|
|
|
(t/is (th/ex-of-code? error :wrong-credentials)))))
|
|
|
|
|
|
|
|
;; Test with good credentials but profile already activated
|
|
|
|
(t/deftest profile-login-success
|
|
|
|
(let [profile (th/create-profile* 1 {:is-active true})
|
|
|
|
data {::th/type :login
|
|
|
|
:email "profile1.test@nodomain.com"
|
|
|
|
:password "123123"
|
|
|
|
:scope "foobar"}
|
|
|
|
out (th/mutation! data)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(t/is (= (:id profile) (get-in out [:result :id])))))
|
2020-02-17 03:49:04 -05:00
|
|
|
|
|
|
|
(t/deftest profile-query-and-manipulation
|
2021-01-31 11:02:10 -05:00
|
|
|
(let [profile (th/create-profile* 1)]
|
2020-02-17 03:49:04 -05:00
|
|
|
(t/testing "query profile"
|
2020-12-24 08:32:19 -05:00
|
|
|
(let [data {::th/type :profile
|
2020-02-17 03:49:04 -05:00
|
|
|
:profile-id (:id profile)}
|
2020-12-24 08:32:19 -05:00
|
|
|
out (th/query! data)]
|
2020-02-17 03:49:04 -05:00
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= "Profile 1" (:fullname result)))
|
|
|
|
(t/is (= "profile1.test@nodomain.com" (:email result)))
|
|
|
|
(t/is (not (contains? result :password))))))
|
|
|
|
|
|
|
|
(t/testing "update profile"
|
|
|
|
(let [data (assoc profile
|
2020-12-24 08:32:19 -05:00
|
|
|
::th/type :update-profile
|
2021-01-30 05:31:51 -05:00
|
|
|
:profile-id (:id profile)
|
2020-02-17 03:49:04 -05:00
|
|
|
:fullname "Full Name"
|
2020-04-08 03:57:29 -05:00
|
|
|
:lang "en"
|
|
|
|
:theme "dark")
|
2020-12-24 08:32:19 -05:00
|
|
|
out (th/mutation! data)]
|
2020-02-17 03:49:04 -05:00
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(t/is (nil? (:result out)))))
|
|
|
|
|
2020-05-14 06:49:11 -05:00
|
|
|
(t/testing "query profile after update"
|
2020-12-24 08:32:19 -05:00
|
|
|
(let [data {::th/type :profile
|
2020-05-14 06:49:11 -05:00
|
|
|
:profile-id (:id profile)}
|
2020-12-24 08:32:19 -05:00
|
|
|
out (th/query! data)]
|
2020-02-17 03:49:04 -05:00
|
|
|
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
|
|
|
|
(let [result (:result out)]
|
2020-05-14 06:49:11 -05:00
|
|
|
(t/is (= "Full Name" (:fullname result)))
|
|
|
|
(t/is (= "en" (:lang result)))
|
|
|
|
(t/is (= "dark" (:theme result))))))
|
2020-02-17 03:49:04 -05:00
|
|
|
|
2020-08-07 07:55:22 -05:00
|
|
|
(t/testing "update photo"
|
2020-12-24 08:32:19 -05:00
|
|
|
(let [data {::th/type :update-profile-photo
|
2020-08-07 07:55:22 -05:00
|
|
|
:profile-id (:id profile)
|
|
|
|
:file {:filename "sample.jpg"
|
|
|
|
:size 123123
|
2021-05-28 06:50:42 -05:00
|
|
|
:tempfile (th/tempfile "app/test_files/sample.jpg")
|
2020-08-07 07:55:22 -05:00
|
|
|
:content-type "image/jpeg"}}
|
2020-12-24 08:32:19 -05:00
|
|
|
out (th/mutation! data)]
|
2020-02-17 03:49:04 -05:00
|
|
|
|
2020-08-07 07:55:22 -05:00
|
|
|
;; (th/print-result! out)
|
2020-12-24 08:32:19 -05:00
|
|
|
(t/is (nil? (:error out)))))
|
|
|
|
))
|
2020-02-17 03:49:04 -05:00
|
|
|
|
2021-01-31 11:02:10 -05:00
|
|
|
(t/deftest profile-deletion-simple
|
|
|
|
(let [task (:app.tasks.delete-profile/handler th/*system*)
|
|
|
|
prof (th/create-profile* 1)
|
|
|
|
file (th/create-file* 1 {:profile-id (:id prof)
|
|
|
|
:project-id (:default-project-id prof)
|
|
|
|
:is-shared false})]
|
|
|
|
|
|
|
|
;; profile is not deleted because it does not meet all
|
|
|
|
;; conditions to be deleted.
|
|
|
|
(let [result (task {:props {:profile-id (:id prof)}})]
|
|
|
|
(t/is (nil? result)))
|
|
|
|
|
|
|
|
;; Request profile to be deleted
|
2021-03-30 08:35:18 -05:00
|
|
|
(with-mocks [mock {:target 'app.worker/submit! :return nil}]
|
2021-01-31 11:02:10 -05:00
|
|
|
(let [params {::th/type :delete-profile
|
|
|
|
:profile-id (:id prof)}
|
|
|
|
out (th/mutation! params)]
|
|
|
|
(t/is (nil? (:error out)))
|
2020-02-17 03:49:04 -05:00
|
|
|
|
2021-01-31 11:02:10 -05:00
|
|
|
;; check the mock
|
|
|
|
(let [mock (deref mock)
|
2021-03-30 08:35:18 -05:00
|
|
|
mock-params (first (:call-args mock))]
|
2021-01-31 11:02:10 -05:00
|
|
|
(t/is (:called? mock))
|
|
|
|
(t/is (= 1 (:call-count mock)))
|
2021-03-30 08:35:18 -05:00
|
|
|
(t/is (= :delete-profile (:app.worker/task mock-params)))
|
|
|
|
(t/is (= (:id prof) (:profile-id mock-params))))))
|
2021-01-31 11:02:10 -05:00
|
|
|
|
|
|
|
;; query files after profile soft deletion
|
|
|
|
(let [params {::th/type :files
|
|
|
|
:project-id (:default-project-id prof)
|
|
|
|
:profile-id (:id prof)}
|
|
|
|
out (th/query! params)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:error out)))
|
|
|
|
(t/is (= 1 (count (:result out)))))
|
|
|
|
|
|
|
|
;; execute permanent deletion task
|
|
|
|
(let [result (task {:props {:profile-id (:id prof)}})]
|
|
|
|
(t/is (true? result)))
|
|
|
|
|
|
|
|
;; query profile after delete
|
|
|
|
(let [params {::th/type :profile
|
|
|
|
:profile-id (:id prof)}
|
|
|
|
out (th/query! params)]
|
|
|
|
;; (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))))
|
|
|
|
|
|
|
|
;; query files after profile soft deletion
|
|
|
|
(let [params {::th/type :files
|
|
|
|
:project-id (:default-project-id prof)
|
|
|
|
:profile-id (:id prof)}
|
|
|
|
out (th/query! params)]
|
|
|
|
;; (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))))
|
2020-05-14 06:49:11 -05:00
|
|
|
))
|
2020-04-08 03:57:29 -05:00
|
|
|
|
2021-01-31 11:02:10 -05:00
|
|
|
(t/deftest registration-domain-whitelist
|
2021-05-25 14:19:13 -05:00
|
|
|
(let [whitelist #{"gmail.com" "hey.com" "ya.ru"}]
|
2021-01-31 11:02:10 -05:00
|
|
|
(t/testing "allowed email domain"
|
|
|
|
(t/is (true? (profile/email-domain-in-whitelist? whitelist "username@ya.ru")))
|
2021-05-25 14:19:13 -05:00
|
|
|
(t/is (true? (profile/email-domain-in-whitelist? #{} "username@somedomain.com"))))
|
2020-02-17 03:49:04 -05:00
|
|
|
|
2021-01-31 11:02:10 -05:00
|
|
|
(t/testing "not allowed email domain"
|
|
|
|
(t/is (false? (profile/email-domain-in-whitelist? whitelist "username@somedomain.com"))))))
|
2020-03-16 10:55:44 -05:00
|
|
|
|
2021-03-02 05:48:51 -05:00
|
|
|
(t/deftest test-register-with-no-terms-and-privacy
|
|
|
|
(let [data {::th/type :register-profile
|
|
|
|
:email "user@example.com"
|
|
|
|
:password "foobar"
|
|
|
|
:fullname "foobar"
|
|
|
|
:terms-privacy nil}
|
|
|
|
out (th/mutation! data)
|
|
|
|
error (:error out)
|
|
|
|
edata (ex-data error)]
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (= (:type edata) :validation))
|
|
|
|
(t/is (= (:code edata) :spec-validation))))
|
|
|
|
|
|
|
|
(t/deftest test-register-with-bad-terms-and-privacy
|
|
|
|
(let [data {::th/type :register-profile
|
|
|
|
:email "user@example.com"
|
|
|
|
:password "foobar"
|
|
|
|
:fullname "foobar"
|
|
|
|
:terms-privacy false}
|
|
|
|
out (th/mutation! data)
|
|
|
|
error (:error out)
|
|
|
|
edata (ex-data error)]
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (= (:type edata) :validation))
|
|
|
|
(t/is (= (:code edata) :invalid-terms-and-privacy))))
|
|
|
|
|
2021-02-11 07:36:46 -05:00
|
|
|
(t/deftest test-register-when-registration-disabled
|
|
|
|
(with-mocks [mock {:target 'app.config/get
|
|
|
|
:return (th/mock-config-get-with
|
|
|
|
{:registration-enabled false})}]
|
|
|
|
(let [data {::th/type :register-profile
|
|
|
|
:email "user@example.com"
|
|
|
|
:password "foobar"
|
2021-03-02 05:48:51 -05:00
|
|
|
:fullname "foobar"
|
|
|
|
:terms-privacy true}
|
2021-02-11 07:36:46 -05:00
|
|
|
out (th/mutation! data)
|
|
|
|
error (:error out)
|
|
|
|
edata (ex-data error)]
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (= (:type edata) :restriction))
|
|
|
|
(t/is (= (:code edata) :registration-disabled)))))
|
|
|
|
|
|
|
|
(t/deftest test-register-existing-profile
|
|
|
|
(let [profile (th/create-profile* 1)
|
|
|
|
data {::th/type :register-profile
|
|
|
|
:email (:email profile)
|
|
|
|
:password "foobar"
|
2021-03-02 05:48:51 -05:00
|
|
|
:fullname "foobar"
|
|
|
|
:terms-privacy true}
|
2021-02-11 07:36:46 -05:00
|
|
|
out (th/mutation! data)
|
|
|
|
error (:error out)
|
|
|
|
edata (ex-data error)]
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (= (:type edata) :validation))
|
|
|
|
(t/is (= (:code edata) :email-already-exists))))
|
|
|
|
|
|
|
|
(t/deftest test-register-profile
|
|
|
|
(with-mocks [mock {:target 'app.emails/send!
|
|
|
|
:return nil}]
|
|
|
|
(let [pool (:app.db/pool th/*system*)
|
|
|
|
data {::th/type :register-profile
|
|
|
|
:email "user@example.com"
|
|
|
|
:password "foobar"
|
2021-03-02 05:48:51 -05:00
|
|
|
:fullname "foobar"
|
|
|
|
:terms-privacy true}
|
2021-02-11 07:36:46 -05:00
|
|
|
out (th/mutation! data)]
|
|
|
|
;; (th/print-result! out)
|
2021-03-30 08:35:18 -05:00
|
|
|
(let [mock (deref mock)
|
|
|
|
[params] (:call-args mock)]
|
2021-02-11 07:36:46 -05:00
|
|
|
;; (clojure.pprint/pprint params)
|
|
|
|
(t/is (:called? mock))
|
|
|
|
(t/is (= (:email data) (:to params)))
|
|
|
|
(t/is (contains? params :extra-data))
|
|
|
|
(t/is (contains? params :token)))
|
|
|
|
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (false? (:is-demo result)))
|
|
|
|
(t/is (= (:email data) (:email result)))
|
|
|
|
(t/is (= "penpot" (:auth-backend result)))
|
|
|
|
(t/is (= "foobar" (:fullname result)))
|
|
|
|
(t/is (not (contains? result :password)))))))
|
|
|
|
|
|
|
|
(t/deftest test-register-profile-with-bounced-email
|
|
|
|
(with-mocks [mock {:target 'app.emails/send!
|
|
|
|
:return nil}]
|
|
|
|
(let [pool (:app.db/pool th/*system*)
|
|
|
|
data {::th/type :register-profile
|
|
|
|
:email "user@example.com"
|
|
|
|
:password "foobar"
|
2021-03-02 05:48:51 -05:00
|
|
|
:fullname "foobar"
|
|
|
|
:terms-privacy true}
|
2021-02-11 07:36:46 -05:00
|
|
|
_ (th/create-global-complaint-for pool {:type :bounce :email "user@example.com"})
|
|
|
|
out (th/mutation! data)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
|
|
|
|
(let [mock (deref mock)]
|
|
|
|
(t/is (false? (:called? mock))))
|
|
|
|
|
|
|
|
(let [error (:error out)
|
|
|
|
edata (ex-data error)]
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (= (:type edata) :validation))
|
|
|
|
(t/is (= (:code edata) :email-has-permanent-bounces))))))
|
|
|
|
|
|
|
|
(t/deftest test-register-profile-with-complained-email
|
2021-02-11 11:57:41 -05:00
|
|
|
(with-mocks [mock {:target 'app.emails/send! :return nil}]
|
2021-02-11 07:36:46 -05:00
|
|
|
(let [pool (:app.db/pool th/*system*)
|
|
|
|
data {::th/type :register-profile
|
|
|
|
:email "user@example.com"
|
|
|
|
:password "foobar"
|
2021-03-02 05:48:51 -05:00
|
|
|
:fullname "foobar"
|
|
|
|
:terms-privacy true}
|
2021-02-11 07:36:46 -05:00
|
|
|
_ (th/create-global-complaint-for pool {:type :complaint :email "user@example.com"})
|
|
|
|
out (th/mutation! data)]
|
|
|
|
|
|
|
|
(let [mock (deref mock)]
|
|
|
|
(t/is (true? (:called? mock))))
|
|
|
|
|
|
|
|
(let [result (:result out)]
|
|
|
|
(t/is (= (:email data) (:email result)))))))
|
2021-02-11 11:57:41 -05:00
|
|
|
|
|
|
|
(t/deftest test-email-change-request
|
2021-02-24 05:51:57 -05:00
|
|
|
(with-mocks [email-send-mock {:target 'app.emails/send! :return nil}
|
|
|
|
cfg-get-mock {:target 'app.config/get
|
|
|
|
:return (th/mock-config-get-with
|
|
|
|
{:smtp-enabled true})}]
|
2021-02-11 11:57:41 -05:00
|
|
|
(let [profile (th/create-profile* 1)
|
2021-02-24 05:51:57 -05:00
|
|
|
pool (:app.db/pool th/*system*)
|
|
|
|
data {::th/type :request-email-change
|
|
|
|
:profile-id (:id profile)
|
|
|
|
:email "user1@example.com"}]
|
2021-02-11 11:57:41 -05:00
|
|
|
|
|
|
|
;; without complaints
|
|
|
|
(let [out (th/mutation! data)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:result out)))
|
2021-02-24 05:51:57 -05:00
|
|
|
(let [mock (deref email-send-mock)]
|
2021-02-11 11:57:41 -05:00
|
|
|
(t/is (= 1 (:call-count mock)))
|
|
|
|
(t/is (true? (:called? mock)))))
|
|
|
|
|
|
|
|
;; with complaints
|
|
|
|
(th/create-global-complaint-for pool {:type :complaint :email (:email data)})
|
|
|
|
(let [out (th/mutation! data)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:result out)))
|
2021-02-24 05:51:57 -05:00
|
|
|
(t/is (= 2 (:call-count (deref email-send-mock)))))
|
2021-02-11 11:57:41 -05:00
|
|
|
|
|
|
|
;; with bounces
|
|
|
|
(th/create-global-complaint-for pool {:type :bounce :email (:email data)})
|
|
|
|
(let [out (th/mutation! data)
|
|
|
|
error (:error out)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :validation))
|
|
|
|
(t/is (th/ex-of-code? error :email-has-permanent-bounces))
|
2021-02-24 05:51:57 -05:00
|
|
|
(t/is (= 2 (:call-count (deref email-send-mock))))))))
|
|
|
|
|
|
|
|
|
|
|
|
(t/deftest test-email-change-request-without-smtp
|
|
|
|
(with-mocks [email-send-mock {:target 'app.emails/send! :return nil}
|
|
|
|
cfg-get-mock {:target 'app.config/get
|
|
|
|
:return (th/mock-config-get-with
|
|
|
|
{:smtp-enabled false})}]
|
|
|
|
(let [profile (th/create-profile* 1)
|
|
|
|
pool (:app.db/pool th/*system*)
|
|
|
|
data {::th/type :request-email-change
|
|
|
|
:profile-id (:id profile)
|
|
|
|
:email "user1@example.com"}]
|
|
|
|
|
|
|
|
;; without complaints
|
|
|
|
(let [out (th/mutation! data)
|
|
|
|
res (:result out)]
|
|
|
|
(t/is (= {:changed true} res))
|
|
|
|
(let [mock (deref email-send-mock)]
|
|
|
|
(t/is (false? (:called? mock))))))))
|
|
|
|
|
2021-02-11 11:57:41 -05:00
|
|
|
|
|
|
|
(t/deftest test-request-profile-recovery
|
|
|
|
(with-mocks [mock {:target 'app.emails/send! :return nil}]
|
|
|
|
(let [profile1 (th/create-profile* 1)
|
|
|
|
profile2 (th/create-profile* 2 {:is-active true})
|
|
|
|
pool (:app.db/pool th/*system*)
|
|
|
|
data {::th/type :request-profile-recovery}]
|
|
|
|
|
|
|
|
;; with invalid email
|
|
|
|
(let [data (assoc data :email "foo@bar.com")
|
|
|
|
out (th/mutation! data)]
|
|
|
|
(t/is (nil? (:result out)))
|
|
|
|
(t/is (= 0 (:call-count (deref mock)))))
|
|
|
|
|
|
|
|
;; with valid email inactive user
|
|
|
|
(let [data (assoc data :email (:email profile1))
|
|
|
|
out (th/mutation! data)
|
|
|
|
error (:error out)]
|
|
|
|
(t/is (= 0 (:call-count (deref mock))))
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :validation))
|
|
|
|
(t/is (th/ex-of-code? error :profile-not-verified)))
|
|
|
|
|
|
|
|
;; with valid email and active user
|
|
|
|
(let [data (assoc data :email (:email profile2))
|
|
|
|
out (th/mutation! data)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:result out)))
|
|
|
|
(t/is (= 1 (:call-count (deref mock)))))
|
|
|
|
|
|
|
|
;; with valid email and active user with global complaints
|
|
|
|
(th/create-global-complaint-for pool {:type :complaint :email (:email profile2)})
|
|
|
|
(let [data (assoc data :email (:email profile2))
|
|
|
|
out (th/mutation! data)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (nil? (:result out)))
|
|
|
|
(t/is (= 2 (:call-count (deref mock)))))
|
|
|
|
|
|
|
|
;; with valid email and active user with global bounce
|
|
|
|
(th/create-global-complaint-for pool {:type :bounce :email (:email profile2)})
|
|
|
|
(let [data (assoc data :email (:email profile2))
|
|
|
|
out (th/mutation! data)
|
|
|
|
error (:error out)]
|
|
|
|
;; (th/print-result! out)
|
|
|
|
(t/is (= 2 (:call-count (deref mock))))
|
|
|
|
(t/is (th/ex-info? error))
|
|
|
|
(t/is (th/ex-of-type? error :validation))
|
|
|
|
(t/is (th/ex-of-code? error :email-has-permanent-bounces)))
|
|
|
|
|
|
|
|
)))
|