diff --git a/backend/profiles.clj b/backend/profiles.clj new file mode 100644 index 000000000..2662e7bde --- /dev/null +++ b/backend/profiles.clj @@ -0,0 +1,8 @@ +{:dev + {:plugins [[lein-ancient "0.6.10"]] + :dependencies [[clj-http "2.1.0"]] + :main ^:skip-aot uxbox.main} + + :prod + {:jvm-opts ^:replace ["-Xms4g" "-Xmx4g" "-XX:+UseG1GC" + "-XX:+AggressiveOpts" "-server"]}} diff --git a/backend/project.clj b/backend/project.clj new file mode 100644 index 000000000..e4deec835 --- /dev/null +++ b/backend/project.clj @@ -0,0 +1,48 @@ +(defproject uxbox-backend "0.1.0-SNAPSHOT" + :description "UXBox backend." + :url "http://uxbox.github.io" + :license {:name "MPL 2.0" :url "https://www.mozilla.org/en-US/MPL/2.0/"} + :source-paths ["src" "vendor"] + :javac-options ["-target" "1.8" "-source" "1.8" "-Xlint:-options"] + :jvm-opts ["-Dclojure.compiler.direct-linking=true" + ;; "-Dcom.sun.management.jmxremote.port=9090" + ;; "-Dcom.sun.management.jmxremote.authenticate=false" + ;; "-Dcom.sun.management.jmxremote.ssl=false" + ;; "-Dcom.sun.management.jmxremote.rmi.port=9090" + ;; "-Djava.rmi.server.hostname=0.0.0.0" + "-Dclojure.spec.check-asserts=true" + "-Dclojure.spec.compile-asserts=true" + "-XX:+UseG1GC" "-Xms1g" "-Xmx1g"] + :global-vars {*assert* true} + :dependencies [[org.clojure/clojure "1.9.0-alpha14"] + [org.clojure/tools.logging "0.3.1"] + [funcool/struct "1.0.0"] + [funcool/suricatta "1.2.0"] + [funcool/promesa "1.6.0"] + [funcool/catacumba "2.0.0-SNAPSHOT"] + + [org.clojure/data.xml "0.1.0-beta2"] + [org.jsoup/jsoup "1.10.1"] + + [hiccup "1.0.5"] + [org.im4java/im4java "1.4.0"] + + [org.slf4j/slf4j-simple "1.7.21"] + [com.layerware/hugsql-core "0.4.7" + :exclusions [org.clojure/tools.reader]] + [niwinz/migrante "0.1.0"] + + [buddy/buddy-sign "1.3.0" :exclusions [org.clojure/tools.reader]] + [buddy/buddy-hashers "1.1.0"] + + [org.xerial.snappy/snappy-java "1.1.2.6"] + [com.github.spullara.mustache.java/compiler "0.9.4"] + [org.postgresql/postgresql "9.4.1212"] + [org.quartz-scheduler/quartz "2.2.3"] + [org.quartz-scheduler/quartz-jobs "2.2.3"] + [commons-io/commons-io "2.5"] + [com.draines/postal "2.0.2"] + + [hikari-cp "1.7.5"] + [mount "0.1.10"] + [environ "1.1.0"]]) diff --git a/backend/resources/.catacumba.basedir b/backend/resources/.catacumba.basedir new file mode 100644 index 000000000..e69de29bb diff --git a/backend/resources/builtin.edn b/backend/resources/builtin.edn new file mode 100644 index 000000000..7e3216732 --- /dev/null +++ b/backend/resources/builtin.edn @@ -0,0 +1,11 @@ +{:icons + [{:name "Material Design (Action)" + :path "./material/action/svg/production" + :regex #"^.*_48px\.svg$"}] + + :images + [{:name "Generic Collection 1" + :path "./my-images/collection1/" + :regex #"^.*\.(png|jpg|webp)$"}]} + + diff --git a/backend/resources/config/default.edn b/backend/resources/config/default.edn new file mode 100644 index 000000000..d7cf66269 --- /dev/null +++ b/backend/resources/config/default.edn @@ -0,0 +1,37 @@ +{;; A secret key used for create tokens + ;; WARNING: this is a default secret key and + ;; it should be overwritten in production env. + :secret "5qjiAn-QUpawUNqGP10UZKklSqbLKcdGY3sJpq0UUACpVXGg2HOFJCBejDWVHskhRyp7iHb4rjOLXX2ZjF-5cw" + :smtp + {:host "localhost" ;; Hostname of the desired SMTP server. + :port 25 ;; Port of SMTP server. + :user nil ;; Username to authenticate with (if authenticating). + :pass nil ;; Password to authenticate with (if authenticating). + :ssl false ;; Enables SSL encryption if value is truthy. + :tls false ;; Enables TLS encryption if value is truthy. + :noop true} + + :auth-options {:alg :a256kw :enc :a128cbc-hs256} + + :email {:reply-to "no-reply@uxbox.io" + :from "no-reply@uxbox.io"} + + :http {:port 6060 + :max-body-size 52428800 + :debug true} + + :media + {:basedir "resources/public/media" + :baseuri "http://localhost:6060/media/"} + + :static + {:basedir "resources/public/static" + :baseuri "http://localhost:6060/static/"} + + :database + {:adapter "postgresql" + :username nil + :password nil + :database-name "uxbox" + :server-name "localhost" + :port-number 5432}} diff --git a/backend/resources/config/test.edn b/backend/resources/config/test.edn new file mode 100644 index 000000000..2ded91c1b --- /dev/null +++ b/backend/resources/config/test.edn @@ -0,0 +1,18 @@ +{:migrations + {:verbose false} + + :media + {:basedir "/tmp/uxbox/media" + :baseuri "http://localhost:6060/media/"} + + :static + {:basedir "/tmp/uxbox/static" + :baseuri "http://localhost:6060/static/"} + + :database + {:adapter "postgresql" + :username nil + :password nil + :database-name "test" + :server-name "localhost" + :port-number 5432}} diff --git a/backend/resources/emails/en/register.mustache b/backend/resources/emails/en/register.mustache new file mode 100644 index 000000000..c23b78dbe --- /dev/null +++ b/backend/resources/emails/en/register.mustache @@ -0,0 +1,17 @@ +-- begin :subject +Welcome to UXBOX. +-- end + +-- begin :body-text +Hello {{user}}! + +Welcome to UXBOX. + +UXBOX team. +-- end + +-- begin :body-html +

Hello {{user}}!

+

Welcome to UXBOX.

+

UXBOX team.

+-- end \ No newline at end of file diff --git a/backend/resources/migrations/0000.main.up.sql b/backend/resources/migrations/0000.main.up.sql new file mode 100644 index 000000000..d4f5e4c0f --- /dev/null +++ b/backend/resources/migrations/0000.main.up.sql @@ -0,0 +1,28 @@ +CREATE EXTENSION IF NOT EXISTS "uuid-ossp"; +CREATE EXTENSION IF NOT EXISTS "pgcrypto"; + +-- OCC + +CREATE OR REPLACE FUNCTION handle_occ() + RETURNS TRIGGER AS $occ$ + BEGIN + IF (NEW.version != OLD.version) THEN + RAISE EXCEPTION 'Version missmatch: expected % given %', + OLD.version, NEW.version + USING ERRCODE='P0002'; + ELSE + NEW.version := NEW.version + 1; + END IF; + RETURN NEW; + END; +$occ$ LANGUAGE plpgsql; + +-- Modified At + +CREATE OR REPLACE FUNCTION update_modified_at() + RETURNS TRIGGER AS $updt$ + BEGIN + NEW.modified_at := clock_timestamp(); + RETURN NEW; + END; +$updt$ LANGUAGE plpgsql; diff --git a/backend/resources/migrations/0001.txlog.up.sql b/backend/resources/migrations/0001.txlog.up.sql new file mode 100644 index 000000000..a2777ccda --- /dev/null +++ b/backend/resources/migrations/0001.txlog.up.sql @@ -0,0 +1,17 @@ +-- A table that will store the whole transaction log of the database. +CREATE TABLE IF NOT EXISTS txlog ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + payload bytea NOT NULL +); + +CREATE OR REPLACE FUNCTION handle_txlog_notify() + RETURNS TRIGGER AS $notify$ + BEGIN + PERFORM pg_notify('uxbox.transaction', (NEW.id)::text); + RETURN NEW; + END; +$notify$ LANGUAGE plpgsql; + +CREATE TRIGGER txlog_notify_tgr AFTER INSERT ON txlog + FOR EACH ROW EXECUTE PROCEDURE handle_txlog_notify(); diff --git a/backend/resources/migrations/0002.auth.up.sql b/backend/resources/migrations/0002.auth.up.sql new file mode 100644 index 000000000..eb722f845 --- /dev/null +++ b/backend/resources/migrations/0002.auth.up.sql @@ -0,0 +1,49 @@ +CREATE TABLE users ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + deleted_at timestamptz DEFAULT NULL, + + fullname text NOT NULL DEFAULT '', + username text NOT NULL, + email text NOT NULL, + photo text NOT NULL, + password text NOT NULL, + metadata bytea NOT NULL +); + +-- Insert a placeholder system user. +INSERT INTO users (id, fullname, username, email, photo, password, metadata) +VALUES ('00000000-0000-0000-0000-000000000000'::uuid, + 'System User', + '00000000-0000-0000-0000-000000000000', + 'system@uxbox.io', + '', + '!', + ''::bytea); + +CREATE UNIQUE INDEX users_username_idx + ON users USING btree (username); + +CREATE UNIQUE INDEX users_email_idx + ON users USING btree (email); + +CREATE TRIGGER users_modified_at_tgr BEFORE UPDATE ON users + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); + +CREATE TABLE user_pswd_recovery ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + "user" uuid REFERENCES users(id) ON DELETE CASCADE, + token text NOT NULL, + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + used_at timestamptz DEFAULT NULL +); + +CREATE INDEX user_pswd_recovery_user_idx + ON user_pswd_recovery USING btree ("user"); + +CREATE UNIQUE INDEX user_pswd_recovery_token_idx + ON user_pswd_recovery USING btree (token); + diff --git a/backend/resources/migrations/0003.projects.up.sql b/backend/resources/migrations/0003.projects.up.sql new file mode 100644 index 000000000..1b3233599 --- /dev/null +++ b/backend/resources/migrations/0003.projects.up.sql @@ -0,0 +1,62 @@ +-- Table + +CREATE TABLE IF NOT EXISTS projects ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + "user" uuid NOT NULL REFERENCES users(id) ON DELETE CASCADE, + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + deleted_at timestamptz DEFAULT NULL, + + version bigint NOT NULL DEFAULT 0, + name text NOT NULL +); + +CREATE TABLE IF NOT EXISTS project_shares ( + project uuid PRIMARY KEY REFERENCES projects(id) ON DELETE CASCADE, + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + token text +); + +-- Indexes + +CREATE INDEX projects_user_idx + ON projects("user"); + +CREATE UNIQUE INDEX projects_shares_token_idx + ON project_shares(token); + +-- Triggers + +CREATE OR REPLACE FUNCTION handle_project_create() + RETURNS TRIGGER AS $$ + DECLARE + token text; + BEGIN + SELECT encode(digest(gen_random_bytes(128), 'sha256'), 'hex') + INTO token; + + INSERT INTO project_shares (project, token) + VALUES (NEW.id, token); + + RETURN NEW; + END; +$$ LANGUAGE plpgsql; + +CREATE TRIGGER project_on_create_tgr + AFTER INSERT ON projects + FOR EACH ROW EXECUTE PROCEDURE handle_project_create(); + +CREATE TRIGGER project_occ_tgr +BEFORE UPDATE ON projects + FOR EACH ROW EXECUTE PROCEDURE handle_occ(); + +CREATE TRIGGER projects_modified_at_tgr +BEFORE UPDATE ON projects + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); + +CREATE TRIGGER project_shares_modified_at_tgr +BEFORE UPDATE ON project_shares + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); diff --git a/backend/resources/migrations/0004.pages.up.sql b/backend/resources/migrations/0004.pages.up.sql new file mode 100644 index 000000000..3a2ca69c1 --- /dev/null +++ b/backend/resources/migrations/0004.pages.up.sql @@ -0,0 +1,74 @@ +-- Tables + +CREATE TABLE IF NOT EXISTS pages ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + + "user" uuid NOT NULL REFERENCES users(id) ON DELETE CASCADE, + project uuid NOT NULL REFERENCES projects(id) ON DELETE CASCADE, + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + deleted_at timestamptz DEFAULT NULL, + version bigint DEFAULT 0, + + name text NOT NULL, + data bytea NOT NULL, + metadata bytea NOT NULL +); + +CREATE TABLE IF NOT EXISTS pages_history ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + + "user" uuid NOT NULL REFERENCES users(id) ON DELETE CASCADE, + page uuid NOT NULL REFERENCES pages(id) ON DELETE CASCADE, + + created_at timestamptz NOT NULL, + modified_at timestamptz NOT NULL, + version bigint NOT NULL DEFAULT 0, + + pinned bool NOT NULL DEFAULT false, + label text NOT NULL DEFAULT '', + data bytea NOT NULL +); + +-- Indexes + +CREATE INDEX pages_project_idx ON pages(project); +CREATE INDEX pages_user_idx ON pages("user"); +CREATE INDEX pages_history_page_idx ON pages_history(page); +CREATE INDEX pages_history_user_idx ON pages_history("user"); + +-- Triggers + +CREATE OR REPLACE FUNCTION handle_page_update() + RETURNS TRIGGER AS $pagechange$ + BEGIN + --- Update projects modified_at attribute when a + --- page of that project is modified. + UPDATE projects SET modified_at = clock_timestamp() + WHERE id = OLD.project; + + --- Register a new history entry if the data + --- property is changed. + IF (OLD.data != NEW.data) THEN + INSERT INTO pages_history (page, "user", created_at, + modified_at, data, version) + VALUES (OLD.id, OLD."user", OLD.modified_at, + OLD.modified_at, OLD.data, OLD.version); + END IF; + + RETURN NEW; + END; +$pagechange$ LANGUAGE plpgsql; + +CREATE TRIGGER page_on_update_tgr BEFORE UPDATE ON pages + FOR EACH ROW EXECUTE PROCEDURE handle_page_update(); + +CREATE TRIGGER page_occ_tgr BEFORE UPDATE ON pages + FOR EACH ROW EXECUTE PROCEDURE handle_occ(); + +CREATE TRIGGER pages_modified_at_tgr BEFORE UPDATE ON pages + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); + +CREATE TRIGGER pages_history_modified_at_tgr BEFORE UPDATE ON pages + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); diff --git a/backend/resources/migrations/0005.kvstore.up.sql b/backend/resources/migrations/0005.kvstore.up.sql new file mode 100644 index 000000000..0b540c1d0 --- /dev/null +++ b/backend/resources/migrations/0005.kvstore.up.sql @@ -0,0 +1,19 @@ +CREATE TABLE IF NOT EXISTS kvstore ( + "user" uuid NOT NULL REFERENCES users(id) ON DELETE CASCADE, + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + + version bigint NOT NULL DEFAULT 0, + + key text NOT NULL, + value bytea NOT NULL, + + PRIMARY KEY (key, "user") +); + +CREATE TRIGGER kvstore_occ_tgr BEFORE UPDATE ON kvstore + FOR EACH ROW EXECUTE PROCEDURE handle_occ(); + +CREATE TRIGGER kvstore_modified_at_tgr BEFORE UPDATE ON kvstore + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); diff --git a/backend/resources/migrations/0006.emails.up.sql b/backend/resources/migrations/0006.emails.up.sql new file mode 100644 index 000000000..b621501b7 --- /dev/null +++ b/backend/resources/migrations/0006.emails.up.sql @@ -0,0 +1,27 @@ +CREATE TYPE email_status AS ENUM ('pending', 'ok', 'failed'); + +CREATE TABLE IF NOT EXISTS email_queue ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + deleted_at timestamptz DEFAULT NULL, + + data bytea NOT NULL, + + priority smallint NOT NULL DEFAULT 10 + CHECK (priority BETWEEN 0 and 10), + + status email_status NOT NULL DEFAULT 'pending', + retries integer NOT NULL DEFAULT -1 +); + +-- Triggers + +CREATE TRIGGER email_queue_modified_at_tgr BEFORE UPDATE ON email_queue + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); + +-- Indexes + +CREATE INDEX email_status_idx + ON email_queue (status); diff --git a/backend/resources/migrations/0007.images.up.sql b/backend/resources/migrations/0007.images.up.sql new file mode 100644 index 000000000..4abb6a86b --- /dev/null +++ b/backend/resources/migrations/0007.images.up.sql @@ -0,0 +1,59 @@ +-- Tables + +CREATE TABLE IF NOT EXISTS images_collections ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + "user" uuid NOT NULL REFERENCES users(id) ON DELETE CASCADE, + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + deleted_at timestamptz DEFAULT NULL, + version bigint NOT NULL DEFAULT 0, + + name text NOT NULL +); + +CREATE TABLE IF NOT EXISTS images ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + "user" uuid NOT NULL REFERENCES users(id) ON DELETE CASCADE, + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + deleted_at timestamptz DEFAULT NULL, + + version bigint NOT NULL DEFAULT 0, + + width int NOT NULL, + height int NOT NULL, + mimetype text NOT NULL, + collection uuid REFERENCES images_collections(id) + ON DELETE SET NULL + DEFAULT NULL, + name text NOT NULL, + path text NOT NULL +); + +-- Indexes + +CREATE INDEX images_collections_user_idx + ON images_collections ("user"); + +CREATE INDEX images_collection_idx + ON images (collection); + +CREATE INDEX images_user_idx + ON images ("user"); + +-- Triggers + +CREATE TRIGGER images_collections_occ_tgr BEFORE UPDATE ON images_collections + FOR EACH ROW EXECUTE PROCEDURE handle_occ(); + +CREATE TRIGGER images_collections_modified_at_tgr BEFORE UPDATE ON images_collections + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); + +CREATE TRIGGER images_occ_tgr BEFORE UPDATE ON images + FOR EACH ROW EXECUTE PROCEDURE handle_occ(); + +CREATE TRIGGER images_modified_at_tgr BEFORE UPDATE ON images + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); + diff --git a/backend/resources/migrations/0008.icons.up.sql b/backend/resources/migrations/0008.icons.up.sql new file mode 100644 index 000000000..aee7ad025 --- /dev/null +++ b/backend/resources/migrations/0008.icons.up.sql @@ -0,0 +1,56 @@ +-- Tables + +CREATE TABLE IF NOT EXISTS icons_collections ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + "user" uuid NOT NULL REFERENCES users(id) ON DELETE CASCADE, + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + deleted_at timestamptz DEFAULT NULL, + version bigint NOT NULL DEFAULT 0, + + name text NOT NULL +); + +CREATE TABLE IF NOT EXISTS icons ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + "user" uuid NOT NULL REFERENCES users(id) ON DELETE CASCADE, + + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + modified_at timestamptz NOT NULL DEFAULT clock_timestamp(), + deleted_at timestamptz DEFAULT NULL, + version bigint NOT NULL DEFAULT 0, + + name text NOT NULL, + content text NOT NULL, + metadata bytea NOT NULL, + collection uuid REFERENCES icons_collections(id) + ON DELETE SET NULL + DEFAULT NULL +); + +-- Indexes + +CREATE INDEX icon_colections_user_idx + ON icons_collections ("user"); + +CREATE INDEX icons_user_idx + ON icons ("user"); + +CREATE INDEX icons_collection_idx + ON icons (collection); + +-- Triggers + +CREATE TRIGGER icons_collections_occ_tgr BEFORE UPDATE ON icons_collections + FOR EACH ROW EXECUTE PROCEDURE handle_occ(); + +CREATE TRIGGER icons_collections_modified_at_tgr BEFORE UPDATE ON icons_collections + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); + +CREATE TRIGGER icons_occ_tgr BEFORE UPDATE ON icons + FOR EACH ROW EXECUTE PROCEDURE handle_occ(); + +CREATE TRIGGER icons_modified_at_tgr BEFORE UPDATE ON icons + FOR EACH ROW EXECUTE PROCEDURE update_modified_at(); + diff --git a/backend/resources/migrations/XXXX.workers.up.sql b/backend/resources/migrations/XXXX.workers.up.sql new file mode 100644 index 000000000..31237318c --- /dev/null +++ b/backend/resources/migrations/XXXX.workers.up.sql @@ -0,0 +1,11 @@ +CREATE TYPE task_status + AS ENUM ('pending', 'canceled', 'completed', 'failed'); + +CREATE TABLE task ( + id uuid PRIMARY KEY DEFAULT uuid_generate_v4(), + created_at timestamptz NOT NULL DEFAULT clock_timestamp(), + completed_at timestamptz DEFAULT NULL, + queue text NOT NULL DEFAULT '', + status task_status NOT NULL DEFAULT 'pending', + error text NOT NULL DEFAULT '' +) WITH (OIDS=FALSE); diff --git a/backend/resources/public/static/images/email/facebook.png b/backend/resources/public/static/images/email/facebook.png new file mode 100644 index 000000000..bb0fe0529 Binary files /dev/null and b/backend/resources/public/static/images/email/facebook.png differ diff --git a/backend/resources/public/static/images/email/img-header.jpg b/backend/resources/public/static/images/email/img-header.jpg new file mode 100644 index 000000000..1b685cd50 Binary files /dev/null and b/backend/resources/public/static/images/email/img-header.jpg differ diff --git a/backend/resources/public/static/images/email/linkedin.png b/backend/resources/public/static/images/email/linkedin.png new file mode 100644 index 000000000..79aa66bc7 Binary files /dev/null and b/backend/resources/public/static/images/email/linkedin.png differ diff --git a/backend/resources/public/static/images/email/logo.png b/backend/resources/public/static/images/email/logo.png new file mode 100644 index 000000000..b4917b065 Binary files /dev/null and b/backend/resources/public/static/images/email/logo.png differ diff --git a/backend/resources/public/static/images/email/twitter.png b/backend/resources/public/static/images/email/twitter.png new file mode 100644 index 000000000..fec79b21a Binary files /dev/null and b/backend/resources/public/static/images/email/twitter.png differ diff --git a/backend/resources/sql/cli.sql b/backend/resources/sql/cli.sql new file mode 100644 index 000000000..360428296 --- /dev/null +++ b/backend/resources/sql/cli.sql @@ -0,0 +1,28 @@ +-- :name get-image-collection :? :1 +select * + from images_collections as cc + where cc.id = :id + and cc."user" = '00000000-0000-0000-0000-000000000000'::uuid; + +-- :name create-image : (locked_tasks.created_at, locked_tasks.id) + order by created_at, id + limit 1 + ) as j + from locked_tasks + where locked_tasks.id is not null + limit 1 + ) as t1 + ) +) +select id, status, error, created_at + from locked_tasks + where locked + limit 1; + +-- :name create-task :? :1 +insert into tasks (queue) +values (:queue) +returning *; + +-- :name mark-task-done +update tasks + set status = 'completed', + completed_at = clock_timestamp() + where id = :id; + +-- :name mark-task-failed +update tasks + set status = 'failed', + error = :error, + completed_at = clock_timestamp() + where id = :id; diff --git a/backend/scripts/fixtures.sh b/backend/scripts/fixtures.sh new file mode 100755 index 000000000..f06e934ca --- /dev/null +++ b/backend/scripts/fixtures.sh @@ -0,0 +1,2 @@ +#!/usr/bin/env bash +lein run -m uxbox.fixtures/init diff --git a/backend/scripts/run.sh b/backend/scripts/run.sh new file mode 100755 index 000000000..f3401bf97 --- /dev/null +++ b/backend/scripts/run.sh @@ -0,0 +1,3 @@ +#!/usr/bin/env bash +export PATH=/opt/img/bin:$PATH +lein trampoline run -m uxbox.main diff --git a/backend/scripts/smtpd.sh b/backend/scripts/smtpd.sh new file mode 100644 index 000000000..a4aa39cc6 --- /dev/null +++ b/backend/scripts/smtpd.sh @@ -0,0 +1,2 @@ +#!/usr/bin/env bash +python -m smtpd -n -c DebuggingServer localhost:25 diff --git a/backend/src/data_readers.clj b/backend/src/data_readers.clj new file mode 100644 index 000000000..bd70c26c8 --- /dev/null +++ b/backend/src/data_readers.clj @@ -0,0 +1 @@ +{instant uxbox.util.time/from-string} diff --git a/backend/src/uxbox/cli/collimp.clj b/backend/src/uxbox/cli/collimp.clj new file mode 100644 index 000000000..e19d02705 --- /dev/null +++ b/backend/src/uxbox/cli/collimp.clj @@ -0,0 +1,172 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.cli.collimp + "Collection importer command line helper." + (:require [clojure.spec :as s] + [clojure.pprint :refer [pprint]] + [clojure.java.io :as io] + [mount.core :as mount] + [cuerdas.core :as str] + [suricatta.core :as sc] + [storages.core :as st] + [storages.util :as fs] + [uxbox.config] + [uxbox.db :as db] + [uxbox.migrations] + [uxbox.media :as media] + [uxbox.cli.sql :as sql] + [uxbox.util.spec :as us] + [uxbox.util.cli :as cli] + [uxbox.util.uuid :as uuid] + [uxbox.util.data :as data]) + (:import [java.io Reader PushbackReader] + [javax.imageio ImageIO])) + +;; --- Constants & Specs + +(def ^:const +imates-uuid-ns+ #uuid "3642a582-565f-4070-beba-af797ab27a6e") + +(s/def ::name string?) +(s/def ::type keyword?) +(s/def ::path string?) +(s/def ::regex us/regex?) + +(s/def ::import-entry + (s/keys :req-un [::name ::type ::path ::regex])) + +;; --- CLI Helpers + + +(defn printerr + [& args] + (binding [*out* *err*] + (apply println args))) + +(defn pushback-reader + [reader] + (PushbackReader. ^Reader reader)) + +;; --- Colors Collections Importer + +(def storage media/images-storage) + +(defn- create-image-collection + "Create or replace image collection by its name." + [conn {:keys [name] :as entry}] + (let [id (uuid/namespaced +imates-uuid-ns+ name) + sqlv (sql/create-image-collection {:id id :name name})] + (sc/execute conn sqlv) + id)) + +(defn- retrieve-image-size + [path] + (let [path (fs/path path) + file (.toFile path) + buff (ImageIO/read file)] + [(.getWidth buff) + (.getHeight buff)])) + +(defn- retrieve-image + [conn id] + {:pre [(uuid? id)]} + (let [sqlv (sql/get-image {:id id})] + (some->> (sc/fetch-one conn sqlv) + (data/normalize-attrs)))) + +(defn- delete-image + [conn {:keys [id path] :as image}] + {:pre [(uuid? id) + (fs/path? path)]} + (let [sqlv (sql/delete-image {:id id})] + @(st/delete storage path) + (sc/execute conn sqlv))) + +(defn- create-image + [conn collid imageid localpath] + {:pre [(fs/path? localpath) + (uuid? collid) + (uuid? imageid)]} + (let [filename (fs/base-name localpath) + [width height] (retrieve-image-size localpath) + extension (second (fs/split-ext filename)) + path @(st/save storage filename localpath) + params {:name filename + :path (str path) + :mimetype (case extension + ".jpg" "image/jpeg" + ".png" "image/png") + :width width + :height height + :collection collid + :id imageid} + sqlv (sql/create-image params)] + (sc/execute conn sqlv))) + +(defn- import-image + [conn id fpath] + {:pre [(uuid? id) (fs/path? fpath)]} + #_(let [imageid (uuid/namespaced +imates-uuid-ns+ (str id fpath))] + (if-let [image (retrieve-image conn imageid)] + (do + (delete-image conn image) + (create-image conn id imageid fpath)) + (create-image conn id imageid fpath)))) + +(defn- process-images-entry + [conn {:keys [path regex] :as entry}] + {:pre [(s/valid? ::import-entry entry)]} + (let [id (uuid/random) #_(create-image-collection conn entry)] + (doseq [fpath (fs/list-files path)] + (when (re-matches regex (str fpath)) + (import-image conn id fpath))))) + +;; --- Entry Point + +(defn- check-path! + [path] + (when-not path + (cli/print-err! "No path is provided.") + (cli/exit! -1)) + (when-not (fs/exists? path) + (cli/print-err! "Path does not exists.") + (cli/exit! -1)) + (when (fs/directory? path) + (cli/print-err! "The provided path is a directory.") + (cli/exit! -1)) + (fs/path path)) + +(defn- read-import-file + [path] + (let [path (check-path! path) + parent (fs/parent path) + reader (pushback-reader (io/reader path))] + [parent (read reader)])) + +(defn- start-system + [] + (-> (mount/only #{#'uxbox.config/config + #'uxbox.db/datasource + #'uxbox.migrations/migrations}) + (mount/start))) + +(defn- stop-system + [] + (mount/stop)) + +(defn- run-importer + [directory data] + (println "Running importer on:") + (pprint data)) + +(defn -main + [& [path]] + (let [[directory data] (read-import-file path)] + (start-system) + (try + (run-importer directory data) + (finally + (stop-system))))) diff --git a/backend/src/uxbox/cli/sql.clj b/backend/src/uxbox/cli/sql.clj new file mode 100644 index 000000000..77890198f --- /dev/null +++ b/backend/src/uxbox/cli/sql.clj @@ -0,0 +1,4 @@ +(ns uxbox.cli.sql + (:require [hugsql.core :as hugsql])) + +(hugsql/def-sqlvec-fns "sql/cli.sql" {:quoting :ansi :fn-suffix ""}) diff --git a/backend/src/uxbox/config.clj b/backend/src/uxbox/config.clj new file mode 100644 index 000000000..8b6ce12bc --- /dev/null +++ b/backend/src/uxbox/config.clj @@ -0,0 +1,52 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.config + "A configuration management." + (:require [mount.core :refer [defstate]] + [environ.core :refer (env)] + [buddy.core.hash :as hash] + [clojure.java.io :as io] + [clojure.edn :as edn] + [uxbox.util.exceptions :as ex] + [uxbox.util.data :refer [deep-merge]])) + +;; --- Configuration Loading & Parsing + +(def ^:dynamic *default-config-path* "config/default.edn") +(def ^:dynamic *local-config-path* "config/local.edn") + +(defn read-config + [] + (let [builtin (io/resource *default-config-path*) + local (io/resource *local-config-path*) + external (io/file (:uxbox-config env))] + (deep-merge (edn/read-string (slurp builtin)) + (when local (edn/read-string (slurp local))) + (when (and external (.exists external)) + (edn/read-string (slurp external)))))) + +(defn read-test-config + [] + (binding [*local-config-path* "config/test.edn"] + (read-config))) + +(defstate config + :start (read-config)) + +;; --- Secret Loading & Parsing + +(defn- initialize-secret + [config] + (let [secret (:secret config)] + (when-not secret + (ex/raise :code ::missing-secret-key + :message "Missing `:secret` key in config.")) + (hash/blake2b-256 secret))) + +(defstate secret + :start (initialize-secret config)) + diff --git a/backend/src/uxbox/db.clj b/backend/src/uxbox/db.clj new file mode 100644 index 000000000..9d60a3889 --- /dev/null +++ b/backend/src/uxbox/db.clj @@ -0,0 +1,89 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.db + "Database access layer for UXBOX." + (:require [mount.core :as mount :refer (defstate)] + [promesa.core :as p] + [hikari-cp.core :as hikari] + [executors.core :as exec] + [suricatta.core :as sc] + [suricatta.proto :as scp] + [suricatta.types :as sct] + [suricatta.transaction :as sctx] + [uxbox.config :as cfg]) + (:import org.jooq.TransactionContext + org.jooq.TransactionProvider + org.jooq.Configuration)) + +;; --- State + +(def ^:const +defaults+ + {:connection-timeout 30000 + :idle-timeout 600000 + :max-lifetime 1800000 + :minimum-idle 10 + :maximum-pool-size 10 + :adapter "postgresql" + :username "" + :password "" + :database-name "" + :server-name "localhost" + :port-number 5432}) + +(defn create-datasource + [config] + (let [dbconf (merge +defaults+ config)] + (hikari/make-datasource dbconf))) + +(defstate datasource + :start (create-datasource (:database cfg/config)) + :stop (hikari/close-datasource datasource)) + +;; --- Suricatta Async Adapter + +(defn transaction + "Asynchronous transaction handling." + {:internal true} + [ctx func] + (let [^Configuration conf (.derive (scp/-config ctx)) + ^TransactionContext txctx (sctx/transaction-context conf) + ^TransactionProvider provider (.transactionProvider conf)] + (doto conf + (.data "suricatta.rollback" false) + (.data "suricatta.transaction" true)) + (try + (.begin provider txctx) + (->> (func (sct/context conf)) + (p/map (fn [result] + (if (.data conf "suricatta.rollback") + (.rollback provider txctx) + (.commit provider txctx)) + result)) + (p/error (fn [error] + (.rollback provider (.cause txctx error)) + (p/rejected error)))) + (catch Exception cause + (.rollback provider (.cause txctx cause)) + (p/rejected cause))))) + +;; --- Public Api + +(defmacro atomic + [ctx & body] + `(transaction ~ctx (fn [~ctx] ~@body))) + +(defn connection + [] + (sc/context datasource)) + +(defn fetch + [& args] + (exec/submit #(apply sc/fetch args))) + +(defn execute + [& args] + (exec/submit #(apply sc/execute args))) diff --git a/backend/src/uxbox/emails.clj b/backend/src/uxbox/emails.clj new file mode 100644 index 000000000..d4d89f7bd --- /dev/null +++ b/backend/src/uxbox/emails.clj @@ -0,0 +1,14 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.emails + "Main api for send emails." + (:require [uxbox.emails.core :as core])) + +(def send! core/send!) +(def render core/render) + +(load "emails/users") diff --git a/backend/src/uxbox/emails/core.clj b/backend/src/uxbox/emails/core.clj new file mode 100644 index 000000000..454bae340 --- /dev/null +++ b/backend/src/uxbox/emails/core.clj @@ -0,0 +1,89 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.emails.core + (:require [hiccup.core :refer (html)] + [hiccup.page :refer (html4)] + [suricatta.core :as sc] + [uxbox.db :as db] + [uxbox.config :as cfg] + [uxbox.sql :as sql] + [uxbox.emails.layouts :as layouts] + [uxbox.util.blob :as blob] + [uxbox.util.transit :as t])) + +(def emails + "A global state for registring emails." + (atom {})) + +(defmacro defemail + [type & args] + (let [email (apply hash-map args)] + `(do + (swap! emails assoc ~type ~email) + nil))) + +(defn- render-subject + [{:keys [subject]} context] + (cond + (delay? subject) (deref subject) + (ifn? subject) (subject context) + (string? subject) subject + :else (throw (ex-info "Invalid subject." {})))) + +(defn- render-body + [[type bodyfn] layout context] + (let [layoutfn (get layout type)] + {:content (cond-> (bodyfn context) + layoutfn (layoutfn context) + (= type :text/html) (html4)) + ::type type + :type (subs (str type) 1)})) + +(defn- render-body-alternatives + [{:keys [layout body] :as email} context] + (reduce #(conj %1 (render-body %2 layout context)) [:alternatives] body)) + +(defn render-email + [email context] + (let [config (:email cfg/config) + from (or (:email/from context) + (:from config)) + reply-to (or (:email/reply-to context) + (:reply-to config) + from)] + {:subject (render-subject email context) + :body (render-body-alternatives email context) + :to (:email/to context) + :from from + :reply-to reply-to})) + +(def valid-priority? #{:high :low}) +(def valid-email-identifier? #(contains? @emails %)) + +(defn render + "Render a email as data structure." + [{name :email/name :as context}] + {:pre [(valid-email-identifier? name)]} + (let [email (get @emails name)] + (render-email email context))) + +(defn send! + "Schedule the email for sending." + [{name :email/name + priority :email/priority + :or {priority :high} + :as context}] + {:pre [(valid-priority? priority) + (valid-email-identifier? name)]} + (let [email (get @emails name) + email (render-email email context) + data (-> email t/encode blob/encode) + priority (case priority :low 1 :high 10) + sqlv (sql/insert-email {:data data :priority priority})] + (with-open [conn (db/connection)] + (sc/atomic conn + (sc/execute conn sqlv))))) diff --git a/backend/src/uxbox/emails/layouts.clj b/backend/src/uxbox/emails/layouts.clj new file mode 100644 index 000000000..9f26be117 --- /dev/null +++ b/backend/src/uxbox/emails/layouts.clj @@ -0,0 +1,228 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.emails.layouts + (:require [uxbox.media :as md])) + +(def default-embedded-styles + "/* GLOBAL */ + * { + margin:0; + padding:0; + font-family: Arial, sans-serif; + font-size: 100%; + line-height: 1.6; + } + + img { + max-width: 100%; + width: 100%; + } + + .img-header { + border-top-left-radius: 5px; + border-top-right-radius: 5px; + } + + body { + -webkit-font-smoothing:antialiased; + -webkit-text-size-adjust:none; + width: 100%!important; + height: 100%; + } + + /* ELEMENTS */ + a { + color: #78dbbe; + text-decoration:none; + font-weight: bold; + } + + .btn-primary { + text-decoration:none; + color: #fff; + background-color: #78dbbe; + padding: 10px 30px; + font-weight: bold; + margin: 20px 0; + text-align: center; + cursor: pointer; + display: inline-block; + border-radius: 4px; + } + + .btn-primary:hover { + color: #FFF; + background-color: #8eefcf; + } + + .last { + margin-bottom: 0; + } + + .first{ + margin-top: 0; + } + + .logo { + background-color: #f6f6f6; + padding: 10px; + text-align: center; + padding-bottom: 25px; + } + .logo h2 { + color: #777; + font-size: 20px; + font-weight: bold; + margin-top: 15px; + } + .logo img { + max-width: 150px; + } + + /* BODY */ + table.body-wrap { + width: 100%; + padding: 20px; + } + + table.body-wrap .container{ + border-radius: 5px; + color: #ababab; + } + + + /* FOOTER */ + table.footer-wrap { + width: 100%; + clear:both!important; + } + + .footer-wrap .container p { + font-size: 12px; + color:#666; + + } + + table.footer-wrap a{ + color: #999; + } + + + /* TYPOGRAPHY */ + h1,h2,h3{ + font-family: Arial, sans-serif; + line-height: 1.1; + margin-bottom:15px; + color:#000; + margin: 40px 0 10px; + line-height: 1.2; + font-weight:200; + } + + h1 { + color: #777; + font-size: 28px; + font-weight: bold; + } + h2 { + font-size: 24px; + } + h3 { + font-size: 18px; + } + + p, ul { + margin-bottom: 10px; + font-weight: normal; + } + + ul li { + margin-left:5px; + list-style-position: inside; + } + + /* RESPONSIVE */ + + /* Set a max-width, and make it display as block so it will automatically stretch to that width, but will also shrink down on a phone or something */ + .container { + display: block !important; + max-width: 620px !important; + margin: 0 auto !important; /* makes it centered */ + clear: both !important; + } + + /* This should also be a block element, so that it will fill 100% of the .container */ + .content { + padding: 20px; + max-width: 620px; + margin: 0 auto; + display: block; + } + + /* Let's make sure tables in the content area are 100% wide */ + .content table { + width: 100%; + }") + +(defn- default-html + [body context] + [:html + [:head + [:meta {:http-equiv "Content-Type" + :content "text/html; charset=UTF-8"}] + [:meta {:name "viewport" + :content "width=device-width"}] + [:title "title"] + [:style default-embedded-styles]] + [:body {:bgcolor "#f6f6f6" + :cz-shortcut-listen "true"} + [:table.body-wrap + [:tbody + [:tr + [:td] + [:td.container {:bgcolor "#FFFFFF"} + [:div.logo + [:img {:src (md/resolve-asset "images/email/logo.png") + :alt "UXBOX"}]] + body] + [:td]]]] + [:table.footer-wrap + [:tbody + [:tr + [:td] + [:td.container + [:div.content + [:table + [:tbody + [:tr + [:td + [:div {:style "text-align: center;"} + [:a {:href "#" :target "_blank"} + [:img {:style "display: inline-block; width: 25px; margin-right: 5px;" + :src (md/resolve-asset "images/email/twitter.png")}]] + [:a {:href "#" :target "_blank"} + [:img {:style "display: inline-block; width: 25px; margin-right: 5px;" + :src (md/resolve-asset "images/email/facebook.png")}]] + [:a {:href "#" :target "_blank"} + [:img {:style "display: inline-block; width: 25px; margin-right: 5px;" + :src (md/resolve-asset "images/email/linkedin.png")}]]]]] + [:tr + [:td {:align "center"} + [:p + [:span "Sent from UXBOX | "] + [:a {:href "#" :target "_blank"} + [:unsubscribe "Email preferences"]]]]]]]]] + [:td]]]]]]) + +(defn default-text + [body context] + body) + +(def default + "Default layout instance." + {:text/html default-html + :text/plain default-text}) diff --git a/backend/src/uxbox/emails/users.clj b/backend/src/uxbox/emails/users.clj new file mode 100644 index 000000000..d6f1f0cf7 --- /dev/null +++ b/backend/src/uxbox/emails/users.clj @@ -0,0 +1,77 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.emails.users + (:require [uxbox.media :as md] + [uxbox.emails.core :refer (defemail)] + [uxbox.emails.layouts :as layouts])) + +;; --- User Register + +(defn- register-body-html + [{:keys [name] :as ctx}] + [:div + [:img.img-header {:src (md/resolve-asset "images/email/img-header.jpg") + :alt "UXBOX"}] + [:div.content + [:table + [:tbody + [:tr + [:td + [:h1 "Hi " name] + [:p "Welcome to uxbox."] + [:p + [:a.btn-primary {:href "#"} "Sign in"]] + [:p "Sincerely," [:br] [:strong "The UXBOX Team"]] + #_[:p "P.S. Having trouble signing up? please contact " + [:a {:href "#"} "Support email"]]]]]]]]) + +(defn- register-body-text + [{:keys [name] :as ctx}] + (str "Hi " name "\n\n" + "Welcome to uxbox!\n\n" + "Sincerely, the UXBOX team.\n")) + +(defemail :users/register + :layout layouts/default + :subject "UXBOX: Welcome!" + :body {:text/html register-body-html + :text/plain register-body-text}) + +;; --- Password Recovery + +(defn- password-recovery-body-html + [{:keys [name token] :as ctx}] + [:div + [:img.img-header {:src (md/resolve-asset "images/img-header.jpg") + :alt "UXBOX"}] + [:div.content + [:table + [:tbody + [:tr + [:td + [:h1 "Hi " name] + [:p "A password recovery is requested."] + [:p + "Please, follow the following url in order to" + "change your password." + [:a {:href "#"} "http://uxbox.io/..."]] + [:p "Sincerely," [:br] [:strong "The UXBOX Team"]]]]]]]]) + +(defn- password-recovery-body-text + [{:keys [name token] :as ctx}] + (str "Hi " name "\n\n" + "A password recovery is requested.\n\n" + "Please follow the following url in order to change the password:\n\n" + " http://uxbox.io/recovery/" token "\n\n\n" + "Sincerely, the UXBOX team.\n")) + +(defemail :users/password-recovery + :layout layouts/default + :subject "Password recovery requested." + :body {:text/html password-recovery-body-html + :text/plain password-recovery-body-text}) + diff --git a/backend/src/uxbox/fixtures.clj b/backend/src/uxbox/fixtures.clj new file mode 100644 index 000000000..d75f33d52 --- /dev/null +++ b/backend/src/uxbox/fixtures.clj @@ -0,0 +1,86 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.fixtures + "A initial fixtures." + (:require [buddy.hashers :as hashers] + [buddy.core.codecs :as codecs] + [catacumba.serializers :as sz] + [mount.core :as mount] + [clj-uuid :as uuid] + [suricatta.core :as sc] + [uxbox.config :as cfg] + [uxbox.db :as db] + [uxbox.migrations] + [uxbox.util.transit :as t] + [uxbox.services.users :as susers] + [uxbox.services.projects :as sproj] + [uxbox.services.pages :as spag])) + +(defn- mk-uuid + [prefix i] + (uuid/v5 uuid/+namespace-oid+ (str prefix i))) + +(defn- data-encode + [data] + (-> (t/encode data) + (codecs/bytes->str))) + +(defn- create-user + [conn i] + (println "create user" i) + (susers/create-user conn + {:username (str "user" i) + :id (mk-uuid "user" i) + :fullname (str "User " i) + :metadata (data-encode {}) + :password "123123" + :email (str "user" i ".test@uxbox.io")})) + +(defn- create-project + [conn i ui] + ;; (Thread/sleep 20) + (println "create project" i "for user" ui) + (sproj/create-project conn + {:id (mk-uuid "project" i) + :user (mk-uuid "user" ui) + :name (str "project " i)})) + +(defn- create-page + [conn i pi ui] + ;; (Thread/sleep 1) + (println "create page" i "for user" ui "for project" pi) + (spag/create-page conn + {:id (mk-uuid "page" i) + :user (mk-uuid "user" ui) + :project (mk-uuid "project" pi) + :data nil + :metadata {:width 1024 + :height 768 + :layout "tablet"} + :name (str "page " i)})) + +(def num-users 50) +(def num-projects 5) +(def num-pages 5) + +(defn init + [] + (mount/start) + (with-open [conn (db/connection)] + (sc/atomic conn + (doseq [i (range num-users)] + (create-user conn i)) + + (doseq [ui (range num-users)] + (doseq [i (range num-projects)] + (create-project conn (str ui i) ui))) + + (doseq [pi (range num-projects)] + (doseq [ui (range num-users)] + (doseq [i (range num-pages)] + (create-page conn (str pi ui i) (str ui pi) ui)))))) + (mount/stop)) diff --git a/backend/src/uxbox/frontend.clj b/backend/src/uxbox/frontend.clj new file mode 100644 index 000000000..385ace05d --- /dev/null +++ b/backend/src/uxbox/frontend.clj @@ -0,0 +1,155 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend + (:require [mount.core :refer [defstate]] + [catacumba.core :as ct] + [catacumba.http :as http] + [catacumba.serializers :as sz] + [catacumba.handlers.auth :as cauth] + [catacumba.handlers.parse :as cparse] + [catacumba.handlers.misc :as cmisc] + [uxbox.config :as cfg] + [uxbox.frontend.auth :as auth] + [uxbox.frontend.users :as users] + [uxbox.frontend.errors :as errors] + [uxbox.frontend.projects :as projects] + [uxbox.frontend.pages :as pages] + [uxbox.frontend.images :as images] + [uxbox.frontend.icons :as icons] + [uxbox.frontend.kvstore :as kvstore] + [uxbox.frontend.svgparse :as svgparse] + [uxbox.frontend.debug-emails :as dbgemails] + [uxbox.util.response :refer [rsp]] + [uxbox.util.uuid :as uuid])) + +;; --- Top Level Handlers + +(defn- welcome-api + "A GET entry point for the api that shows + a welcome message." + [context] + (let [body {:message "Welcome to UXBox api."}] + (-> (sz/encode body :json) + (http/ok {:content-type "application/json"})))) + +(defn- debug-only + [context] + (if (-> cfg/config :server :debug) + (ct/delegate) + (http/not-found ""))) + +;; --- Config + +(def cors-conf + {:origin "*" + :max-age 3600 + :allow-methods #{:post :put :get :delete :trace} + :allow-headers #{:x-requested-with :content-type :authorization}}) + +;; --- Routes + +(defn routes + ([] (routes cfg/config)) + ([config] + (let [auth-opts {:secret cfg/secret + :options (:auth-options cfg/config)}] + (ct/routes + [[:any (cauth/auth (cauth/jwe-backend auth-opts))] + [:any (cmisc/autoreloader)] + + [:get "api" #'welcome-api] + [:assets "media" {:dir "public/media"}] + [:assets "static" {:dir "public/static"}] + + [:prefix "debug" + [:any debug-only] + [:get "emails" #'dbgemails/list-emails] + [:get "emails/email" #'dbgemails/show-email]] + + [:prefix "api" + [:any (cmisc/cors cors-conf)] + [:any (cparse/body-params)] + [:error #'errors/handler] + + [:post "auth/token" #'auth/login] + [:post "auth/register" #'users/register-user] + [:get "auth/recovery/:token" #'users/validate-recovery-token] + [:post "auth/recovery" #'users/request-recovery] + [:put "auth/recovery" #'users/recover-password] + + [:get "projects-by-token/:token" #'projects/retrieve-project-by-share-token] + + ;; SVG Parse + [:post "svg/parse" #'svgparse/parse] + + [:any #'auth/authorization] + + ;; KVStore + [:put "kvstore" #'kvstore/update] + [:get "kvstore/:key" #'kvstore/retrieve] + [:delete "kvstore/:key" #'kvstore/delete] + + ;; Projects + [:get "projects/:id/pages" #'pages/list-pages-by-project] + [:put "projects/:id" #'projects/update-project] + [:delete "projects/:id" #'projects/delete-project] + [:post "projects" #'projects/create-project] + [:get "projects" #'projects/list-projects] + + ;; Image Collections + [:put "library/image-collections/:id" #'images/update-collection] + [:delete "library/image-collections/:id" #'images/delete-collection] + [:get "library/image-collections" #'images/list-collections] + [:post "library/image-collections" #'images/create-collection] + [:get "library/image-collections/:id/images" #'images/list-images] + [:get "library/image-collections/images" #'images/list-images] + + ;; Images + [:put "library/images/copy" #'images/copy-image] + [:delete "library/images/:id" #'images/delete-image] + [:get "library/images/:id" #'images/retrieve-image] + [:put "library/images/:id" #'images/update-image] + [:post "library/images" #'images/create-image] + + ;; Icon Collections + [:put "library/icon-collections/:id" #'icons/update-collection] + [:delete "library/icon-collections/:id" #'icons/delete-collection] + [:get "library/icon-collections" #'icons/list-collections] + [:post "library/icon-collections" #'icons/create-collection] + [:get "library/icon-collections/:id/icons" #'icons/list-icons] + [:get "library/icon-collections/icons" #'icons/list-icons] + + ;; Icons + [:put "library/icons/copy" #'icons/copy-icon] + [:delete "library/icons/:id" #'icons/delete-icon] + [:put "library/icons/:id" #'icons/update-icon] + [:post "library/icons" #'icons/create-icon] + + ;; Pages + [:put "pages/:id/metadata" #'pages/update-page-metadata] + [:get "pages/:id/history" #'pages/retrieve-page-history] + [:put "pages/:id/history/:hid" #'pages/update-page-history] + [:put "pages/:id" #'pages/update-page] + [:delete "pages/:id" #'pages/delete-page] + [:post "pages" #'pages/create-page] + + ;; Profile + [:get "profile/me" #'users/retrieve-profile] + [:put "profile/me" #'users/update-profile] + [:put "profile/me/password" #'users/update-password] + [:post "profile/me/photo" #'users/update-photo]]])))) + +;; --- State Initialization + +(defn- start-server + [config] + (let [config (:http config)] + (ct/run-server (routes config) config))) + +(defstate server + :start (start-server cfg/config) + :stop (.stop server)) diff --git a/backend/src/uxbox/frontend/auth.clj b/backend/src/uxbox/frontend/auth.clj new file mode 100644 index 000000000..724f6d994 --- /dev/null +++ b/backend/src/uxbox/frontend/auth.clj @@ -0,0 +1,33 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.auth + (:require [clojure.spec :as s] + [catacumba.core :as ct] + [catacumba.http :as http] + [promesa.core :as p] + [uxbox.util.spec :as us] + [uxbox.services :as sv] + [uxbox.util.uuid :as uuid] + [uxbox.util.response :refer (rsp)])) + +(s/def ::scope string?) +(s/def ::login (s/keys :req-un [::us/username ::us/password ::scope])) + +(defn login + [{data :data}] + (let [data (us/conform ::login data) + message (assoc data :type :login)] + (->> (sv/novelty message) + (p/map #(http/ok (rsp %)))))) + +;; TODO: improve authorization + +(defn authorization + [{:keys [identity] :as context}] + (if identity + (ct/delegate {:identity (uuid/from-string (:id identity))}) + (http/forbidden (rsp nil)))) diff --git a/backend/src/uxbox/frontend/debug_emails.clj b/backend/src/uxbox/frontend/debug_emails.clj new file mode 100644 index 000000000..606177889 --- /dev/null +++ b/backend/src/uxbox/frontend/debug_emails.clj @@ -0,0 +1,68 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.debug-emails + "A helper namespace for just render emails." + (:require [clojure.edn :as edn] + [catacumba.http :as http] + [hiccup.page :refer (html5)] + [uxbox.emails :as emails] + [uxbox.emails.core :as emails-core])) + +(def +available-emails+ + {:users/register + {:name "Cirilla"} + :users/password-recovery + {:name "Cirilla" + :token "agNFhA6SolcFb4Us2NOTNWh0cfFDquVLAav400xQPjw"}}) + +(defn- render-emails-list + [] + (html5 + [:section {:style "font-family: Monoid, monospace; font-size: 14px;"} + [:h1 "Available emails"] + [:table {:style "width: 500px;"} + [:tbody + [:tr + (for [[type email] @emails-core/emails] + [:tr + [:td (pr-str type)] + [:td + [:a {:href (str "/debug/emails/email?id=" + (pr-str type) + "&type=:text/html")} + "(html)"]] + [:td + [:a {:href (str "/debug/emails/email?id=" + (pr-str type) + "&type=:text/plain")} + "(text)"]]])]]]])) + +(defn list-emails + [context] + (http/ok (render-emails-list) + {:content-type "text/html; charset=utf-8"})) + +(defn- render-email + [type content] + (if (= type :text/html) + content + (html5 + [:pre content]))) + +(defn show-email + [{params :query-params}] + (let [id (edn/read-string (:id params)) + type (or (edn/read-string (:type params)) :text/html) + params (-> (get +available-emails+ id) + (assoc :email/name id)) + email (emails/render params) + content (->> (:body email) + (filter #(= (:uxbox.emails.core/type %) type)) + (first) + (:content))] + (-> (render-email type content) + (http/ok {:content-type "text/html; charset=utf-8"})))) diff --git a/backend/src/uxbox/frontend/errors.clj b/backend/src/uxbox/frontend/errors.clj new file mode 100644 index 000000000..b7ffe1143 --- /dev/null +++ b/backend/src/uxbox/frontend/errors.clj @@ -0,0 +1,74 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.errors + "A errors handling for frontend api." + (:require [catacumba.core :as ct] + [catacumba.http :as http] + [uxbox.util.response :refer (rsp)])) + +(defmulti handle-exception #(:type (ex-data %))) + +(defmethod handle-exception :validation + [err] + (println "\n*********** stack trace ***********") + (.printStackTrace err) + (println "\n********* end stack trace *********") + (let [response (ex-data err)] + (http/bad-request (rsp response)))) + +(defmethod handle-exception :default + [err] + (println "\n*********** stack trace ***********") + (.printStackTrace err) + (println "\n********* end stack trace *********") + (let [response (ex-data err)] + (http/internal-server-error (rsp response)))) + +;; --- Entry Point + +(defn- handle-data-access-exception + [err] + (let [err (.getCause err) + state (.getSQLState err) + message (.getMessage err)] + (case state + "P0002" + (-> (rsp {:message message + :payload nil + :type :occ}) + (http/precondition-failed)) + + (do + (.printStackTrace err) + (-> (rsp {:message message + :type :unexpected + :payload nil}) + (http/internal-server-error)))))) + +(defn- handle-unexpected-exception + [err] + (.printStackTrace err) + (let [message (.getMessage err)] + (-> (rsp {:message message + :type :unexpected + :payload nil}) + (http/internal-server-error)))) + +(defn handler + [context err] + (cond + (instance? clojure.lang.ExceptionInfo err) + (handle-exception err) + + (instance? java.util.concurrent.CompletionException err) + (handler context (.getCause err)) + + (instance? org.jooq.exception.DataAccessException err) + (handle-data-access-exception err) + + :else + (handle-unexpected-exception err))) diff --git a/backend/src/uxbox/frontend/icons.clj b/backend/src/uxbox/frontend/icons.clj new file mode 100644 index 000000000..c69c41ef4 --- /dev/null +++ b/backend/src/uxbox/frontend/icons.clj @@ -0,0 +1,156 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.icons + (:require [clojure.spec :as s] + [promesa.core :as p] + [catacumba.http :as http] + [storages.core :as st] + [uxbox.util.spec :as us] + [uxbox.services :as sv] + [uxbox.util.response :refer (rsp)] + [uxbox.util.uuid :as uuid])) + +;; --- Constants & Config + +(s/def ::collection (s/nilable ::us/uuid-string)) + +(s/def ::width (s/and number? pos?)) +(s/def ::height (s/and number? pos?)) +(s/def ::view-box (s/and (s/coll-of number?) + #(= 4 (count %)) + vector?)) + +(s/def ::mimetype string?) +(s/def ::metadata + (s/keys :opt-un [::width ::height ::view-box ::mimetype])) + +(s/def ::content string?) + +;; --- Create Collection + +(s/def ::create-collection + (s/keys :req-un [::us/name] :opt-un [::us/id])) + +(defn create-collection + [{user :identity data :data}] + (let [data (us/conform ::create-collection data) + message (assoc data + :type :create-icon-collection + :user user)] + (->> (sv/novelty message) + (p/map (fn [result] + (let [loc (str "/api/library/icons/" (:id result))] + (http/created loc (rsp result)))))))) + +;; --- Update Collection + +(s/def ::update-collection + (s/merge ::create-collection (s/keys :req-un [::us/version]))) + +(defn update-collection + [{user :identity params :route-params data :data}] + (let [data (us/conform ::update-collection data) + message (assoc data + :id (uuid/from-string (:id params)) + :type :update-icon-collection + :user user)] + (-> (sv/novelty message) + (p/then #(http/ok (rsp %)))))) + +;; --- Delete Collection + +(defn delete-collection + [{user :identity params :route-params}] + (let [message {:id (uuid/from-string (:id params)) + :type :delete-icon-collection + :user user}] + (-> (sv/novelty message) + (p/then (fn [v] (http/no-content)))))) + +;; --- List collections + +(defn list-collections + [{user :identity}] + (let [params {:user user + :type :list-icon-collections}] + (-> (sv/query params) + (p/then #(http/ok (rsp %)))))) + +;; --- Create Icon + +(s/def ::create-icon + (s/keys :req-un [::metadata ::us/name ::metadata ::content] + :opt-un [::us/id ::collection])) + +(defn create-icon + [{user :identity data :data :as request}] + (let [{:keys [id name content metadata collection]} (us/conform ::create-icon data) + id (or id (uuid/random))] + (->> (sv/novelty {:id id + :type :create-icon + :user user + :name name + :collection collection + :metadata metadata + :content content}) + (p/map (fn [entry] + (let [loc (str "/api/library/icons/" (:id entry))] + (http/created loc (rsp entry)))))))) + +;; --- Update Icon + +(s/def ::update-icon + (s/keys :req-un [::us/name ::us/version ::collection] :opt-un [::us/id])) + +(defn update-icon + [{user :identity params :route-params data :data}] + (let [data (us/conform ::update-icon data) + message (assoc data + :id (uuid/from-string (:id params)) + :type :update-icon + :user user)] + (->> (sv/novelty message) + (p/map #(http/ok (rsp %)))))) + +;; --- Copy Icon + +(s/def ::copy-icon + (s/keys :req-un [:us/id ::collection])) + +(defn copy-icon + [{user :identity data :data}] + (let [data (us/conform ::copy-icon data) + message (assoc data + :user user + :type :copy-icon)] + (->> (sv/novelty message) + (p/map #(http/ok (rsp %)))))) + +;; --- Delete Icon + +(defn delete-icon + [{user :identity params :route-params}] + (let [message {:id (uuid/from-string (:id params)) + :type :delete-icon + :user user}] + (->> (sv/novelty message) + (p/map (fn [v] (http/no-content)))))) + +;; --- List collections + +(s/def ::list-icons + (s/keys :opt-un [::us/id])) + +(defn list-icons + [{user :identity route-params :route-params}] + (let [{:keys [id]} (us/conform ::list-icons route-params) + params {:collection id + :type :list-icons + :user user}] + (->> (sv/query params) + (p/map rsp) + (p/map http/ok)))) diff --git a/backend/src/uxbox/frontend/images.clj b/backend/src/uxbox/frontend/images.clj new file mode 100644 index 000000000..95708d7e9 --- /dev/null +++ b/backend/src/uxbox/frontend/images.clj @@ -0,0 +1,196 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.images + (:require [clojure.spec :as s] + [promesa.core :as p] + [catacumba.http :as http] + [storages.core :as st] + [storages.util :as path] + [uxbox.media :as media] + [uxbox.images :as images] + [uxbox.util.spec :as us] + [uxbox.services :as sv] + [uxbox.util.response :refer (rsp)] + [uxbox.util.uuid :as uuid])) + +;; --- Constants & Config + +(s/def ::file ::us/uploaded-file) +(s/def ::width ::us/integer-string) +(s/def ::height ::us/integer-string) +(s/def ::collection (s/nilable ::us/uuid-string)) +(s/def ::mimetype string?) + +(def +thumbnail-options+ {:src :path + :dst :thumbnail + :size [300 110] + :quality 92 + :format "webp"}) + +(def populate-thumbnails + #(images/populate-thumbnails % +thumbnail-options+)) + +(def populate-urls + #(images/populate-urls % media/images-storage :path :url)) + +;; --- Create Collection + +(s/def ::create-collection + (s/keys :req-un [::us/name] :opt-un [::us/id])) + +(defn create-collection + [{user :identity data :data}] + (let [data (us/conform ::create-collection data) + message (assoc data + :type :create-image-collection + :user user)] + (->> (sv/novelty message) + (p/map (fn [result] + (let [loc (str "/api/library/images/" (:id result))] + (http/created loc (rsp result)))))))) + +;; --- Update Collection + +(s/def ::update-collection + (s/merge ::create-collection (s/keys :req-un [::us/version]))) + +(defn update-collection + [{user :identity params :route-params data :data}] + (let [data (us/conform ::update-collection data) + message (assoc data + :id (uuid/from-string (:id params)) + :type :update-image-collection + :user user)] + (-> (sv/novelty message) + (p/then #(http/ok (rsp %)))))) + +;; --- Delete Collection + +(defn delete-collection + [{user :identity params :route-params}] + (let [message {:id (uuid/from-string (:id params)) + :type :delete-image-collection + :user user}] + (-> (sv/novelty message) + (p/then (fn [v] (http/no-content)))))) + +;; --- List collections + +(defn list-collections + [{user :identity}] + (let [params {:user user + :type :list-image-collections}] + (-> (sv/query params) + (p/then #(http/ok (rsp %)))))) + +;; --- Retrieve Image + +(s/def ::retrieve-image + (s/keys :req-un [::us/id])) + +(defn retrieve-image + [{user :identity params :route-params}] + (let [params (us/conform ::retrieve-image params) + params (assoc params :user user :type :retrieve-image)] + (->> (sv/query params) + (p/map (fn [result] + (if result + (-> (populate-thumbnails result) + (populate-urls) + (rsp) + (http/ok)) + (http/not-found ""))))))) + +;; --- Create Image + +(s/def ::create-image + (s/keys :req-un [::file ::width ::height ::mimetype] + :opt-un [::us/id ::collection])) + +(defn create-image + [{user :identity data :data}] + (let [{:keys [file id width height + mimetype collection]} (us/conform ::create-image data) + id (or id (uuid/random)) + filename (path/base-name file) + storage media/images-storage] + (letfn [(persist-image-entry [path] + (sv/novelty {:id id + :type :create-image + :user user + :width width + :height height + :mimetype mimetype + :collection collection + :name filename + :path (str path)})) + (create-response [entry] + (let [loc (str "/api/library/images/" (:id entry))] + (http/created loc (rsp entry))))] + (->> (st/save storage filename file) + (p/mapcat persist-image-entry) + (p/map populate-thumbnails) + (p/map populate-urls) + (p/map create-response))))) + +;; --- Update Image + +(s/def ::update-image + (s/keys :req-un [::us/name ::us/version ::collection] :opt-un [::us/id])) + +(defn update-image + [{user :identity params :route-params data :data}] + (let [data (us/conform ::update-image data) + message (assoc data + :id (uuid/from-string (:id params)) + :type :update-image + :user user)] + (->> (sv/novelty message) + (p/map populate-thumbnails) + (p/map populate-urls) + (p/map #(http/ok (rsp %)))))) + +;; --- Copy Image + +(s/def ::copy-image + (s/keys :req-un [::us/id ::collection])) + +(defn copy-image + [{user :identity data :data}] + (let [data (us/conform ::copy-image data) + params (assoc data :user user :type :copy-image)] + (->> (sv/novelty params) + (p/map populate-thumbnails) + (p/map populate-urls) + (p/map #(http/ok (rsp %)))))) + +;; --- Delete Image + +(defn delete-image + [{user :identity params :route-params}] + (let [message {:id (uuid/from-string (:id params)) + :type :delete-image + :user user}] + (->> (sv/novelty message) + (p/map (fn [v] (http/no-content)))))) + +;; --- List collections + +(s/def ::list-images + (s/keys :opt-un [::us/id])) + +(defn list-images + [{user :identity route-params :route-params}] + (let [{:keys [id]} (us/conform ::list-images route-params) + params {:collection id + :type :list-images + :user user}] + (->> (sv/query params) + (p/map (partial map populate-thumbnails)) + (p/map (partial map populate-urls)) + (p/map rsp) + (p/map http/ok)))) diff --git a/backend/src/uxbox/frontend/kvstore.clj b/backend/src/uxbox/frontend/kvstore.clj new file mode 100644 index 000000000..968453f4f --- /dev/null +++ b/backend/src/uxbox/frontend/kvstore.clj @@ -0,0 +1,60 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.kvstore + (:refer-clojure :exclude [update]) + (:require [clojure.spec :as s] + [promesa.core :as p] + [catacumba.http :as http] + [uxbox.media :as media] + [uxbox.util.spec :as us] + [uxbox.services :as sv] + [uxbox.util.response :refer (rsp)] + [uxbox.util.uuid :as uuid])) + +(s/def ::version integer?) +(s/def ::key string?) +(s/def ::value any?) + +;; --- Retrieve + +(s/def ::retrieve (s/keys :req-un [::key])) + +(defn retrieve + [{user :identity params :route-params}] + (let [data (us/conform ::retrieve params) + params (assoc data + :type :retrieve-kvstore + :user user)] + (->> (sv/query params) + (p/map #(http/ok (rsp %)))))) + +;; --- Update (or Create) + +(s/def ::update (s/keys :req-un [::key ::value] + :opt-un [::version])) + +(defn update + [{user :identity data :data}] + (let [data (us/conform ::update data) + params (assoc data + :type :update-kvstore + :user user)] + (->> (sv/novelty params) + (p/map #(http/ok (rsp %)))))) + +;; --- Delete + +(s/def ::delete (s/keys :req-un [::key])) + +(defn delete + [{user :identity params :route-params}] + (let [data (us/conform ::delete params) + params (assoc data + :type :delete-kvstore + :user user)] + (->> (sv/novelty params) + (p/map (fn [_] (http/no-content)))))) diff --git a/backend/src/uxbox/frontend/pages.clj b/backend/src/uxbox/frontend/pages.clj new file mode 100644 index 000000000..d6cfb39e9 --- /dev/null +++ b/backend/src/uxbox/frontend/pages.clj @@ -0,0 +1,119 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.pages + (:require [clojure.spec :as s] + [promesa.core :as p] + [catacumba.http :as http] + [uxbox.util.spec :as us] + [uxbox.services :as sv] + [uxbox.util.response :refer (rsp)] + [uxbox.util.uuid :as uuid])) + +;; --- List Pages + +(defn list-pages-by-project + [{user :identity params :route-params}] + (let [params {:user user + :project (uuid/from-string (:id params)) + :type :list-pages-by-project}] + (-> (sv/query params) + (p/then #(http/ok (rsp %)))))) + +;; --- Create Page + +(s/def ::data any?) +(s/def ::metadata any?) +(s/def ::project ::us/id) +(s/def ::create-page + (s/keys :req-un [::data ::metadata ::project ::us/name] + :opt-un [::us/id])) + +(defn create-page + [{user :identity data :data}] + (let [data (us/conform ::create-page data) + message (assoc data + :type :create-page + :user user)] + (->> (sv/novelty message) + (p/map (fn [result] + (let [loc (str "/api/pages/" (:id result))] + (http/created loc (rsp result)))))))) + +;; --- Update Page + +(s/def ::update-page + (s/merge ::create-page (s/keys :req-un [::us/version]))) + +(defn update-page + [{user :identity params :route-params data :data}] + (let [data (us/conform ::update-page data) + message (assoc data + :id (uuid/from-string (:id params)) + :type :update-page + :user user)] + (->> (sv/novelty message) + (p/map #(http/ok (rsp %)))))) + +;; --- Update Page Metadata + +(s/def ::update-page-metadata + (s/keys :req-un [::us/id ::metadata ::project ::us/name])) + +(defn update-page-metadata + [{user :identity params :route-params data :data}] + (let [data (us/conform ::update-page-metadata data) + message (assoc data + :id (uuid/from-string (:id params)) + :type :update-page-metadata + :user user)] + (->> (sv/novelty message) + (p/map #(http/ok (rsp %)))))) + +;; --- Delete Page + +(defn delete-page + [{user :identity params :route-params}] + (let [message {:id (uuid/from-string (:id params)) + :type :delete-page + :user user}] + (-> (sv/novelty message) + (p/then (fn [v] (http/no-content)))))) + +;; --- Retrieve Page History + +(s/def ::max (s/and ::us/integer-string ::us/positive-integer)) +(s/def ::since ::us/integer-string) +(s/def ::pinned ::us/boolean-string) + +(s/def ::retrieve-page-history + (s/keys :opt-un [::max ::since ::pinned])) + +(defn retrieve-page-history + [{user :identity params :route-params query :query-params}] + (let [query (us/conform ::retrieve-page-history query) + message (assoc query + :id (uuid/from-string (:id params)) + :type :list-page-history + :user user)] + (->> (sv/query message) + (p/map #(http/ok (rsp %)))))) + +;; --- Update Page History + +(s/def ::label string?) +(s/def ::update-page-history + (s/keys :req-un [::label ::pinned])) + +(defn update-page-history + [{user :identity params :route-params data :data}] + (let [data (us/conform ::update-page-history data) + message (assoc data + :type :update-page-history + :id (uuid/from-string (:hid params)) + :user user)] + (->> (sv/novelty message) + (p/map #(http/ok (rsp %)))))) diff --git a/backend/src/uxbox/frontend/projects.clj b/backend/src/uxbox/frontend/projects.clj new file mode 100644 index 000000000..645ca8c4d --- /dev/null +++ b/backend/src/uxbox/frontend/projects.clj @@ -0,0 +1,73 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.projects + (:require [clojure.spec :as s] + [promesa.core :as p] + [catacumba.http :as http] + [uxbox.util.spec :as us] + [uxbox.services :as sv] + [uxbox.util.response :refer (rsp)] + [uxbox.util.uuid :as uuid])) + +;; --- List Projects + +(defn list-projects + [{user :identity}] + (let [message {:user user :type :list-projects}] + (->> (sv/query message) + (p/map #(http/ok (rsp %)))))) + +;; --- Create Projects + +(s/def ::create-project + (s/keys :req-un [::us/name] :opt-un [::us/id])) + +(defn create-project + [{user :identity data :data}] + (let [data (us/conform ::create-project data) + message (assoc data + :type :create-project + :user user)] + (->> (sv/novelty message) + (p/map (fn [result] + (let [loc (str "/api/projects/" (:id result))] + (http/created loc (rsp result)))))))) + +;; --- Update Project + +(s/def ::update-project + (s/keys :req-un [::us/name ::us/version])) + +(defn update-project + [{user :identity params :route-params data :data}] + (let [data (us/conform ::update-project data) + message (assoc data + :id (uuid/from-string (:id params)) + :type :update-project + :user user)] + (-> (sv/novelty message) + (p/then #(http/ok (rsp %)))))) + +;; --- Delete Project + +(defn delete-project + [{user :identity params :route-params}] + (let [message {:id (uuid/from-string (:id params)) + :type :delete-project + :user user}] + (-> (sv/novelty message) + (p/then (fn [v] (http/no-content)))))) + + +;; --- Retrieve project + +(defn retrieve-project-by-share-token + [{params :route-params}] + (let [message {:token (:token params) + :type :retrieve-project-by-share-token}] + (->> (sv/query message) + (p/map #(http/ok (rsp %)))))) diff --git a/backend/src/uxbox/frontend/svgparse.clj b/backend/src/uxbox/frontend/svgparse.clj new file mode 100644 index 000000000..f8198f38c --- /dev/null +++ b/backend/src/uxbox/frontend/svgparse.clj @@ -0,0 +1,22 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.svgparse + "A frontend exposed endpoints for svgparse functionality." + (:require [clojure.spec :as s] + [promesa.core :as p] + [catacumba.http :as http] + [uxbox.util.spec :as us] + [uxbox.services :as sv] + [uxbox.util.response :refer (rsp)] + [uxbox.util.uuid :as uuid])) + +(defn parse + [{body :body :as context}] + (let [message {:data (slurp body) + :type :parse-svg}] + (->> (sv/query message) + (p/map #(http/ok (rsp %)))))) diff --git a/backend/src/uxbox/frontend/users.clj b/backend/src/uxbox/frontend/users.clj new file mode 100644 index 000000000..31467d1a0 --- /dev/null +++ b/backend/src/uxbox/frontend/users.clj @@ -0,0 +1,149 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.frontend.users + (:require [clojure.spec :as s] + [promesa.core :as p] + [catacumba.http :as http] + [storages.core :as st] + [storages.util :as path] + [uxbox.media :as media] + [uxbox.images :as images] + [uxbox.util.spec :as us] + [uxbox.services :as sv] + [uxbox.services.users :as svu] + [uxbox.util.response :refer (rsp)] + [uxbox.util.uuid :as uuid])) + +;; --- Helpers + +(defn- resolve-thumbnail + [user] + (let [opts {:src :photo + :dst :photo + :size [100 100] + :quality 90 + :format "jpg"}] + (images/populate-thumbnails user opts))) + +;; --- Retrieve Profile + +(defn retrieve-profile + [{user :identity}] + (let [message {:user user + :type :retrieve-profile}] + (->> (sv/query message) + (p/map resolve-thumbnail) + (p/map #(http/ok (rsp %)))))) + +;; --- Update Profile + +(s/def ::fullname string?) +(s/def ::metadata any?) +(s/def ::update-profile + (s/keys :req-un [::us/id ::us/username ::us/email + ::fullname ::metadata])) + +(defn update-profile + [{user :identity data :data}] + (let [data (us/conform ::update-profile data) + message (assoc data + :type :update-profile + :user user)] + (->> (sv/novelty message) + (p/map resolve-thumbnail) + (p/map #(http/ok (rsp %)))))) + +;; --- Update Password + +(s/def ::old-password ::us/password) +(s/def ::update-password + (s/keys :req-un [::us/password ::old-password])) + +(defn update-password + [{user :identity data :data}] + (let [data (us/conform ::update-password data) + message (assoc data + :type :update-profile-password + :user user)] + (-> (sv/novelty message) + (p/then #(http/ok (rsp %)))))) + +;; --- Update Profile Photo + +(s/def ::file ::us/uploaded-file) +(s/def ::update-photo (s/keys :req-un [::file])) + +(defn update-photo + [{user :identity data :data}] + (letfn [(store-photo [file] + (let [filename (path/base-name file) + storage media/images-storage] + (st/save storage filename file))) + (assign-photo [path] + (sv/novelty {:user user + :path (str path) + :type :update-profile-photo})) + (create-response [_] + (http/no-content))] + (let [{:keys [file]} (us/conform ::update-photo data)] + (->> (store-photo file) + (p/mapcat assign-photo) + (p/map create-response))))) + +;; --- Register User + +(s/def ::register + (s/keys :req-un [::us/username ::us/email ::us/password ::fullname])) + +(defn register-user + [{data :data}] + (let [data (us/conform ::register data) + message (assoc data :type :register-profile)] + (->> (sv/novelty message) + (p/map #(http/ok (rsp %)))))) + + +;; --- Request Password Recovery + +;; FIXME: rename for consistency + +(s/def ::request-recovery + (s/keys :req-un [::us/username])) + +(defn request-recovery + [{data :data}] + (let [data (us/conform ::request-recovery data) + message (assoc data :type :request-profile-password-recovery)] + (->> (sv/novelty message) + (p/map (fn [_] (http/no-content)))))) + +;; --- Password Recovery + +;; FIXME: rename for consistency + +(s/def ::token string?) +(s/def ::password-recovery + (s/keys :req-un [::token ::us/password])) + +(defn recover-password + [{data :data}] + (let [data (us/conform ::password-recovery data) + message (assoc data :type :recover-profile-password)] + (->> (sv/novelty message) + (p/map (fn [_] (http/no-content)))))) + +;; --- Valiadate Recovery Token + +(defn validate-recovery-token + [{params :route-params}] + (let [message {:type :validate-profile-password-recovery-token + :token (:token params)}] + (->> (sv/query message) + (p/map (fn [v] + (if v + (http/no-content) + (http/not-found ""))))))) diff --git a/backend/src/uxbox/images.clj b/backend/src/uxbox/images.clj new file mode 100644 index 000000000..af9faad68 --- /dev/null +++ b/backend/src/uxbox/images.clj @@ -0,0 +1,66 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.images + "Image postprocessing." + (:require [storages.core :as st] + [storages.util :as path] + [clojure.spec :as s] + [uxbox.util.spec :as us] + [uxbox.media :as media] + [uxbox.util.images :as images] + [uxbox.util.data :refer (dissoc-in)])) + +;; FIXME: add spec for thumbnail config + +(defn make-thumbnail + [path {:keys [size format quality] :as cfg}] + (let [parent (path/parent path) + [filename ext] (path/split-ext path) + + suffix-parts [(nth size 0) (nth size 1) quality format] + final-name (apply str filename "-" (interpose "." suffix-parts)) + final-path (path/path parent final-name) + + images-storage media/images-storage + thumbs-storage media/thumbnails-storage] + (if @(st/exists? thumbs-storage final-path) + (str (st/public-url thumbs-storage final-path)) + (if @(st/exists? images-storage path) + (let [datapath @(st/lookup images-storage path) + content (images/thumbnail datapath cfg) + path @(st/save thumbs-storage final-path content)] + (str (st/public-url thumbs-storage path))) + nil)))) + +(defn populate-thumbnail + [entry {:keys [src dst] :as cfg}] + (assert (map? entry) "`entry` should be map") + + (let [src (if (vector? src) src [src]) + dst (if (vector? dst) dst [dst]) + src (get-in entry src)] + (if (empty? src) + entry + (assoc-in entry dst (make-thumbnail src cfg))))) + +(defn populate-thumbnails + [entry & settings] + (reduce populate-thumbnail entry settings)) + +(defn populate-urls + [entry storage src dst] + (assert (map? entry) "`entry` should be map") + (assert (st/storage? storage) "`storage` should be a valid storage instance.") + (let [src (if (vector? src) src [src]) + dst (if (vector? dst) dst [dst]) + value (get-in entry src)] + (if (empty? value) + entry + (let [url (str (st/public-url storage value))] + (-> entry + (dissoc-in src) + (assoc-in dst url)))))) diff --git a/backend/src/uxbox/locks.clj b/backend/src/uxbox/locks.clj new file mode 100644 index 000000000..aaa881068 --- /dev/null +++ b/backend/src/uxbox/locks.clj @@ -0,0 +1,21 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.locks + "Advirsory locks for specific handling concurrent modifications + on particular objects in the database." + (:require [suricatta.core :as sc]) + (:import clojure.lang.Murmur3)) + +(defn- uuid->long + [v] + (Murmur3/hashUnencodedChars (str v))) + +(defn acquire! + [conn v] + (let [id (uuid->long v)] + (sc/execute conn ["select pg_advisory_xact_lock(?);" id]) + nil)) diff --git a/backend/src/uxbox/main.clj b/backend/src/uxbox/main.clj new file mode 100644 index 000000000..a5ae1346c --- /dev/null +++ b/backend/src/uxbox/main.clj @@ -0,0 +1,92 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.main + (:require [clojure.tools.namespace.repl :as repl] + [clojure.walk :refer [macroexpand-all]] + [clojure.pprint :refer [pprint]] + [clojure.test :as test] + [clojure.java.io :as io] + [mount.core :as mount] + [buddy.core.codecs :as codecs] + [buddy.core.codecs.base64 :as b64] + [buddy.core.nonce :as nonce] + [uxbox.config :as cfg] + [uxbox.migrations] + [uxbox.db] + [uxbox.frontend] + [uxbox.scheduled-jobs]) + (:gen-class)) + +;; --- Development Stuff + +(defn- start + [] + (mount/start)) + +(defn- start-minimal + [] + (-> (mount/only #{#'uxbox.config/config + #'uxbox.db/datasource + #'uxbox.migrations/migrations}) + (mount/start))) + +(defn- stop + [] + (mount/stop)) + +(defn- refresh + [] + (stop) + (repl/refresh)) + +(defn- refresh-all + [] + (stop) + (repl/refresh-all)) + +(defn- go + "starts all states defined by defstate" + [] + (start) + :ready) + +(defn- reset + [] + (stop) + (repl/refresh :after 'uxbox.main/start)) + +(defn make-secret + [] + (-> (nonce/random-bytes 64) + (b64/encode true) + (codecs/bytes->str))) + +;; --- Entry point (only for uberjar) + +(defn test-vars + [& vars] + (repl/refresh) + (test/test-vars + (map (fn [sym] + (require (symbol (namespace sym))) + (resolve sym)) + vars))) + +(defn test-ns + [ns] + (repl/refresh) + (test/test-ns ns)) + +(defn test-all + ([] (test/run-all-tests #"^uxbox.tests.*")) + ([re] (test/run-all-tests re))) + +;; --- Entry point (only for uberjar) + +(defn -main + [& args] + (mount/start)) diff --git a/backend/src/uxbox/media.clj b/backend/src/uxbox/media.clj new file mode 100644 index 000000000..1a88fac27 --- /dev/null +++ b/backend/src/uxbox/media.clj @@ -0,0 +1,39 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.media + "A media storage impl for uxbox." + (:require [mount.core :as mount :refer (defstate)] + [clojure.java.io :as io] + [cuerdas.core :as str] + [storages.core :as st] + [storages.fs.local :refer (filesystem)] + [storages.fs.misc :refer (hashed scoped)] + [uxbox.config :refer (config)])) + +;; --- State + +(defstate static-storage + :start (let [{:keys [basedir baseuri]} (:static config)] + (filesystem {:basedir basedir :baseuri baseuri}))) + +(defstate media-storage + :start (let [{:keys [basedir baseuri]} (:media config)] + (filesystem {:basedir basedir :baseuri baseuri}))) + +(defstate images-storage + :start (-> media-storage + (scoped "images") + (hashed))) + +(defstate thumbnails-storage + :start (scoped media-storage "thumbs")) + +;; --- Public Api + +(defn resolve-asset + [path] + (str (st/public-url static-storage path))) diff --git a/backend/src/uxbox/migrations.clj b/backend/src/uxbox/migrations.clj new file mode 100644 index 000000000..4036f9317 --- /dev/null +++ b/backend/src/uxbox/migrations.clj @@ -0,0 +1,78 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.migrations + (:require [mount.core :as mount :refer (defstate)] + [migrante.core :as mg :refer (defmigration)] + [uxbox.db :as db] + [uxbox.config :as cfg] + [uxbox.util.template :as tmpl])) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Migrations +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(defmigration utils-0000 + "Create a initial version of txlog table." + :up (mg/resource "migrations/0000.main.up.sql")) + +(defmigration txlog-0001 + "Create a initial version of txlog table." + :up (mg/resource "migrations/0001.txlog.up.sql")) + +(defmigration auth-0002 + "Create initial auth related tables." + :up (mg/resource "migrations/0002.auth.up.sql")) + +(defmigration projects-0003 + "Create initial tables for projects." + :up (mg/resource "migrations/0003.projects.up.sql")) + +(defmigration pages-0004 + "Create initial tables for pages." + :up (mg/resource "migrations/0004.pages.up.sql")) + +(defmigration kvstore-0005 + "Create initial tables for kvstore." + :up (mg/resource "migrations/0005.kvstore.up.sql")) + +(defmigration emails-queue-0006 + "Create initial tables for emails queue." + :up (mg/resource "migrations/0006.emails.up.sql")) + +(defmigration images-0007 + "Create initial tables for image collections." + :up (mg/resource "migrations/0007.images.up.sql")) + +(defmigration icons-0008 + "Create initial tables for image collections." + :up (mg/resource "migrations/0008.icons.up.sql")) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Entry point +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(def +migrations+ + {:name :uxbox-main + :steps [[:0000 utils-0000] + [:0001 txlog-0001] + [:0002 auth-0002] + [:0003 projects-0003] + [:0004 pages-0004] + [:0005 kvstore-0005] + [:0006 emails-queue-0006] + [:0007 images-0007] + [:0008 icons-0008]]}) + +(defn- migrate + [] + (let [options (:migrations cfg/config {})] + (with-open [mctx (mg/context db/datasource options)] + (mg/migrate mctx +migrations+) + nil))) + +(defstate migrations + :start (migrate)) diff --git a/backend/src/uxbox/portation.clj b/backend/src/uxbox/portation.clj new file mode 100644 index 000000000..1cef68743 --- /dev/null +++ b/backend/src/uxbox/portation.clj @@ -0,0 +1,114 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.portation + "Support for export/import operations of projects." + (:refer-clojure :exclude [with-open]) + (:require [clojure.java.io :as io] + [suricatta.core :as sc] + [storages.util :as path] + [uxbox.db :as db] + [uxbox.sql :as sql] + [uxbox.util.uuid :as uuid] + [uxbox.util.closeable :refer (with-open)] + [uxbox.util.tempfile :as tmpfile] + [uxbox.util.transit :as t] + [uxbox.util.snappy :as snappy])) + +;; --- Export + +(defn- write-project + [conn writer id] + (let [sql (sql/get-project-by-id {:id id}) + result (sc/fetch-one conn sql)] + (when-not result + (ex-info "No project found with specified id" {:id id})) + (t/write! writer {::type ::project ::payload result}))) + +(defn- write-pages + [conn writer id] + (let [sql (sql/get-pages-for-project {:project id}) + results (sc/fetch conn sql)] + (run! #(t/write! writer {::type ::page ::payload %}) results))) + +(defn- write-pages-history + [conn writer id] + (let [sql (sql/get-page-history-for-project {:project id}) + results (sc/fetch conn sql)] + (run! #(t/write! writer {::type ::page-history ::payload %}) results))) + +(defn- write-data + [path id] + (with-open [ostream (io/output-stream path) + zstream (snappy/output-stream ostream) + conn (db/connection)] + (let [writer (t/writer zstream {:type :msgpack})] + (sc/atomic conn + (write-project conn writer id) + (write-pages conn writer id) + (write-pages-history conn writer id))))) + +(defn export + "Given an id, returns a path to a temporal file with the exported + bundle of the specified project." + [id] + (let [path (tmpfile/create)] + (write-data path id) + path)) + +;; --- Import + +(defn- read-entry + [reader] + (try + (t/read! reader) + (catch RuntimeException e + (let [cause (.getCause e)] + (if (instance? java.io.EOFException cause) + ::eof + (throw e)))))) + +(defn- persist-project + [conn project] + (let [sql (sql/create-project project)] + (sc/execute conn sql))) + +(defn- persist-page + [conn page] + (let [sql (sql/create-page page)] + (sc/execute conn sql))) + +(defn- persist-page-history + [conn history] + (let [sql (sql/create-page-history history)] + (sc/execute conn sql))) + +(defn- persist-entry + [conn entry] + (let [payload (::payload entry) + type (::type entry)] + (case type + ::project (persist-project conn payload) + ::page (persist-page conn payload) + ::page-history (persist-page-history conn payload)))) + +(defn- read-data + [conn reader] + (loop [entry (read-entry reader)] + (when (not= entry ::eof) + (persist-entry conn entry) + (recur (read-entry reader))))) + +(defn import! + "Given a path to the previously exported bundle, try to import it." + [path] + (with-open [istream (io/input-stream (path/path path)) + zstream (snappy/input-stream istream) + conn (db/connection)] + (let [reader (t/reader zstream {:type :msgpack})] + (sc/atomic conn + (read-data conn reader) + nil)))) diff --git a/backend/src/uxbox/scheduled_jobs.clj b/backend/src/uxbox/scheduled_jobs.clj new file mode 100644 index 000000000..952c6eeac --- /dev/null +++ b/backend/src/uxbox/scheduled_jobs.clj @@ -0,0 +1,23 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.scheduled-jobs + "Time-based scheduled jobs." + (:require [mount.core :as mount :refer (defstate)] + [uxbox.config :as cfg] + [uxbox.db] + [uxbox.util.quartz :as qtz])) + +(defn- initialize + [] + (let [nss #{'uxbox.scheduled-jobs.garbage + 'uxbox.scheduled-jobs.emails}] + (-> (qtz/scheduler) + (qtz/start! {:search-on nss})))) + +(defstate scheduler + :start (initialize) + :stop (qtz/stop! scheduler)) diff --git a/backend/src/uxbox/scheduled_jobs/emails.clj b/backend/src/uxbox/scheduled_jobs/emails.clj new file mode 100644 index 000000000..abe4c7976 --- /dev/null +++ b/backend/src/uxbox/scheduled_jobs/emails.clj @@ -0,0 +1,123 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.scheduled-jobs.emails + "Email sending async tasks." + (:require [clojure.tools.logging :as log] + [suricatta.core :as sc] + [postal.core :as postal] + [uxbox.db :as db] + [uxbox.config :as cfg] + [uxbox.sql :as sql] + [uxbox.util.quartz :as qtz] + [uxbox.util.blob :as blob] + [uxbox.util.transit :as t] + [uxbox.util.data :as data])) + +;; --- Impl details + +(defn- decode-email-data + [{:keys [data] :as result}] + (merge result (when data + {:data (-> data blob/decode t/decode)}))) + +(defn- fetch-pending-emails + [conn] + (let [sqlv (sql/get-pending-emails)] + (->> (sc/fetch conn sqlv) + (map data/normalize-attrs) + (map decode-email-data)))) + +(defn- fetch-immediate-emails + [conn] + (let [sqlv (sql/get-immediate-emails)] + (->> (sc/fetch conn sqlv) + (map data/normalize-attrs) + (map decode-email-data)))) + +(defn- fetch-failed-emails + [conn] + (let [sqlv (sql/get-pending-emails)] + (->> (sc/fetch conn sqlv) + (map data/normalize-attrs) + (map decode-email-data)))) + +(defn- mark-email-as-sent + [conn id] + (let [sqlv (sql/mark-email-as-sent {:id id})] + (sc/execute conn sqlv))) + +(defn- mark-email-as-failed + [conn id] + (let [sqlv (sql/mark-email-as-failed {:id id})] + (sc/execute conn sqlv))) + +(defn- send-email-to-console + [{:keys [id data] :as entry}] + (println "******** start email:" id "**********") + (println (->> (:body data) + (filter #(= (:uxbox.emails.core/type %) :text/plain)) + (first) + (:content))) + (println "********** end email:" id "**********") + {:error :SUCCESS}) + +(defn- send-email + [{:keys [id data] :as entry}] + (let [config (:smtp cfg/config) + result (if (:noop config) + (send-email-to-console entry) + (postal/send-message config data))] + (if (= (:error result) :SUCCESS) + (log/debug "Message" id "sent successfully.") + (log/warn "Message" id "failed with:" (:message result))) + (if (= (:error result) :SUCCESS) + true + false))) + +(defn- send-emails + [conn entries] + (loop [entries entries] + (if-let [entry (first entries)] + (do (if (send-email entry) + (mark-email-as-sent conn (:id entry)) + (mark-email-as-failed conn (:id entry))) + (recur (rest entries)))))) + +;; --- Jobs + +(defn send-immediate-emails + {::qtz/interval (* 60 1 1000) ;; every 1min + ::qtz/repeat? true + ::qtz/job true} + [] + (log/info "task-send-immediate-emails...") + (with-open [conn (db/connection)] + (sc/atomic conn + (->> (fetch-immediate-emails conn) + (send-emails conn))))) + +(defn send-pending-emails + {::qtz/interval (* 60 5 1000) ;; every 5min + ::qtz/repeat? true + ::qtz/job true} + [] + (with-open [conn (db/connection)] + (sc/atomic conn + (->> (fetch-pending-emails conn) + (send-emails conn))))) + +(defn send-failed-emails + "Job that resends failed to send messages." + {::qtz/interval (* 60 5 1000) ;; every 5min + ::qtz/repeat? true + ::qtz/job true} + [] + (log/info "task-send-failed-emails...") + (with-open [conn (db/connection)] + (sc/atomic conn + (->> (fetch-failed-emails conn) + (send-emails conn))))) diff --git a/backend/src/uxbox/scheduled_jobs/garbage.clj b/backend/src/uxbox/scheduled_jobs/garbage.clj new file mode 100644 index 000000000..fedfaa180 --- /dev/null +++ b/backend/src/uxbox/scheduled_jobs/garbage.clj @@ -0,0 +1,28 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.scheduled-jobs.garbage + "Garbage Collector related tasks." + (:require [suricatta.core :as sc] + [uxbox.db :as db] + [uxbox.util.quartz :as qtz])) + +;; --- Delete projects + +;; TODO: move inline sql into resources/sql directory + +(defn clean-deleted-projects + "Task that cleans the deleted projects." + {::qtz/repeat? true + ::qtz/interval (* 1000 3600 24) + ::qtz/job true} + [] + (with-open [conn (db/connection)] + (sc/atomic conn + (let [sql (str "DELETE FROM projects " + " WHERE deleted_at is not null AND " + " (now()-deleted_at)::interval > '10 day'::interval;")] + (sc/execute conn sql))))) diff --git a/backend/src/uxbox/services.clj b/backend/src/uxbox/services.clj new file mode 100644 index 000000000..30d095ae8 --- /dev/null +++ b/backend/src/uxbox/services.clj @@ -0,0 +1,61 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services + "Main namespace for access to all uxbox services." + (:require [suricatta.core :as sc] + [executors.core :as exec] + [promesa.core :as p] + [uxbox.db :as db] + [uxbox.services.core :as core] + [uxbox.util.transit :as t] + [uxbox.util.blob :as blob])) + +;; Load relevant subnamespaces with the implementation +(load "services/auth") +(load "services/projects") +(load "services/pages") +(load "services/images") +(load "services/icons") +(load "services/kvstore") + +;; --- Implementation + +(def ^:private encode (comp blob/encode t/encode)) + +(defn- insert-txlog + [data] + (with-open [conn (db/connection)] + (let [sql (str "INSERT INTO txlog (payload) VALUES (?)") + sqlv [sql (encode data)]] + (sc/execute conn sqlv)))) + +(defn- handle-novelty + [data] + (let [rs (core/novelty data) + rs (if (p/promise? rs) rs (p/resolved rs))] + (p/map (fn [v] + (insert-txlog data) + v) rs))) + +(defn- handle-query + [data] + (let [result (core/query data)] + (if (p/promise? result) + result + (p/resolved result)))) + +;; --- Public Api + +(defn novelty + [data] + (->> (exec/submit (partial handle-novelty data)) + (p/mapcat identity))) + +(defn query + [data] + (->> (exec/submit (partial handle-query data)) + (p/mapcat identity))) diff --git a/backend/src/uxbox/services/auth.clj b/backend/src/uxbox/services/auth.clj new file mode 100644 index 000000000..00d3659af --- /dev/null +++ b/backend/src/uxbox/services/auth.clj @@ -0,0 +1,47 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services.auth + (:require [clojure.spec :as s] + [suricatta.core :as sc] + [buddy.hashers :as hashers] + [buddy.sign.jwt :as jwt] + [buddy.core.hash :as hash] + [uxbox.config :as cfg] + [uxbox.util.spec :as us] + [uxbox.db :as db] + [uxbox.services.core :as core] + [uxbox.services.users :as users] + [uxbox.util.exceptions :as ex])) + +;; --- Login + +(defn- check-user-password + [user password] + (hashers/check password (:password user))) + +(defn generate-token + [user] + (let [data {:id (:id user)} + opts (:auth-options cfg/config)] + (jwt/encrypt data cfg/secret opts))) + +(s/def ::scope string?) +(s/def ::login + (s/keys :req-un [::us/username ::us/password ::scope])) + +(defmethod core/novelty :login + [{:keys [username password scope] :as params}] + (s/assert ::login params) + (with-open [conn (db/connection)] + (let [user (users/find-user-by-username-or-email conn username)] + (when-not user + (ex/raise :type :validation + :code ::wrong-credentials)) + (if (check-user-password user password) + {:token (generate-token user)} + (ex/raise :type :validation + :code ::wrong-credentials))))) diff --git a/backend/src/uxbox/services/core.clj b/backend/src/uxbox/services/core.clj new file mode 100644 index 000000000..2b2bd7296 --- /dev/null +++ b/backend/src/uxbox/services/core.clj @@ -0,0 +1,27 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services.core + (:require [clojure.walk :as walk] + [cuerdas.core :as str] + [uxbox.util.exceptions :as ex])) + +(defmulti novelty :type) + +(defmulti query :type) + +(defmethod novelty :default + [{:keys [type] :as data}] + (ex/raise :code ::not-implemented + :message-category :novelty + :message-type type)) + +(defmethod query :default + [{:keys [type] :as data}] + (ex/raise :code ::not-implemented + :message-category :query + :message-type type)) + diff --git a/backend/src/uxbox/services/icons.clj b/backend/src/uxbox/services/icons.clj new file mode 100644 index 000000000..6c478eb0c --- /dev/null +++ b/backend/src/uxbox/services/icons.clj @@ -0,0 +1,228 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services.icons + "Icons library related services." + (:require [clojure.spec :as s] + [suricatta.core :as sc] + [uxbox.util.spec :as us] + [uxbox.sql :as sql] + [uxbox.db :as db] + [uxbox.services.core :as core] + [uxbox.util.exceptions :as ex] + [uxbox.util.transit :as t] + [uxbox.util.uuid :as uuid] + [uxbox.util.blob :as blob] + [uxbox.util.data :as data]) + (:import ratpack.form.UploadedFile + org.apache.commons.io.FilenameUtils)) + +;; --- Helpers & Specs + +(s/def ::user uuid?) +(s/def ::collection (s/nilable uuid?)) +(s/def ::width (s/and number? pos?)) +(s/def ::height (s/and number? pos?)) +(s/def ::view-box (s/and (s/coll-of number?) + #(= 4 (count %)) + vector?)) + +(s/def ::content string?) +(s/def ::mimetype string?) +(s/def ::metadata + (s/keys :opt-un [::width ::height ::view-box ::mimetype])) + +(defn decode-metadata + [{:keys [metadata] :as data}] + (if metadata + (assoc data :metadata (-> metadata blob/decode t/decode)) + data)) + +;; --- Create Collection + +(defn create-collection + [conn {:keys [id user name]}] + (let [id (or id (uuid/random)) + params {:id id :user user :name name} + sqlv (sql/create-icon-collection params)] + (-> (sc/fetch-one conn sqlv) + (data/normalize)))) + +(s/def ::create-icon-collection + (s/keys :req-un [::user ::us/name] + :opt-un [::us/id])) + +(defmethod core/novelty :create-icon-collection + [params] + (s/assert ::create-icon-collection params) + (with-open [conn (db/connection)] + (create-collection conn params))) + +;; --- Update Collection + +(defn update-collection + [conn {:keys [id user name version]}] + (let [sqlv (sql/update-icon-collection {:id id + :user user + :name name + :version version})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize)))) + +(s/def ::update-icon-collection + (s/keys :req-un [::user ::us/name ::us/version] + :opt-un [::us/id])) + +(defmethod core/novelty :update-icon-collection + [params] + (s/assert ::update-icon-collection params) + (with-open [conn (db/connection)] + (sc/apply-atomic conn update-collection params))) + +;; --- Copy Icon + +(s/def ::copy-icon + (s/keys :req-un [:us/id ::collection ::user])) + +(defn- retrieve-icon + [conn {:keys [user id]}] + (let [sqlv (sql/get-icon {:user user :id id})] + (some->> (sc/fetch-one conn sqlv) + (data/normalize-attrs)))) + +(declare create-icon) + +(defn- copy-icon + [conn {:keys [user id collection]}] + (let [icon (retrieve-icon conn {:id id :user user})] + (when-not icon + (ex/raise :type :validation + :code ::icon-does-not-exists)) + (let [params (dissoc icon :id)] + (create-icon conn params)))) + +(defmethod core/novelty :copy-icon + [params] + (s/assert ::copy-icon params) + (with-open [conn (db/connection)] + (sc/apply-atomic conn copy-icon params))) + +;; --- List Collections + +(defn get-collections-by-user + [conn user] + (let [sqlv (sql/get-icon-collections {:user user})] + (->> (sc/fetch conn sqlv) + (map data/normalize)))) + +(defmethod core/query :list-icon-collections + [{:keys [user] :as params}] + (s/assert ::user user) + (with-open [conn (db/connection)] + (get-collections-by-user conn user))) + +;; --- Delete Collection + +(defn delete-collection + [conn {:keys [id user]}] + (let [sqlv (sql/delete-icon-collection {:id id :user user})] + (pos? (sc/execute conn sqlv)))) + +(s/def ::delete-icon-collection + (s/keys :req-un [::user] + :opt-un [::us/id])) + +(defmethod core/novelty :delete-icon-collection + [params] + (s/assert ::delete-icon-collection params) + (with-open [conn (db/connection)] + (delete-collection conn params))) + +;; --- Create Icon (Upload) + +(defn create-icon + [conn {:keys [id user name collection + metadata content] :as params}] + (let [id (or id (uuid/random)) + params {:id id + :name name + :content content + :metadata (-> metadata t/encode blob/encode) + :collection collection + :user user} + sqlv (sql/create-icon params)] + (some-> (sc/fetch-one conn sqlv) + (data/normalize) + (decode-metadata)))) + +(s/def ::create-icon + (s/keys :req-un [::user ::us/name ::metadata ::content] + :opt-un [::us/id ::collection])) + +(defmethod core/novelty :create-icon + [params] + (s/assert ::create-icon params) + (with-open [conn (db/connection)] + (create-icon conn params))) + +;; --- Update Icon + +(defn update-icon + [conn {:keys [id name version user collection]}] + (let [sqlv (sql/update-icon {:id id + :collection collection + :name name + :user user + :version version})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize) + (decode-metadata)))) + +(s/def ::update-icon + (s/keys :req-un [::us/id ::user ::us/name ::us/version ::collection])) + +(defmethod core/novelty :update-icon + [params] + (s/assert ::update-icon params) + (with-open [conn (db/connection)] + (update-icon conn params))) + +;; --- Delete Icon + +(defn delete-icon + [conn {:keys [user id]}] + (let [sqlv (sql/delete-icon {:id id :user user})] + (pos? (sc/execute conn sqlv)))) + +(s/def ::delete-icon + (s/keys :req-un [::user] + :opt-un [::us/id])) + +(defmethod core/novelty :delete-icon + [params] + (s/assert ::delete-icon params) + (with-open [conn (db/connection)] + (delete-icon conn params))) + +;; --- List Icons + +(defn get-icons-by-user + [conn user collection] + (let [sqlv (if collection + (sql/get-icons-by-collection {:user user :collection collection}) + (sql/get-icons {:user user}))] + (->> (sc/fetch conn sqlv) + (map data/normalize) + (map decode-metadata)))) + +(s/def ::list-icons + (s/keys :req-un [::user ::collection])) + +(defmethod core/query :list-icons + [{:keys [user collection] :as params}] + (s/assert ::list-icons params) + (with-open [conn (db/connection)] + (get-icons-by-user conn user collection))) diff --git a/backend/src/uxbox/services/images.clj b/backend/src/uxbox/services/images.clj new file mode 100644 index 000000000..1e24accad --- /dev/null +++ b/backend/src/uxbox/services/images.clj @@ -0,0 +1,236 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services.images + "Images library related services." + (:require [clojure.spec :as s] + [promesa.core :as p] + [suricatta.core :as sc] + [buddy.core.codecs :as codecs] + [storages.core :as st] + [storages.util :as path] + [uxbox.config :as ucfg] + [uxbox.util.spec :as us] + [uxbox.sql :as sql] + [uxbox.db :as db] + [uxbox.media :as media] + [uxbox.services.core :as core] + [uxbox.util.exceptions :as ex] + [uxbox.util.transit :as t] + [uxbox.util.uuid :as uuid] + [uxbox.util.data :as data]) + (:import ratpack.form.UploadedFile + org.apache.commons.io.FilenameUtils)) + +(s/def ::width integer?) +(s/def ::height integer?) +(s/def ::mimetype string?) +(s/def ::user uuid?) +(s/def ::path string?) +(s/def ::collection (s/nilable uuid?)) + +;; --- Create Collection + +(defn create-collection + [conn {:keys [id user name]}] + (let [id (or id (uuid/random)) + params {:id id :user user :name name} + sqlv (sql/create-image-collection params)] + (-> (sc/fetch-one conn sqlv) + (data/normalize-attrs)))) + +(s/def ::create-image-collection + (s/keys :req-un [::user ::us/name] + :opt-un [::us/id])) + +(defmethod core/novelty :create-image-collection + [params] + (s/assert ::create-image-collection params) + (with-open [conn (db/connection)] + (create-collection conn params))) + +;; --- Update Collection + +(defn update-collection + [conn {:keys [id user name version]}] + (let [sqlv (sql/update-image-collection {:id id + :user user + :name name + :version version})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs)))) + +(s/def ::update-image-collection + (s/keys :req-un [::user ::us/name ::us/version] + :opt-un [::us/id])) + +(defmethod core/novelty :update-image-collection + [params] + (s/assert ::update-image-collection params) + (with-open [conn (db/connection)] + (update-collection conn params))) + +;; --- List Collections + +(defn get-collections-by-user + [conn user] + (let [sqlv (sql/get-image-collections {:user user})] + (->> (sc/fetch conn sqlv) + (map data/normalize-attrs)))) + +(defmethod core/query :list-image-collections + [{:keys [user] :as params}] + (s/assert ::user user) + (with-open [conn (db/connection)] + (get-collections-by-user conn user))) + +;; --- Delete Collection + +(defn delete-collection + [conn {:keys [id user]}] + (let [sqlv (sql/delete-image-collection {:id id :user user})] + (pos? (sc/execute conn sqlv)))) + +(s/def ::delete-image-collection + (s/keys :req-un [::user] + :opt-un [::us/id])) + +(defmethod core/novelty :delete-image-collection + [params] + (s/assert ::delete-image-collection params) + (with-open [conn (db/connection)] + (delete-collection conn params))) + +;; --- Retrieve Image + +(defn retrieve-image + [conn {:keys [user id]}] + (let [sqlv (sql/get-image {:user user :id id})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs)))) + +(s/def ::retrieve-image + (s/keys :req-un [::user ::us/id])) + +(defmethod core/query :retrieve-image + [params] + (s/assert ::retrieve-image params) + (with-open [conn (db/connection)] + (retrieve-image conn params))) + +;; --- Create Image (Upload) + +(defn create-image + [conn {:keys [id user name path collection + height width mimetype]}] + (let [id (or id (uuid/random)) + sqlv (sql/create-image {:id id + :name name + :mimetype mimetype + :path path + :width width + :height height + :collection collection + :user user})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs)))) + +(s/def ::create-image + (s/keys :req-un [::user ::us/name ::path ::width ::height ::mimetype] + :opt-un [::us/id])) + +(defmethod core/novelty :create-image + [params] + (s/assert ::create-image params) + (with-open [conn (db/connection)] + (create-image conn params))) + +;; --- Update Image + +(defn update-image + [conn {:keys [id name version user collection]}] + (let [sqlv (sql/update-image {:id id + :collection collection + :name name + :user user + :version version})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs)))) + +(s/def ::update-image + (s/keys :req-un [::user ::us/name ::us/version ::collection] + :opt-un [::us/id])) + +(defmethod core/novelty :update-image + [params] + (s/assert ::update-image params) + (with-open [conn (db/connection)] + (update-image conn params))) + +;; --- Copy Image + +(s/def ::copy-image + (s/keys :req-un [::us/id ::collection ::user])) + +(declare retrieve-image) + +(defn- copy-image + [conn {:keys [user id collection]}] + (let [image (retrieve-image conn {:id id :user user}) + storage media/images-storage] + (when-not image + (ex/raise :type :validation + :code ::image-does-not-exists)) + (let [path @(st/lookup storage (:path image)) + filename (path/base-name path) + path @(st/save storage filename path) + image (assoc image + :path (str path) + :collection collection) + image (dissoc image :id)] + (create-image conn image)))) + +(defmethod core/novelty :copy-image + [params] + (s/assert ::copy-image params) + (with-open [conn (db/connection)] + (sc/apply-atomic conn copy-image params))) + +;; --- Delete Image + +(defn delete-image + [conn {:keys [user id]}] + (let [sqlv (sql/delete-image {:id id :user user})] + (pos? (sc/execute conn sqlv)))) + +(s/def ::delete-image + (s/keys :req-un [::user] + :opt-un [::us/id])) + +(defmethod core/novelty :delete-image + [params] + (s/assert ::delete-image params) + (with-open [conn (db/connection)] + (delete-image conn params))) + +;; --- List Images + +(defn get-images-by-user + [conn user collection] + (let [sqlv (if collection + (sql/get-images-by-collection {:user user :collection collection}) + (sql/get-images {:user user}))] + (->> (sc/fetch conn sqlv) + (map data/normalize-attrs)))) + +(s/def ::list-images + (s/keys :req-un [::user ::collection])) + +(defmethod core/query :list-images + [{:keys [user collection] :as params}] + (s/assert ::list-images params) + (with-open [conn (db/connection)] + (get-images-by-user conn user collection))) diff --git a/backend/src/uxbox/services/kvstore.clj b/backend/src/uxbox/services/kvstore.clj new file mode 100644 index 000000000..4f5c1ae2a --- /dev/null +++ b/backend/src/uxbox/services/kvstore.clj @@ -0,0 +1,87 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services.kvstore + (:require [clojure.spec :as s] + [suricatta.core :as sc] + [buddy.core.codecs :as codecs] + [uxbox.config :as ucfg] + [uxbox.sql :as sql] + [uxbox.db :as db] + [uxbox.util.spec :as us] + [uxbox.services.core :as core] + [uxbox.util.time :as dt] + [uxbox.util.data :as data] + [uxbox.util.transit :as t] + [uxbox.util.blob :as blob] + [uxbox.util.uuid :as uuid])) + +(s/def ::version integer?) +(s/def ::key string?) +(s/def ::value any?) +(s/def ::user uuid?) + +(defn decode-value + [{:keys [value] :as data}] + (if value + (assoc data :value (-> value blob/decode t/decode)) + data)) + +;; --- Update KVStore + +(s/def ::update-kvstore + (s/keys :req-un [::key ::value ::user ::version])) + +(defn update-kvstore + [conn {:keys [user key value version] :as data}] + (let [opts {:user user + :key key + :version version + :value (-> value t/encode blob/encode)} + sqlv (sql/update-kvstore opts)] + (some->> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (decode-value)))) + +(defmethod core/novelty :update-kvstore + [params] + (s/assert ::update-kvstore params) + (with-open [conn (db/connection)] + (sc/apply-atomic conn update-kvstore params))) + +;; --- Retrieve KVStore + +(s/def ::retrieve-kvstore + (s/keys :req-un [::key ::user])) + +(defn retrieve-kvstore + [conn {:keys [user key] :as params}] + (let [sqlv (sql/retrieve-kvstore params)] + (some->> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (decode-value)))) + +(defmethod core/query :retrieve-kvstore + [params] + (s/assert ::retrieve-kvstore params) + (with-open [conn (db/connection)] + (retrieve-kvstore conn params))) + +;; --- Delete KVStore + +(s/def ::delete-kvstore + (s/keys :req-un [::key ::user])) + +(defn delete-kvstore + [conn {:keys [user key] :as params}] + (let [sqlv (sql/delete-kvstore params)] + (pos? (sc/execute conn sqlv)))) + +(defmethod core/novelty :delete-kvstore + [params] + (s/assert ::delete-kvstore params) + (with-open [conn (db/connection)] + (sc/apply-atomic conn delete-kvstore params))) diff --git a/backend/src/uxbox/services/pages.clj b/backend/src/uxbox/services/pages.clj new file mode 100644 index 000000000..4dc6d2eb2 --- /dev/null +++ b/backend/src/uxbox/services/pages.clj @@ -0,0 +1,226 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services.pages + (:require [clojure.spec :as s] + [suricatta.core :as sc] + [buddy.core.codecs :as codecs] + [uxbox.config :as ucfg] + [uxbox.sql :as sql] + [uxbox.db :as db] + [uxbox.util.spec :as us] + [uxbox.services.core :as core] + [uxbox.services.auth :as usauth] + [uxbox.util.time :as dt] + [uxbox.util.data :as data] + [uxbox.util.transit :as t] + [uxbox.util.blob :as blob] + [uxbox.util.uuid :as uuid])) + +(declare decode-page-data) +(declare decode-page-metadata) +(declare encode-data) + +(s/def ::data any?) +(s/def ::user uuid?) +(s/def ::project uuid?) +(s/def ::metadata any?) +(s/def ::max integer?) +(s/def ::pinned boolean?) +(s/def ::since integer?) + +;; --- Create Page + +(defn create-page + [conn {:keys [id user project name data metadata] :as params}] + (let [opts {:id (or id (uuid/random)) + :user user + :project project + :name name + :data (-> data t/encode blob/encode) + :metadata (-> metadata t/encode blob/encode)} + sqlv (sql/create-page opts)] + (->> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (decode-page-data) + (decode-page-metadata)))) + +(s/def ::create-page + (s/keys :req-un [::data ::user ::project ::us/name ::metadata] + :opt-un [::us/id])) + +(defmethod core/novelty :create-page + [params] + (s/assert ::create-page params) + (with-open [conn (db/connection)] + (create-page conn params))) + +;; --- Update Page + +(defn update-page + [conn {:keys [id user project name + data version metadata] :as params}] + (let [opts {:id (or id (uuid/random)) + :user user + :project project + :name name + :version version + :data (-> data t/encode blob/encode) + :metadata (-> metadata t/encode blob/encode)} + sqlv (sql/update-page opts)] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (decode-page-data) + (decode-page-metadata)))) + +(s/def ::update-page + (s/merge ::create-page (s/keys :req-un [::us/version]))) + +(defmethod core/novelty :update-page + [params] + (s/assert ::update-page params) + (with-open [conn (db/connection)] + (update-page conn params))) + +;; --- Update Page Metadata + +(defn update-page-metadata + [conn {:keys [id user project name + version metadata] :as params}] + (let [opts {:id (or id (uuid/random)) + :user user + :project project + :name name + :version version + :metadata (-> metadata t/encode blob/encode)} + sqlv (sql/update-page-metadata opts)] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (decode-page-data) + (decode-page-metadata)))) + +(s/def ::update-page-metadata + (s/keys :req-un [::user ::project ::us/name ::us/version ::metadata] + :opt-un [::us/id ::data])) + +(defmethod core/novelty :update-page-metadata + [params] + (s/assert ::update-page-metadata params) + (with-open [conn (db/connection)] + (update-page-metadata conn params))) + +;; --- Delete Page + +(defn delete-page + [conn {:keys [id user] :as params}] + (let [sqlv (sql/delete-page {:id id :user user})] + (pos? (sc/execute conn sqlv)))) + +(s/def ::delete-page + (s/keys :req-un [::user ::us/id])) + +(defmethod core/novelty :delete-page + [params] + (s/assert ::delete-page params) + (with-open [conn (db/connection)] + (delete-page conn params))) + +;; --- List Pages by Project + +(defn get-pages-for-project + [conn project] + (let [sqlv (sql/get-pages-for-project {:project project})] + (->> (sc/fetch conn sqlv) + (map data/normalize-attrs) + (map decode-page-data) + (map decode-page-metadata)))) + +(defn get-pages-for-user-and-project + [conn {:keys [user project]}] + (let [sqlv (sql/get-pages-for-user-and-project + {:user user :project project})] + (->> (sc/fetch conn sqlv) + (map data/normalize-attrs) + (map decode-page-data) + (map decode-page-metadata)))) + +(s/def ::list-pages-by-project + (s/keys :req-un [::user ::project])) + +(defmethod core/query :list-pages-by-project + [params] + (s/assert ::list-pages-by-project params) + (with-open [conn (db/connection)] + (get-pages-for-user-and-project conn params))) + +;; --- Page History (Query) + +(defn get-page-history + [conn {:keys [id user since max pinned] + :or {since Long/MAX_VALUE max 10}}] + (let [sqlv (sql/get-page-history {:user user + :page id + :since since + :max max + :pinned pinned})] + (->> (sc/fetch conn sqlv) + (map data/normalize-attrs) + (map decode-page-data)))) + +(s/def ::list-page-history + (s/keys :req-un [::us/id ::user] + :opt-un [::max ::pinned ::since])) + +(defmethod core/query :list-page-history + [params] + (s/assert ::list-page-history params) + (with-open [conn (db/connection)] + (get-page-history conn params))) + +;; --- Update Page History + +(defn update-page-history + [conn {:keys [user id label pinned]}] + (let [sqlv (sql/update-page-history {:user user + :id id + :label label + :pinned pinned})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (decode-page-data)))) + +(s/def ::label string?) +(s/def ::update-page-history + (s/keys :req-un [::user ::us/id ::pinned ::label])) + +(defmethod core/novelty :update-page-history + [params] + (s/assert ::update-page-history params) + (with-open [conn (db/connection)] + (update-page-history conn params))) + +;; --- Helpers + +(defn- decode-page-metadata + [{:keys [metadata] :as result}] + (s/assert ::us/bytes metadata) + (merge result (when metadata + {:metadata (-> metadata blob/decode t/decode)}))) + +(defn- decode-page-data + [{:keys [data] :as result}] + (s/assert ::us/bytes data) + (merge result (when data + {:data (-> data blob/decode t/decode)}))) + +(defn get-page-by-id + [conn id] + (s/assert ::us/id id) + (let [sqlv (sql/get-page-by-id {:id id})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (decode-page-data) + (decode-page-metadata)))) diff --git a/backend/src/uxbox/services/projects.clj b/backend/src/uxbox/services/projects.clj new file mode 100644 index 000000000..e958917e0 --- /dev/null +++ b/backend/src/uxbox/services/projects.clj @@ -0,0 +1,143 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services.projects + (:require [clojure.spec :as s] + [suricatta.core :as sc] + [buddy.core.codecs :as codecs] + [uxbox.config :as ucfg] + [uxbox.sql :as sql] + [uxbox.db :as db] + [uxbox.util.spec :as us] + [uxbox.services.core :as core] + [uxbox.services.pages :as pages] + [uxbox.util.data :as data] + [uxbox.util.transit :as t] + [uxbox.util.blob :as blob] + [uxbox.util.uuid :as uuid])) + +(s/def ::token string?) +(s/def ::data string?) +(s/def ::user uuid?) +(s/def ::project uuid?) + +;; --- Create Project + +(defn create-project + [conn {:keys [id user name] :as data}] + (let [id (or id (uuid/random)) + sqlv (sql/create-project {:id id :user user :name name})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize)))) + +(s/def ::create-project + (s/keys :req-un [::user ::us/name] + :opt-un [::us/id])) + +(defmethod core/novelty :create-project + [params] + (s/assert ::create-project params) + (with-open [conn (db/connection)] + (create-project conn params))) + +;; --- Update Project + +(defn- update-project + [conn {:keys [name version id user] :as data}] + (let [sqlv (sql/update-project {:name name + :version version + :id id + :user user})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize)))) + +(s/def ::update-project + (s/merge ::create-project (s/keys :req-un [::us/version]))) + +(defmethod core/novelty :update-project + [params] + (s/assert ::update-project params) + (with-open [conn (db/connection)] + (update-project conn params))) + +;; --- Delete Project + +(defn- delete-project + [conn {:keys [id user] :as data}] + (let [sqlv (sql/delete-project {:id id :user user})] + (pos? (sc/execute conn sqlv)))) + +(s/def ::delete-project + (s/keys :req-un [::us/id ::user])) + +(defmethod core/novelty :delete-project + [params] + (s/assert ::delete-project params) + (with-open [conn (db/connection)] + (delete-project conn params))) + +;; --- List Projects + +(declare decode-page-metadata) +(declare decode-page-data) + +(defn get-projects + [conn user] + (let [sqlv (sql/get-projects {:user user})] + (->> (sc/fetch conn sqlv) + (map data/normalize) + + ;; This is because the project comes with + ;; the first page preloaded and it need + ;; to be decoded. + (map decode-page-metadata) + (map decode-page-data)))) + +(defmethod core/query :list-projects + [{:keys [user] :as params}] + (s/assert ::user user) + (with-open [conn (db/connection)] + (get-projects conn user))) + +;; --- Retrieve Project by share token + +(defn- get-project-by-share-token + [conn token] + (let [sqlv (sql/get-project-by-share-token {:token token}) + project (some-> (sc/fetch-one conn sqlv) + (data/normalize))] + (when-let [id (:id project)] + (let [pages (vec (pages/get-pages-for-project conn id))] + (assoc project :pages pages))))) + +(defmethod core/query :retrieve-project-by-share-token + [{:keys [token]}] + (s/assert ::token token) + (with-open [conn (db/connection)] + (get-project-by-share-token conn token))) + +;; --- Retrieve share tokens + +(defn get-share-tokens-for-project + [conn project] + (s/assert ::project project) + (let [sqlv (sql/get-share-tokens-for-project {:project project})] + (->> (sc/fetch conn sqlv) + (map data/normalize)))) + +;; Helpers + +(defn- decode-page-metadata + [{:keys [page-metadata] :as result}] + (merge result (when page-metadata + {:page-metadata (-> page-metadata blob/decode t/decode)}))) + +(defn- decode-page-data + [{:keys [page-data] :as result}] + (merge result (when page-data + {:page-data (-> page-data blob/decode t/decode)}))) + + diff --git a/backend/src/uxbox/services/svgparse.clj b/backend/src/uxbox/services/svgparse.clj new file mode 100644 index 000000000..a41b6a2cf --- /dev/null +++ b/backend/src/uxbox/services/svgparse.clj @@ -0,0 +1,95 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services.svgparse + (:require [clojure.spec :as s] + [cuerdas.core :as str] + [uxbox.util.spec :as us] + [uxbox.services.core :as core] + [uxbox.util.exceptions :as ex]) + (:import org.jsoup.Jsoup + java.io.InputStream)) + +(s/def ::content string?) +(s/def ::width number?) +(s/def ::height number?) +(s/def ::name string?) +(s/def ::view-box (s/coll-of number? :min-count 4 :max-count 4)) +(s/def ::svg-entity (s/keys :req-un [::content ::width ::height ::view-box] + :opt-un [::name])) + +;; --- Implementation + +(defn- parse-double + [data] + {:pre [(string? data)]} + (Double/parseDouble data)) + +(defn- parse-viewbox + [data] + {:pre [(string? data)]} + (mapv parse-double (str/split data #"\s+"))) + +(defn- assoc-attr + [acc attr] + (let [key (.getKey attr) + val (.getValue attr)] + (case key + "width" (assoc acc :width (parse-double val)) + "height" (assoc acc :height (parse-double val)) + "viewbox" (assoc acc :view-box (parse-viewbox val)) + "sodipodi:docname" (assoc acc :name val) + acc))) + +(defn- parse-attrs + [element] + (let [attrs (.attributes element)] + (reduce assoc-attr {} attrs))) + +(defn- parse-svg + [data] + (try + (let [document (Jsoup/parse data) + svgelement (some-> (.body document) + (.getElementsByTag "svg") + (first)) + innerxml (.html svgelement) + attrs (parse-attrs svgelement)] + (merge {:content innerxml} attrs)) + (catch java.lang.IllegalArgumentException e + (ex/raise :type :validation + :code ::invalid-input + :message "Input does not seems to be a valid svg.")) + (catch java.lang.NullPointerException e + (ex/raise :type :validation + :code ::invalid-input + :message "Input does not seems to be a valid svg.")) + (catch Exception e + (.printStackTrace e) + (ex/raise :code ::unexpected)))) + +;; --- Public Api + +(defn parse-string + "Parse SVG from a string." + [data] + {:pre [(string? data)]} + (let [result (parse-svg data)] + (if (s/valid? ::svg-entity result) + result + (ex/raise :type :validation + :code ::invalid-result + :message "The result does not conform valid svg entity.")))) + +(defn parse + [data] + {:pre [(instance? InputStream data)]} + (parse-string (slurp data))) + +(defmethod core/query :parse-svg + [{:keys [data] :as params}] + {:pre [(string? data)]} + (parse-string data)) diff --git a/backend/src/uxbox/services/users.clj b/backend/src/uxbox/services/users.clj new file mode 100644 index 000000000..5b4e8d810 --- /dev/null +++ b/backend/src/uxbox/services/users.clj @@ -0,0 +1,317 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.services.users + (:require [clojure.spec :as s] + [mount.core :as mount :refer (defstate)] + [suricatta.core :as sc] + [buddy.hashers :as hashers] + [buddy.sign.jwe :as jwe] + [uxbox.sql :as sql] + [uxbox.db :as db] + [uxbox.util.spec :as us] + [uxbox.emails :as emails] + [uxbox.services.core :as core] + [uxbox.util.transit :as t] + [uxbox.util.exceptions :as ex] + [uxbox.util.data :as data] + [uxbox.util.blob :as blob] + [uxbox.util.uuid :as uuid] + [uxbox.util.token :as token])) + +(declare decode-user-data) +(declare trim-user-attrs) +(declare find-user-by-id) +(declare find-full-user-by-id) +(declare find-user-by-username-or-email) + +(s/def ::user uuid?) +(s/def ::fullname string?) +(s/def ::metadata any?) +(s/def ::old-password string?) +(s/def ::path string?) + +;; --- Retrieve User Profile (own) + +(defmethod core/query :retrieve-profile + [{:keys [user] :as params}] + (s/assert ::user user) + (with-open [conn (db/connection)] + (some-> (find-user-by-id conn (:user params)) + (decode-user-data)))) + +;; --- Update User Profile (own) + +(defn- check-profile-existence! + [conn {:keys [id username email]}] + (let [sqlv1 (sql/user-with-email-exists? {:id id :email email}) + sqlv2 (sql/user-with-username-exists? {:id id :username username})] + (when (:val (sc/fetch-one conn sqlv1)) + (ex/raise :type :validation + :code ::email-already-exists)) + (when (:val (sc/fetch-one conn sqlv2)) + (ex/raise ::username-already-exists)))) + +(defn- update-profile + [conn {:keys [id username email fullname metadata] :as params}] + (check-profile-existence! conn params) + (let [metadata (-> metadata t/encode blob/encode) + sqlv (sql/update-profile {:username username + :fullname fullname + :metadata metadata + :email email + :id id})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (trim-user-attrs) + (decode-user-data) + (dissoc :password)))) + +(s/def ::update-profile + (s/keys :req-un [::us/id ::us/username ::us/email ::fullname ::metadata])) + +(defmethod core/novelty :update-profile + [params] + (s/assert ::update-profile params) + (with-open [conn (db/connection)] + (sc/apply-atomic conn update-profile params))) + +;; --- Update Password + +(defn update-password + [conn {:keys [user password]}] + (let [password (hashers/encrypt password) + sqlv (sql/update-profile-password {:id user :password password})] + (pos? (sc/execute conn sqlv)))) + +(defn- validate-old-password + [conn {:keys [user old-password] :as params}] + (let [user (find-full-user-by-id conn user)] + (when-not (hashers/check old-password (:password user)) + (ex/raise :type :validation + :code ::old-password-not-match)) + params)) + +(s/def ::update-password + (s/keys :req-un [::user ::us/password ::old-password])) + +(defmethod core/novelty :update-profile-password + [params] + (s/assert ::update-password params) + (with-open [conn (db/connection)] + (->> params + (validate-old-password conn) + (update-password conn)))) + +;; --- Update Photo + +(defn update-photo + [conn {:keys [user path]}] + (let [sqlv (sql/update-profile-photo {:id user :photo path})] + (pos? (sc/execute conn sqlv)))) + +(s/def ::update-photo + (s/keys :req-un [::user ::path])) + +(defmethod core/novelty :update-profile-photo + [params] + (s/assert ::update-photo params) + (with-open [conn (db/connection)] + (update-photo conn params))) + +;; --- Create User + +(s/def ::create-user + (s/keys :req-un [::metadata ::fullname ::us/email ::us/password] + :opt-un [::us/id])) + +(defn create-user + [conn {:keys [id username password email fullname metadata] :as data}] + (s/assert ::create-user data) + (let [id (or id (uuid/random)) + metadata (-> metadata t/encode blob/encode) + password (hashers/encrypt password) + sqlv (sql/create-profile {:id id + :fullname fullname + :username username + :email email + :password password + :metadata metadata})] + (->> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (trim-user-attrs) + (decode-user-data)))) + +;; --- Register User + +(defn- check-user-registred! + "Check if the user identified by username or by email + is already registred in the platform." + [conn {:keys [username email]}] + (let [sqlv1 (sql/user-with-email-exists? {:email email}) + sqlv2 (sql/user-with-username-exists? {:username username})] + (when (:val (sc/fetch-one conn sqlv1)) + (ex/raise :type :validation + :code ::email-already-exists)) + (when (:val (sc/fetch-one conn sqlv2)) + (ex/raise :type :validation + :code ::username-already-exists)))) + +(defn- register-user + "Create the user entry onthe database with limited input + filling all the other fields with defaults." + [conn {:keys [username fullname email password] :as params}] + (check-user-registred! conn params) + (let [metadata (-> nil t/encode blob/encode) + password (hashers/encrypt password) + sqlv (sql/create-profile {:id (uuid/random) + :fullname fullname + :username username + :email email + :password password + :metadata metadata})] + (sc/execute conn sqlv) + (emails/send! {:email/name :users/register + :email/to (:email params) + :email/priority :high + :name (:fullname params)}) + nil)) + +(s/def ::register + (s/keys :req-un [::us/username ::us/email ::us/password ::fullname])) + +(defmethod core/novelty :register-profile + [params] + (s/assert ::register params) + (with-open [conn (db/connection)] + (sc/apply-atomic conn register-user params))) + +;; --- Password Recover + +(defn- recovery-token-exists? + "Checks if the token exists in the system. Just + return `true` or `false`." + [conn token] + (let [sqlv (sql/recovery-token-exists? {:token token}) + result (sc/fetch-one conn sqlv)] + (:token_exists result))) + +(defn- retrieve-user-for-recovery-token + "Retrieve a user id (uuid) for the given token. If + no user is found, an exception is raised." + [conn token] + (let [sqlv (sql/get-recovery-token {:token token}) + data (sc/fetch-one conn sqlv)] + (or (:user data) + (ex/raise :type :validation + :code ::invalid-token)))) + +(defn- mark-token-as-used + [conn token] + (let [sqlv (sql/mark-recovery-token-used {:token token})] + (pos? (sc/execute conn sqlv)))) + +(defn- recover-password + "Given a token and password, resets the password + to corresponding user or raise an exception." + [conn {:keys [token password]}] + (let [user (retrieve-user-for-recovery-token conn token)] + (update-password conn {:user user :password password}) + (mark-token-as-used conn token) + nil)) + +(defn- create-recovery-token + "Creates a new recovery token for specified user and return it." + [conn userid] + (let [token (token/random) + sqlv (sql/create-recovery-token {:user userid + :token token})] + (sc/execute conn sqlv) + token)) + +(defn- retrieve-user-for-password-recovery + [conn username] + (let [user (find-user-by-username-or-email conn username)] + (when-not user + (ex/raise :type :validation :code ::user-does-not-exists)) + user)) + +(defn- request-password-recovery + "Creates a new recovery password token and sends it via email + to the correspondig to the given username or email address." + [conn username] + (let [user (retrieve-user-for-password-recovery conn username) + token (create-recovery-token conn (:id user))] + (emails/send! {:email/name :users/password-recovery + :email/to (:email user) + :name (:fullname user) + :token token}) + token)) + +(defmethod core/query :validate-profile-password-recovery-token + [{:keys [token]}] + (s/assert ::us/token token) + (with-open [conn (db/connection)] + (recovery-token-exists? conn token))) + +(defmethod core/novelty :request-profile-password-recovery + [{:keys [username]}] + (s/assert ::us/username username) + (with-open [conn (db/connection)] + (sc/atomic conn + (request-password-recovery conn username)))) + +(s/def ::recover-password + (s/keys :req-un [::us/token ::us/password])) + +(defmethod core/novelty :recover-profile-password + [params] + (s/assert ::recover-password params) + (with-open [conn (db/connection)] + (sc/apply-atomic conn recover-password params))) + +;; --- Query Helpers + +(defn find-full-user-by-id + "Find user by its id. This function is for internal + use only because it returns a lot of sensitive information. + If no user is found, `nil` is returned." + [conn id] + (let [sqlv (sql/get-profile {:id id})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs)))) + +(defn find-user-by-id + "Find user by its id. If no user is found, `nil` is returned." + [conn id] + (let [sqlv (sql/get-profile {:id id})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (trim-user-attrs) + (dissoc :password)))) + +(defn find-user-by-username-or-email + "Finds a user in the database by username and email. If no + user is found, `nil` is returned." + [conn username] + (let [sqlv (sql/get-profile-by-username {:username username})] + (some-> (sc/fetch-one conn sqlv) + (data/normalize-attrs) + (trim-user-attrs)))) + +;; --- Attrs Helpers + +(defn- decode-user-data + [{:keys [metadata] :as result}] + (merge result (when metadata + {:metadata (-> metadata blob/decode t/decode)}))) + +(defn trim-user-attrs + "Only selects a publicy visible user attrs." + [user] + (select-keys user [:id :username :fullname + :password :metadata :email + :created-at :photo])) diff --git a/backend/src/uxbox/sql.clj b/backend/src/uxbox/sql.clj new file mode 100644 index 000000000..a6e87be93 --- /dev/null +++ b/backend/src/uxbox/sql.clj @@ -0,0 +1,18 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.sql + (:require [hugsql.core :as hugsql])) + +(hugsql/def-sqlvec-fns "sql/projects.sql" {:quoting :ansi :fn-suffix ""}) +(hugsql/def-sqlvec-fns "sql/pages.sql" {:quoting :ansi :fn-suffix ""}) +(hugsql/def-sqlvec-fns "sql/users.sql" {:quoting :ansi :fn-suffix ""}) +(hugsql/def-sqlvec-fns "sql/emails.sql" {:quoting :ansi :fn-suffix ""}) +(hugsql/def-sqlvec-fns "sql/images.sql" {:quoting :ansi :fn-suffix ""}) +(hugsql/def-sqlvec-fns "sql/icons.sql" {:quoting :ansi :fn-suffix ""}) +(hugsql/def-sqlvec-fns "sql/kvstore.sql" {:quoting :ansi :fn-suffix ""}) +(hugsql/def-sqlvec-fns "sql/workers.sql" {:quoting :ansi :fn-suffix ""}) + diff --git a/backend/src/uxbox/util/blob.clj b/backend/src/uxbox/util/blob.clj new file mode 100644 index 000000000..e1b96eff8 --- /dev/null +++ b/backend/src/uxbox/util/blob.clj @@ -0,0 +1,21 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.blob + "A generic blob storage encoding. Mainly used for + page data, page options and txlog payload storage." + (:require [uxbox.util.snappy :as snappy])) + +(defn encode + "Encode data into compressed blob." + [data] + (snappy/compress data)) + +(defn decode + "Decode blob into string." + [^bytes data] + (snappy/uncompress data)) + diff --git a/backend/src/uxbox/util/cli.clj b/backend/src/uxbox/util/cli.clj new file mode 100644 index 000000000..43289e51f --- /dev/null +++ b/backend/src/uxbox/util/cli.clj @@ -0,0 +1,12 @@ +(ns uxbox.util.cli + "Command line interface helpers.") + +(defn exit! + ([] (exit! 0)) + ([code] + (System/exit code))) + +(defmacro print-err! + [& args] + `(binding [*out* *err*] + (println ~@args))) diff --git a/backend/src/uxbox/util/closeable.clj b/backend/src/uxbox/util/closeable.clj new file mode 100644 index 000000000..58d52a3c5 --- /dev/null +++ b/backend/src/uxbox/util/closeable.clj @@ -0,0 +1,31 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.closeable + "A closeable abstraction. A drop in replacement for + clojure builtin `with-open` syntax abstraction." + (:refer-clojure :exclude [with-open])) + +(defprotocol ICloseable + (-close [_] "Close the resource.")) + +(defmacro with-open + [bindings & body] + {:pre [(vector? bindings) + (even? (count bindings)) + (pos? (count bindings))]} + (reduce (fn [acc bindings] + `(let ~(vec bindings) + (try + ~acc + (finally + (-close ~(first bindings)))))) + `(do ~@body) + (reverse (partition 2 bindings)))) + +(extend-protocol ICloseable + java.lang.AutoCloseable + (-close [this] (.close this))) diff --git a/backend/src/uxbox/util/data.clj b/backend/src/uxbox/util/data.clj new file mode 100644 index 000000000..88cf1785d --- /dev/null +++ b/backend/src/uxbox/util/data.clj @@ -0,0 +1,52 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.data + "Data transformations utils." + (:require [clojure.walk :as walk] + [cuerdas.core :as str])) + +(defn dissoc-in + [m [k & ks :as keys]] + (if ks + (if-let [nextmap (get m k)] + (let [newmap (dissoc-in nextmap ks)] + (if (seq newmap) + (assoc m k newmap) + (dissoc m k))) + m) + (dissoc m k))) + +(defn normalize-attrs + "Recursively transforms all map keys from strings to keywords." + [m] + (letfn [(tf [[k v]] + (let [ks (-> (name k) + (str/replace "_" "-"))] + [(keyword ks) v])) + (walker [x] + (if (map? x) + (into {} (map tf) x) + x))] + (walk/postwalk walker m))) + +(defn strip-delete-attrs + [m] + (dissoc m :deleted-at)) + +(defn normalize + "Perform a common normalization transformation + for a entity (database retrieved) data structure." + [m] + (-> m normalize-attrs strip-delete-attrs)) + +(defn deep-merge + [& maps] + (letfn [(merge' [& maps] + (if (every? map? maps) + (apply merge-with merge' maps) + (last maps)))] + (apply merge' (remove nil? maps)))) diff --git a/backend/src/uxbox/util/exceptions.clj b/backend/src/uxbox/util/exceptions.clj new file mode 100644 index 000000000..6d22b9b36 --- /dev/null +++ b/backend/src/uxbox/util/exceptions.clj @@ -0,0 +1,21 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.exceptions + "A helpers for work with exceptions.") + +(defn error + [& {:keys [type code message] :or {type :unexpected} :as payload}] + {:pre [(keyword? type) (keyword? code)]} + (let [message (if message + (str message " / " (pr-str code) "") + (pr-str code)) + payload (assoc payload :type type)] + (ex-info message payload))) + +(defmacro raise + [& args] + `(throw (error ~@args))) diff --git a/backend/src/uxbox/util/images.clj b/backend/src/uxbox/util/images.clj new file mode 100644 index 000000000..dffc0a9be --- /dev/null +++ b/backend/src/uxbox/util/images.clj @@ -0,0 +1,44 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.images + "Images transformation utils." + (:require [clojure.java.io :as io]) + (:import org.im4java.core.IMOperation + org.im4java.core.ConvertCmd + org.im4java.process.Pipe + java.io.ByteArrayInputStream + java.io.ByteArrayOutputStream)) + +;; Related info on how thumbnails generation +;; http://www.imagemagick.org/Usage/thumbnails/ + +(defn thumbnail + ([input] (thumbnail input nil)) + ([input {:keys [size quality format] + :or {format "jpg" + quality 92 + size [200 200]} + :as opts}] + {:pre [(vector? size)]} + (with-open [out (ByteArrayOutputStream.) + in (io/input-stream input)] + (let [[width height] size + pipe (Pipe. in out) + op (doto (IMOperation.) + (.addRawArgs ^java.util.List ["-"]) + (.autoOrient) + ;; (.thumbnail (int width) (int height) "^") + ;; (.gravity "center") + ;; (.extent (int width) (int height)) + (.resize (int width) (int height) "^") + (.quality (double quality)) + (.addRawArgs ^java.util.List [(str format ":-")])) + cmd (doto (ConvertCmd.) + (.setInputProvider pipe) + (.setOutputConsumer pipe))] + (.run cmd op (make-array Object 0)) + (ByteArrayInputStream. (.toByteArray out)))))) diff --git a/backend/src/uxbox/util/quartz.clj b/backend/src/uxbox/util/quartz.clj new file mode 100644 index 000000000..e74cca37a --- /dev/null +++ b/backend/src/uxbox/util/quartz.clj @@ -0,0 +1,139 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.quartz + "A lightweight abstraction layer for quartz job scheduling library." + (:import java.util.Properties + org.quartz.Scheduler + org.quartz.SchedulerException + org.quartz.impl.StdSchedulerFactory + org.quartz.Job + org.quartz.JobBuilder + org.quartz.JobDataMap + org.quartz.JobExecutionContext + org.quartz.TriggerBuilder + org.quartz.CronScheduleBuilder + org.quartz.SimpleScheduleBuilder + org.quartz.PersistJobDataAfterExecution + org.quartz.DisallowConcurrentExecution)) + +;; --- Implementation + +(defn- map->props + [data] + (let [p (Properties.)] + (run! (fn [[k v]] (.setProperty p (name k) (str v))) (seq data)) + p)) + +(deftype JobImpl [] + Job + (execute [_ context] + (let [^JobDataMap data (.. context getJobDetail getJobDataMap) + args (.get data "arguments") + state (.get data "state") + callable (.get data "callable")] + (if state + (apply callable state args) + (apply callable args))))) + +(defn- resolve-var + [sym] + (let [ns (symbol (namespace sym)) + func (symbol (name sym))] + (require ns) + (resolve func))) + +(defn- build-trigger + [opts] + (let [repeat? (::repeat? opts true) + interval (::interval opts 1000) + cron (::cron opts) + group (::group opts "uxbox") + schdl (if cron + (CronScheduleBuilder/cronSchedule cron) + (let [schdl (SimpleScheduleBuilder/simpleSchedule) + schdl (if (number? repeat?) + (.withRepeatCount schdl repeat?) + (.repeatForever schdl))] + (.withIntervalInMilliseconds schdl interval))) + name (str (:name opts) "-trigger") + bldr (doto (TriggerBuilder/newTrigger) + (.startNow) + (.withIdentity name group) + (.withSchedule schdl))] + (.build bldr))) + +(defn- build-job-detail + [fvar args] + (let [opts (meta fvar) + state (::state opts) + group (::group opts "uxbox") + name (str (:name opts)) + data {"callable" @fvar + "arguments" (into [] args) + "state" (if state (atom state) nil)} + bldr (doto (JobBuilder/newJob JobImpl) + (.storeDurably false) + (.usingJobData (JobDataMap. data)) + (.withIdentity name group))] + (.build bldr))) + +(defn- make-scheduler-props + [{:keys [name daemon? threads thread-priority] + :or {name "uxbox-scheduler" + daemon? true + threads 1 + thread-priority Thread/MIN_PRIORITY}}] + (map->props + {"org.quartz.threadPool.threadCount" threads + "org.quartz.threadPool.threadPriority" thread-priority + "org.quartz.threadPool.makeThreadsDaemons" (if daemon? "true" "false") + "org.quartz.scheduler.instanceName" name + "org.quartz.scheduler.makeSchedulerThreadDaemon" (if daemon? "true" "false")})) + +;; --- Public Api + +(defn scheduler + "Create a new scheduler instance." + ([] (scheduler nil)) + ([opts] + (let [props (make-scheduler-props opts) + factory (StdSchedulerFactory. props)] + (.getScheduler factory)))) + +(declare schedule!) + +(defn start! + ([schd] + (start! schd nil)) + ([schd {:keys [delay search-on]}] + ;; Start the scheduler + (if (number? delay) + (.startDelayed schd (int delay)) + (.start schd)) + + (when (coll? search-on) + (run! (fn [ns] + (require ns) + (doseq [v (vals (ns-publics ns))] + (when (::job (meta v)) + (schedule! schd v)))) + search-on)) + schd)) + +(defn stop! + [scheduler] + (.shutdown ^Scheduler scheduler true)) + +;; TODO: add proper handling of `:delay` option that should allow +;; execute a task firstly delayed until some milliseconds or at certain time. + +(defn schedule! + [schd f & args] + (let [vf (if (symbol? f) (resolve-var f) f) + job (build-job-detail vf args) + trigger (build-trigger (meta vf))] + (.scheduleJob ^Scheduler schd job trigger))) diff --git a/backend/src/uxbox/util/response.clj b/backend/src/uxbox/util/response.clj new file mode 100644 index 000000000..086b78f70 --- /dev/null +++ b/backend/src/uxbox/util/response.clj @@ -0,0 +1,66 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.response + "A lightweigt reponse type definition. + + At first instance it allows set the appropriate + content-type headers and encode the body using + the builtin transit abstraction. + + In future it will allow easy adapt for the content + negotiation that is coming to catacumba." + (:require [catacumba.impl.handlers :as ch] + [catacumba.impl.context :as ctx] + [buddy.core.hash :as hash] + [buddy.core.codecs :as codecs] + [buddy.core.codecs.base64 :as b64] + [uxbox.util.transit :as t]) + (:import ratpack.handling.Context + ratpack.http.Response + ratpack.http.Request + ratpack.http.Headers + ratpack.http.MutableHeaders)) + +(defn digest + [^bytes data] + (-> (hash/blake2b-256 data) + (b64/encode true) + (codecs/bytes->str))) + +(defn- etag-match? + [^Request request ^String new-tag] + (let [^Headers headers (.getHeaders request)] + (when-let [etag (.get headers "if-none-match")] + (= etag new-tag)))) + +(deftype Rsp [data] + ch/ISend + (-send [_ ctx] + (let [^Response response (ctx/get-response* ctx) + ^Request request (ctx/get-request* ctx) + ^MutableHeaders headers (.getHeaders response) + ^String method (.. request getMethod getName toLowerCase) + data (t/encode data)] + (if (= method "get") + (let [etag (digest data)] + (if (etag-match? request etag) + (do + (.set headers "etag" etag) + (.status response 304) + (.send response)) + (do + (.set headers "content-type" "application/transit+json") + (.set headers "etag" etag) + (ch/-send data ctx)))) + (do + (.set headers "content-type" "application/transit+json") + (ch/-send data ctx)))))) + +(defn rsp + "A shortcut for create a response instance." + [data] + (Rsp. data)) diff --git a/backend/src/uxbox/util/snappy.clj b/backend/src/uxbox/util/snappy.clj new file mode 100644 index 000000000..731e06c96 --- /dev/null +++ b/backend/src/uxbox/util/snappy.clj @@ -0,0 +1,40 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.snappy + "A lightweight abstraction layer for snappy compression library." + (:require [buddy.core.codecs :as codecs]) + (:import org.xerial.snappy.Snappy + org.xerial.snappy.SnappyFramedInputStream + org.xerial.snappy.SnappyFramedOutputStream + + java.io.OutputStream + java.io.InputStream)) + + +(defn compress + "Compress data unsing snappy compression algorithm." + [data] + (-> (codecs/to-bytes data) + (Snappy/compress))) + +(defn uncompress + "Uncompress data using snappy compression algorithm." + [data] + (-> (codecs/to-bytes data) + (Snappy/uncompress))) + +(defn input-stream + "Create a Snappy framed input stream." + [^InputStream istream] + (SnappyFramedInputStream. istream)) + +(defn output-stream + "Create a Snappy framed output stream." + ([ostream] + (output-stream ostream nil)) + ([^OutputStream ostream {:keys [block-size] :or {block-size 65536}}] + (SnappyFramedOutputStream. ostream (int block-size) 1.0))) diff --git a/backend/src/uxbox/util/spec.clj b/backend/src/uxbox/util/spec.clj new file mode 100644 index 000000000..5acecd424 --- /dev/null +++ b/backend/src/uxbox/util/spec.clj @@ -0,0 +1,113 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.spec + (:refer-clojure :exclude [keyword uuid vector boolean map set]) + (:require [clojure.spec :as s] + [cuerdas.core :as str] + [uxbox.util.exceptions :as ex]) + (:import java.time.Instant)) + +;; --- Constants + +(def email-rx + #"^[a-zA-Z0-9_.+-]+@[a-zA-Z0-9-]+\.[a-zA-Z0-9-.]+$") + +(def uuid-rx + #"^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$") + +;; --- Public Api + +(defn conform + [spec data] + (let [result (s/conform spec data)] + (if (= result ::s/invalid) + (ex/raise :type :validation + :code ::invalid + :message (s/explain-str spec data) + :context (s/explain-data spec data)) + result))) + +;; --- Predicates + +(defn email? + [v] + (and string? + (re-matches email-rx v))) + +(defn instant? + [v] + (instance? Instant v)) + +(defn path? + [v] + (instance? java.nio.file.Path v)) + +(defn regex? + [v] + (instance? java.util.regex.Pattern v)) + +;; --- Conformers + +(defn- uuid-conformer + [v] + (cond + (uuid? v) v + (string? v) + (cond + (re-matches uuid-rx v) + (java.util.UUID/fromString v) + + (str/empty? v) + nil + + :else + ::s/invalid) + :else ::s/invalid)) + +(defn- integer-conformer + [v] + (cond + (integer? v) v + (string? v) + (if (re-matches #"^[-+]?\d+$" v) + (Long/parseLong v) + ::s/invalid) + :else ::s/invalid)) + +(defn boolean-conformer + [v] + (cond + (boolean? v) v + (string? v) + (if (re-matches #"^(?:t|true|false|f|0|1)$" v) + (contains? #{"t" "true" "1"} v) + ::s/invalid) + :else ::s/invalid)) + +(defn boolean-unformer + [v] + (if v "true" "false")) + +;; --- Default Specs + +(s/def ::integer-string (s/conformer integer-conformer str)) +(s/def ::uuid-string (s/conformer uuid-conformer str)) +(s/def ::boolean-string (s/conformer boolean-conformer boolean-unformer)) +(s/def ::positive-integer #(< 0 % Long/MAX_VALUE)) +(s/def ::uploaded-file #(instance? ratpack.form.UploadedFile %)) +(s/def ::uuid uuid?) +(s/def ::bytes bytes?) +(s/def ::path path?) + +(s/def ::id ::uuid-string) +(s/def ::name string?) +(s/def ::username string?) +(s/def ::password string?) +(s/def ::version integer?) +(s/def ::email email?) +(s/def ::token string?) + diff --git a/backend/src/uxbox/util/tempfile.clj b/backend/src/uxbox/util/tempfile.clj new file mode 100644 index 000000000..273c72212 --- /dev/null +++ b/backend/src/uxbox/util/tempfile.clj @@ -0,0 +1,17 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.tempfile + "A temporal file abstractions." + (:require [storages.core :as st] + [storages.util :as path]) + (:import [java.nio.file Files])) + +(defn create + "Create a temporal file." + [& {:keys [suffix prefix]}] + (->> (path/make-file-attrs "rwxr-xr-x") + (Files/createTempFile prefix suffix))) diff --git a/backend/src/uxbox/util/template.clj b/backend/src/uxbox/util/template.clj new file mode 100644 index 000000000..8579f4163 --- /dev/null +++ b/backend/src/uxbox/util/template.clj @@ -0,0 +1,56 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.template + "A lightweight abstraction over mustache.java template engine. + The documentation can be found: http://mustache.github.io/mustache.5.html" + (:require [clojure.walk :as walk] + [clojure.java.io :as io]) + (:import java.io.StringReader + java.io.StringWriter + java.util.HashMap + com.github.mustachejava.DefaultMustacheFactory + com.github.mustachejava.Mustache)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Impl +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(def ^:private + ^DefaultMustacheFactory + +mustache-factory+ (DefaultMustacheFactory.)) + +(defprotocol ITemplate + "A basic template rendering abstraction." + (-render [template context])) + +(extend-type Mustache + ITemplate + (-render [template context] + (with-out-str + (let [scope (HashMap. (walk/stringify-keys context))] + (.execute template *out* scope))))) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Public Api +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(defn render-string + "Render string as mustache template." + ([^String template] + (render-string template {})) + ([^String template context] + (let [reader (StringReader. template) + template (.compile +mustache-factory+ reader "example")] + (-render template context)))) + +(defn render + "Load a file from the class path and render + it using mustache template." + ([^String path] + (render path {})) + ([^String path context] + (render-string (slurp (io/resource path)) context))) diff --git a/backend/src/uxbox/util/time.clj b/backend/src/uxbox/util/time.clj new file mode 100644 index 000000000..2be5ca7eb --- /dev/null +++ b/backend/src/uxbox/util/time.clj @@ -0,0 +1,76 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.time + (:require [suricatta.proto :as proto] + [cognitect.transit :as t]) + (:import java.time.Instant + java.sql.Timestamp)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Serialization Layer conversions +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(declare from-string) + +(def ^:private write-handler + (t/write-handler + (constantly "m") + (fn [v] (str (.toEpochMilli v))))) + +(def ^:private read-handler + (t/read-handler + (fn [v] (-> (Long/parseLong v) + (Instant/ofEpochMilli))))) + +(def +read-handlers+ + {"m" read-handler}) + +(def +write-handlers+ + {Instant write-handler}) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Persistence Layer Conversions +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(extend-protocol proto/IParamType + Instant + (-render [self ctx] + (if (proto/-inline? ctx) + (str "'" (.toString self) "'::timestamptz") + "?::timestamptz")) + + (-bind [self ctx] + (when-not (proto/-inline? ctx) + (let [stmt (proto/-statement ctx) + idx (proto/-next-bind-index ctx) + obj (Timestamp/from self)] + (.setTimestamp stmt idx obj))))) + +(extend-protocol proto/ISQLType + Timestamp + (-convert [self] + (.toInstant self))) + +(defmethod print-method Instant + [mv ^java.io.Writer writer] + (.write writer (str "#instant \"" (.toString mv) "\""))) + +(defmethod print-dup Instant [o w] + (print-method o w)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Helpers +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(defn from-string + [s] + {:pre [(string? s)]} + (Instant/parse s)) + +(defn now + [] + (Instant/now)) diff --git a/backend/src/uxbox/util/token.clj b/backend/src/uxbox/util/token.clj new file mode 100644 index 000000000..605c9af1f --- /dev/null +++ b/backend/src/uxbox/util/token.clj @@ -0,0 +1,24 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.token + "Facilities for generate random tokens." + (:require [buddy.core.nonce :as nonce] + [buddy.core.hash :as hash] + [buddy.core.codecs :as codecs] + [buddy.core.codecs.base64 :as b64])) + +(defn random + "Returns a 32 bytes randomly generated token + with 1024 random seed. The output is encoded + using urlsafe variant of base64." + [] + (-> (nonce/random-bytes 1024) + (hash/blake2b-256) + (b64/encode true) + (codecs/bytes->str))) + + diff --git a/backend/src/uxbox/util/transit.clj b/backend/src/uxbox/util/transit.clj new file mode 100644 index 000000000..b8b0a0e26 --- /dev/null +++ b/backend/src/uxbox/util/transit.clj @@ -0,0 +1,71 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.transit + (:require [cognitect.transit :as t] + [catacumba.handlers.parse :as cparse] + [uxbox.util.time :as dt]) + (:import ratpack.http.TypedData + ratpack.handling.Context + java.io.ByteArrayInputStream + java.io.ByteArrayOutputStream)) + +;; --- Handlers + +(def ^:private +reader-handlers+ + dt/+read-handlers+) + +(def ^:private +write-handlers+ + dt/+write-handlers+) + +;; --- Low-Level Api + +(defn reader + ([istream] + (reader istream nil)) + ([istream {:keys [type] :or {type :json}}] + (t/reader istream type {:handlers +reader-handlers+}))) + +(defn read! + "Read value from streamed transit reader." + [reader] + (t/read reader)) + +(defn writer + ([ostream] + (writer ostream nil)) + ([ostream {:keys [type] :or {type :json}}] + (t/writer ostream type {:handlers +write-handlers+}))) + +(defn write! + [writer data] + (t/write writer data)) + + +;; --- Catacumba Extension + +(defmethod cparse/parse-body :application/transit+json + [^Context ctx ^TypedData body] + (let [reader (reader (.getInputStream body) {:type :json})] + (read! reader))) + +;; --- High-Level Api + +(defn decode + ([data] + (decode data nil)) + ([data opts] + (with-open [input (ByteArrayInputStream. data)] + (read! (reader input opts))))) + +(defn encode + ([data] + (encode data nil)) + ([data opts] + (with-open [out (ByteArrayOutputStream.)] + (let [w (writer out opts)] + (write! w data) + (.toByteArray out))))) diff --git a/backend/src/uxbox/util/uuid.clj b/backend/src/uxbox/util/uuid.clj new file mode 100644 index 000000000..8fe024ce1 --- /dev/null +++ b/backend/src/uxbox/util/uuid.clj @@ -0,0 +1,25 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.uuid + (:require [clj-uuid :as uuid]) + (:import java.util.UUID)) + +(def ^:const zero uuid/+null+) + +(def random + "Alias for clj-uuid/v4." + uuid/v4) + +(defn namespaced + [ns data] + (uuid/v5 ns data)) + +(defn from-string + "Parse string uuid representation into proper UUID instance." + [s] + (UUID/fromString s)) + diff --git a/backend/src/uxbox/util/workers.clj b/backend/src/uxbox/util/workers.clj new file mode 100644 index 000000000..fac845b84 --- /dev/null +++ b/backend/src/uxbox/util/workers.clj @@ -0,0 +1,74 @@ +;; 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) 2016 Andrey Antukh + +(ns uxbox.util.workers + "A distributed asynchronous tasks queue implementation on top + of PostgreSQL reliable advirsory locking mechanism." + (:require [suricatta.core :as sc] + [uxbox.db :as db] + [uxbox.sql :as sql])) + +(defn- poll-for-task + [conn queue] + (let [sql (sql/acquire-task {:queue queue})] + (sc/fetch-one conn sql))) + +(defn- mark-task-done + [conn {:keys [id]}] + (let [sql (sql/mark-task-done {:id id})] + (sc/execute conn sql))) + +(defn- mark-task-failed + [conn {:keys [id]} error] + (let [sql (sql/mark-task-done {:id id :error (.getMessage error)})] + (sc/execute conn sql))) + +(defn- watch-unit + [conn queue callback] + (let [task (poll-for-task conn queue)] + (if (nil? task) + (Thread/sleep 1000) + (try + (sc/atomic conn + (callback conn task) + (mark-task-done conn task)) + (catch Exception e + (mark-task-failed conn task e)))))) + +(defn- watch-loop + "Watch tasks on the specified queue and executes a + callback for each task is received. + NOTE: This function blocks the current thread." + [queue callback] + (try + (loop [] + (with-open [conn (db/connection)] + (sc/atomic conn (watch-unit conn queue callback))) + (recur)) + (catch InterruptedException e + ;; just ignoring + ))) + +(defn watch! + [queue callback] + (let [runnable #(watch-loop queue callback) + thread (Thread. ^Runnable runnable)] + (.setDaemon thread true) + (.start thread) + (reify + java.lang.AutoCloseable + (close [_] + (.interrupt thread) + (.join thread 2000)) + + clojure.lang.IDeref + (deref [_] + (.join thread)) + + clojure.lang.IBlockingDeref + (deref [_ ms default] + (.join thread ms) + default)))) diff --git a/backend/test/storages/tests.clj b/backend/test/storages/tests.clj new file mode 100644 index 000000000..c29e26eb0 --- /dev/null +++ b/backend/test/storages/tests.clj @@ -0,0 +1,106 @@ +(ns storages.tests + (:require [clojure.test :as t] + [storages.core :as st] + [storages.fs.local :as fs] + [storages.fs.misc :as misc]) + (:import java.io.File + org.apache.commons.io.FileUtils)) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Test Fixtures +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(defn- clean-temp-directory + [next] + (next) + (let [directory (File. "/tmp/catacumba/")] + (FileUtils/deleteDirectory directory))) + +(t/use-fixtures :each clean-temp-directory) + +;; --- Tests: FileSystemStorage + +(t/deftest test-localfs-store-and-lookup + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"}) + rpath @(st/save storage "test.txt" "my content") + fpath @(st/lookup storage rpath) + fdata (slurp fpath)] + (t/is (= (str fpath) "/tmp/catacumba/test/test.txt")) + (t/is (= "my content" fdata)))) + +(t/deftest test-localfs-store-and-get-public-url + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"}) + rpath @(st/save storage "test.txt" "my content") + ruri (st/public-url storage rpath)] + (t/is (= (str ruri) "http://localhost:5050/test.txt")))) + +(t/deftest test-localfs-store-and-lookup-with-subdirs + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"}) + rpath @(st/save storage "somepath/test.txt" "my content") + fpath @(st/lookup storage rpath) + fdata (slurp fpath)] + (t/is (= (str fpath) "/tmp/catacumba/test/somepath/test.txt")) + (t/is (= "my content" fdata)))) + +(t/deftest test-localfs-store-and-delete-and-check + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"}) + rpath @(st/save storage "test.txt" "my content")] + (t/is @(st/delete storage rpath)) + (t/is (not @(st/exists? storage rpath))))) + +(t/deftest test-localfs-store-duplicate-file-raises-exception + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"})] + (t/is @(st/save storage "test.txt" "my content")) + (t/is (thrown? java.util.concurrent.ExecutionException + @(st/save storage "test.txt" "my content"))))) + +(t/deftest test-localfs-access-unauthorized-path + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"})] + (t/is (thrown? java.util.concurrent.ExecutionException + @(st/lookup storage "../test.txt"))) + (t/is (thrown? java.util.concurrent.ExecutionException + @(st/lookup storage "/test.txt"))))) + +;; --- Tests: ScopedPathStorage + +(t/deftest test-localfs-scoped-store-and-lookup + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"}) + storage (misc/scoped storage "some/prefix") + rpath @(st/save storage "test.txt" "my content") + fpath @(st/lookup storage rpath) + fdata (slurp fpath)] + (t/is (= (str fpath) "/tmp/catacumba/test/some/prefix/test.txt")) + (t/is (= "my content" fdata)))) + +(t/deftest test-localfs-scoped-store-and-delete-and-check + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"}) + storage (misc/scoped storage "some/prefix") + rpath @(st/save storage "test.txt" "my content")] + (t/is @(st/delete storage rpath)) + (t/is (not @(st/exists? storage rpath))))) + +(t/deftest test-localfs-scoped-store-duplicate-file-raises-exception + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"}) + storage (misc/scoped storage "some/prefix")] + (t/is @(st/save storage "test.txt" "my content")) + (t/is (thrown? java.util.concurrent.ExecutionException + @(st/save storage "test.txt" "my content"))))) + +(t/deftest test-localfs-scoped-access-unauthorized-path + (let [storage (fs/filesystem {:basedir "/tmp/catacumba/test" + :baseuri "http://localhost:5050/"}) + storage (misc/scoped storage "some/prefix")] + (t/is (thrown? java.util.concurrent.ExecutionException + @(st/lookup storage "../test.txt"))) + (t/is (thrown? java.util.concurrent.ExecutionException + @(st/lookup storage "/test.txt"))))) + diff --git a/backend/test/uxbox/tests/_files/sample.jpg b/backend/test/uxbox/tests/_files/sample.jpg new file mode 100644 index 000000000..2339746ec Binary files /dev/null and b/backend/test/uxbox/tests/_files/sample.jpg differ diff --git a/backend/test/uxbox/tests/_files/sample1.svg b/backend/test/uxbox/tests/_files/sample1.svg new file mode 100644 index 000000000..dab1f876a --- /dev/null +++ b/backend/test/uxbox/tests/_files/sample1.svg @@ -0,0 +1,20 @@ + + + + + + + + + + image/svg+xml + + + + + + + + + + diff --git a/backend/test/uxbox/tests/_files/sample2.svg b/backend/test/uxbox/tests/_files/sample2.svg new file mode 100644 index 000000000..4d3a3b5ea --- /dev/null +++ b/backend/test/uxbox/tests/_files/sample2.svg @@ -0,0 +1,54 @@ + + + + + + image/svg+xml + + + + + + + + diff --git a/backend/test/uxbox/tests/helpers.clj b/backend/test/uxbox/tests/helpers.clj new file mode 100644 index 000000000..bc248e7f9 --- /dev/null +++ b/backend/test/uxbox/tests/helpers.clj @@ -0,0 +1,209 @@ +(ns uxbox.tests.helpers + (:refer-clojure :exclude [await]) + (:require [clj-http.client :as http] + [buddy.hashers :as hashers] + [buddy.core.codecs :as codecs] + [catacumba.serializers :as sz] + [mount.core :as mount] + [storages.core :as st] + [suricatta.core :as sc] + [uxbox.services.auth :as usa] + [uxbox.services.users :as usu] + [uxbox.util.transit :as t] + [uxbox.migrations :as umg] + [uxbox.media :as media] + [uxbox.db :as db] + [uxbox.config :as cfg])) + +(def +base-url+ "http://localhost:5050") + +(defn state-init + [next] + (let [config (cfg/read-test-config)] + (-> (mount/only #{#'uxbox.config/config + #'uxbox.config/secret + #'uxbox.db/datasource + #'uxbox.migrations/migrations + #'uxbox.media/static-storage + #'uxbox.media/media-storage + #'uxbox.media/images-storage + #'uxbox.media/thumbnails-storage}) + (mount/swap {#'uxbox.config/config config}) + (mount/start)) + (try + (next) + (finally + (mount/stop))))) + +(defn database-reset + [next] + (state-init + (fn [] + (with-open [conn (db/connection)] + (let [sql (str "SELECT table_name " + " FROM information_schema.tables " + " WHERE table_schema = 'public' " + " AND table_name != 'migrations';") + result (->> (sc/fetch conn sql) + (map :table_name))] + (sc/execute conn (str "TRUNCATE " + (apply str (interpose ", " result)) + " CASCADE;")))) + (try + (next) + (finally + (st/clear! uxbox.media/media-storage) + (st/clear! uxbox.media/static-storage)))))) + +(defmacro await + [expr] + `(try + (deref ~expr) + (catch Exception e# + (.getCause e#)))) + +(defn- strip-response + [{:keys [status headers body]}] + (if (= (get headers "content-type") "application/transit+json") + [status (-> (codecs/str->bytes body) + (t/decode))] + [status body])) + +(defn http-get + ([user uri] (http-get user uri nil)) + ([user uri {:keys [query] :as opts}] + (let [headers (when user + {"Authorization" (str "Token " (usa/generate-token user))}) + params (merge {:headers headers} + (when query + {:query-params query}))] + (try + (strip-response (http/get uri params)) + (catch clojure.lang.ExceptionInfo e + (strip-response (ex-data e))))))) + +(defn http-post + ([uri params] + (http-post nil uri params)) + ([user uri {:keys [body] :as params}] + (let [body (-> (t/encode body) + (codecs/bytes->str)) + headers (merge + {"content-type" "application/transit+json"} + (when user + {"Authorization" (str "Token " (usa/generate-token user))})) + params {:headers headers :body body}] + (try + (strip-response (http/post uri params)) + (catch clojure.lang.ExceptionInfo e + (strip-response (ex-data e))))))) + +(defn http-multipart + [user uri params] + (let [headers (merge + (when user + {"Authorization" (str "Token " (usa/generate-token user))})) + params {:headers headers + :multipart params}] + (try + (strip-response (http/post uri params)) + (catch clojure.lang.ExceptionInfo e + (strip-response (ex-data e)))))) + +(defn http-put + ([uri params] + (http-put nil uri params)) + ([user uri {:keys [body] :as params}] + (let [body (-> (t/encode body) + (codecs/bytes->str)) + headers (merge + {"content-type" "application/transit+json"} + (when user + {"Authorization" (str "Token " (usa/generate-token user))})) + params {:headers headers :body body}] + (try + (strip-response (http/put uri params)) + (catch clojure.lang.ExceptionInfo e + (strip-response (ex-data e))))))) + +(defn http-delete + ([uri] + (http-delete nil uri)) + ([user uri] + (let [headers (when user + {"Authorization" (str "Token " (usa/generate-token user))}) + params {:headers headers}] + (try + (strip-response (http/delete uri params)) + (catch clojure.lang.ExceptionInfo e + (strip-response (ex-data e))))))) + +(defn- decode-response + [{:keys [status headers body] :as response}] + (if (= (get headers "content-type") "application/transit+json") + (assoc response :body (-> (codecs/str->bytes body) + (t/decode))) + response)) + +(defn request + [{:keys [path method body user headers raw?] + :or {raw? false} + :as request}] + {:pre [(string? path) (keyword? method)]} + (let [body (if (and body (not raw?)) + (-> (t/encode body) + (codecs/bytes->str)) + body) + headers (cond-> headers + body (assoc "content-type" "application/transit+json") + raw? (assoc "content-type" "application/octet-stream") + user (assoc "authorization" + (str "Token " (usa/generate-token user)))) + params {:headers headers :body body} + uri (str +base-url+ path)] + (try + (let [response (case method + :get (http/get uri (dissoc params :body)) + :post (http/post uri params) + :put (http/put uri params) + :delete (http/delete uri params))] + (decode-response response)) + (catch clojure.lang.ExceptionInfo e + (decode-response (ex-data e)))))) + +(defn create-user + "Helper for create users" + [conn i] + (let [data {:username (str "user" i) + :password (str "user" i) + :metadata (str i) + :fullname (str "User " i) + :email (str "user" i "@uxbox.io")}] + (usu/create-user conn data))) + +(defmacro try-on + [& body] + `(try + (let [result# (do ~@body)] + [nil result#]) + (catch Throwable e# + [e# nil]))) + +(defn exception? + [v] + (instance? Throwable v)) + +(defn ex-info? + [v] + (instance? clojure.lang.ExceptionInfo v)) + +(defn ex-of-type? + [e type] + (let [data (ex-data e)] + (= type (:type data)))) + +(defn ex-with-code? + [e code] + (let [data (ex-data e)] + (= code (:code data)))) + diff --git a/backend/test/uxbox/tests/test_auth.clj b/backend/test/uxbox/tests/test_auth.clj new file mode 100644 index 000000000..4771dfe7d --- /dev/null +++ b/backend/test/uxbox/tests/test_auth.clj @@ -0,0 +1,53 @@ +(ns uxbox.tests.test-auth + (:require [clojure.test :as t] + [promesa.core :as p] + [clj-http.client :as http] + [catacumba.testing :refer (with-server)] + [buddy.hashers :as hashers] + [uxbox.db :as db] + [uxbox.frontend :as uft] + [uxbox.services.users :as usu] + [uxbox.services :as usv] + [uxbox.tests.helpers :as th])) + +(t/use-fixtures :each th/database-reset) + +(t/deftest test-http-success-auth + (let [data {:username "user1" + :fullname "user 1" + :metadata "1" + :password "user1" + :email "user1@uxbox.io"} + user (with-open [conn (db/connection)] + (usu/create-user conn data))] + (with-server {:handler (uft/routes)} + (let [data {:username "user1" + :password "user1" + :metadata "1" + :scope "foobar"} + uri (str th/+base-url+ "/api/auth/token") + [status data] (th/http-post uri {:body data})] + ;; (println "RESPONSE:" status data) + (t/is (= status 200)) + (t/is (contains? data :token)))))) + +(t/deftest test-http-failed-auth + (let [data {:username "user1" + :fullname "user 1" + :metadata "1" + :password (hashers/encrypt "user1") + :email "user1@uxbox.io"} + user (with-open [conn (db/connection)] + (usu/create-user conn data))] + (with-server {:handler (uft/routes)} + (let [data {:username "user1" + :password "user2" + :metadata "2" + :scope "foobar"} + uri (str th/+base-url+ "/api/auth/token") + [status data] (th/http-post uri {:body data})] + ;; (println "RESPONSE:" status data) + (t/is (= 400 status)) + (t/is (= (:type data) :validation)) + (t/is (= (:code data) :uxbox.services.auth/wrong-credentials)))))) + diff --git a/backend/test/uxbox/tests/test_icons.clj b/backend/test/uxbox/tests/test_icons.clj new file mode 100644 index 000000000..e69d462d5 --- /dev/null +++ b/backend/test/uxbox/tests/test_icons.clj @@ -0,0 +1,163 @@ +(ns uxbox.tests.test-icons + (:require [clojure.test :as t] + [promesa.core :as p] + [suricatta.core :as sc] + [clj-uuid :as uuid] + [clojure.java.io :as io] + [catacumba.testing :refer (with-server)] + [buddy.core.codecs :as codecs] + [uxbox.db :as db] + [uxbox.sql :as sql] + [uxbox.frontend :as uft] + [uxbox.services.icons :as icons] + [uxbox.services :as usv] + [uxbox.tests.helpers :as th])) + +(t/use-fixtures :each th/database-reset) + +(t/deftest test-http-list-icon-collections + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "coll1"} + coll (icons/create-collection conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/icon-collections") + [status data] (th/http-get user uri)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= 1 (count data)))))))) + +(t/deftest test-http-create-icon-collection + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/icon-collections") + data {:user (:id user) + :name "coll1"} + params {:body data} + [status data] (th/http-post user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 201 status)) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "coll1"))))))) + +(t/deftest test-http-update-icon-collection + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "coll1"} + coll (icons/create-collection conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/icon-collections/" (:id coll)) + params {:body (assoc coll :name "coll2")} + [status data] (th/http-put user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "coll2"))))))) + +(t/deftest test-http-icon-collection-delete + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "coll1" + :data #{1}} + coll (icons/create-collection conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/icon-collections/" (:id coll)) + [status data] (th/http-delete user uri)] + (t/is (= 204 status)) + (let [sqlv (sql/get-icon-collections {:user (:id user)}) + result (sc/fetch conn sqlv)] + (t/is (empty? result)))))))) + +(t/deftest test-http-create-icon + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/icons") + data {:name "sample.jpg" + :content "" + :metadata {:width 200 + :height 200 + :view-box [0 0 200 200]} + :collection nil} + params {:body data} + [status data] (th/http-post user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 201 status)) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "sample.jpg")) + (t/is (= (:metadata data) {:width 200 + :height 200 + :view-box [0 0 200 200]}))))))) + +(t/deftest test-http-update-icon + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "test.svg" + :content "" + :metadata {} + :collection nil} + icon (icons/create-icon conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/icons/" (:id icon)) + params {:body (assoc icon :name "my stuff")} + [status data] (th/http-put user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "my stuff"))))))) + +(t/deftest test-http-copy-icon + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "test.svg" + :content "" + :metadata {} + :collection nil} + icon (icons/create-icon conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/icons/copy") + body {:id (:id icon) :collection nil} + params {:body body} + [status data] (th/http-put user uri params)] + (println "RESPONSE:" status data) + (let [sqlv (sql/get-icons {:user (:id user) :collection nil}) + result (sc/fetch conn sqlv)] + (t/is (= 2 (count result))))))))) + +(t/deftest test-http-delete-icon + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "test.svg" + :content "" + :metadata {} + :collection nil} + icon (icons/create-icon conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/icons/" (:id icon)) + [status data] (th/http-delete user uri)] + (t/is (= 204 status)) + (let [sqlv (sql/get-icons {:user (:id user) :collection nil}) + result (sc/fetch conn sqlv)] + (t/is (empty? result)))))))) + +;; (t/deftest test-http-list-icons +;; (with-open [conn (db/connection)] +;; (let [user (th/create-user conn 1) +;; data {:user (:id user) +;; :name "test.png" +;; :path "some/path" +;; :collection nil} +;; icon (icons/create-icon conn data)] +;; (with-server {:handler (uft/routes)} +;; (let [uri (str th/+base-url+ "/api/library/icons") +;; [status data] (th/http-get user uri)] +;; (println "RESPONSE:" status data) +;; (t/is (= 200 status)) +;; (t/is (= 1 (count data)))))))) diff --git a/backend/test/uxbox/tests/test_images.clj b/backend/test/uxbox/tests/test_images.clj new file mode 100644 index 000000000..385183f61 --- /dev/null +++ b/backend/test/uxbox/tests/test_images.clj @@ -0,0 +1,169 @@ +(ns uxbox.tests.test-images + (:require [clojure.test :as t] + [promesa.core :as p] + [suricatta.core :as sc] + [clojure.java.io :as io] + [catacumba.testing :refer (with-server)] + [storages.core :as st] + [uxbox.db :as db] + [uxbox.sql :as sql] + [uxbox.media :as media] + [uxbox.frontend :as uft] + [uxbox.services.images :as images] + [uxbox.services :as usv] + [uxbox.tests.helpers :as th])) + +(t/use-fixtures :each th/database-reset) + +(t/deftest test-http-list-image-collections + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "coll1"} + coll (images/create-collection conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/image-collections") + [status data] (th/http-get user uri)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= 1 (count data)))))))) + +(t/deftest test-http-create-image-collection + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/image-collections") + data {:user (:id user) + :name "coll1"} + params {:body data} + [status data] (th/http-post user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 201 status)) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "coll1"))))))) + +(t/deftest test-http-update-image-collection + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "coll1"} + coll (images/create-collection conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/image-collections/" (:id coll)) + params {:body (assoc coll :name "coll2")} + [status data] (th/http-put user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "coll2"))))))) + +(t/deftest test-http-image-collection-delete + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "coll1" + :data #{1}} + coll (images/create-collection conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/image-collections/" (:id coll)) + [status data] (th/http-delete user uri)] + (t/is (= 204 status)) + (let [sqlv (sql/get-image-collections {:user (:id user)}) + result (sc/fetch conn sqlv)] + (t/is (empty? result)))))))) + +;; (t/deftest test-http-create-image +;; (with-open [conn (db/connection)] +;; (let [user (th/create-user conn 1)] +;; (with-server {:handler (uft/routes)} +;; (let [uri (str th/+base-url+ "/api/library/images") +;; params [{:name "sample.jpg" +;; :part-name "file" +;; :content (io/input-stream +;; (io/resource "uxbox/tests/_files/sample.jpg"))}] +;; [status data] (th/http-multipart user uri params)] +;; ;; (println "RESPONSE:" status data) +;; (t/is (= 201 status)) +;; (t/is (= (:user data) (:id user))) +;; (t/is (= (:name data) "sample.jpg"))))))) + +(t/deftest test-http-update-image + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "test.png" + :path "some/path" + :width 100 + :height 100 + :mimetype "image/png" + :collection nil} + img (images/create-image conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/images/" (:id img)) + params {:body (assoc img :name "my stuff")} + [status data] (th/http-put user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "my stuff"))))))) + +(t/deftest test-http-copy-image + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + storage media/images-storage + filename "sample.jpg" + rcs (io/resource "uxbox/tests/_files/sample.jpg") + path @(st/save storage filename rcs) + data {:user (:id user) + :name filename + :path (str path) + :width 100 + :height 100 + :mimetype "image/jpg" + :collection nil} + img (images/create-image conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/images/copy") + body {:id (:id img) + :collection nil} + params {:body body} + [status data] (th/http-put user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (let [sqlv (sql/get-images {:user (:id user) :collection nil}) + result (sc/fetch conn sqlv)] + (t/is (= 2 (count result))))))))) + +(t/deftest test-http-delete-image + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + data {:user (:id user) + :name "test.png" + :path "some/path" + :width 100 + :height 100 + :mimetype "image/png" + :collection nil} + img (images/create-image conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/library/images/" (:id img)) + [status data] (th/http-delete user uri)] + (t/is (= 204 status)) + (let [sqlv (sql/get-images {:user (:id user) :collection nil}) + result (sc/fetch conn sqlv)] + (t/is (empty? result)))))))) + +;; (t/deftest test-http-list-images +;; (with-open [conn (db/connection)] +;; (let [user (th/create-user conn 1) +;; data {:user (:id user) +;; :name "test.png" +;; :path "some/path" +;; :collection nil} +;; img (images/create-image conn data)] +;; (with-server {:handler (uft/routes)} +;; (let [uri (str th/+base-url+ "/api/library/images") +;; [status data] (th/http-get user uri)] +;; (println "RESPONSE:" status data) +;; (t/is (= 200 status)) +;; (t/is (= 1 (count data)))))))) diff --git a/backend/test/uxbox/tests/test_kvstore.clj b/backend/test/uxbox/tests/test_kvstore.clj new file mode 100644 index 000000000..c56c9fc99 --- /dev/null +++ b/backend/test/uxbox/tests/test_kvstore.clj @@ -0,0 +1,65 @@ +(ns uxbox.tests.test-kvstore + (:require [clojure.test :as t] + [promesa.core :as p] + [suricatta.core :as sc] + [catacumba.testing :refer (with-server)] + [buddy.core.codecs :as codecs] + [uxbox.db :as db] + [uxbox.util.uuid :as uuid] + [uxbox.frontend :as uft] + [uxbox.services.kvstore :as kvs] + [uxbox.tests.helpers :as th])) + +(t/use-fixtures :each th/database-reset) + +(t/deftest test-http-kvstore + (with-open [conn (db/connection)] + (let [{:keys [id] :as user} (th/create-user conn 1)] + + ;; Not exists at this moment + (t/is (nil? (kvs/retrieve-kvstore conn {:user id :key "foo" :version -1}))) + + ;; Creating new one should work as expected + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/kvstore") + body {:key "foo" :value "bar" :version -1} + params {:body body} + [status data] (th/http-put user uri params)] + (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:key data) "foo")) + (t/is (= (:value data) "bar")))) + + ;; Should exists + (let [data (kvs/retrieve-kvstore conn {:user id :key "foo"})] + (t/is (= (:key data) "foo")) + (t/is (= (:value data) "bar")) + + ;; Overwriting should work + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/kvstore") + body (assoc data :key "foo" :value "baz") + params {:body body} + [status data] (th/http-put user uri params)] + (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:key data) "foo")) + (t/is (= (:value data) "baz"))))) + + ;; Should exists and match the overwritten value + (let [data (kvs/retrieve-kvstore conn {:user id :key "foo"})] + (t/is (= (:key data) "foo")) + (t/is (= (:value data) "baz"))) + + ;; Delete should work + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/kvstore/foo") + [status data] (th/http-delete user uri)] + (println "RESPONSE:" status data) + (t/is (= 204 status)))) + + ;; Not exists at this moment + (t/is (nil? (kvs/retrieve-kvstore conn {:user id :key "foo"}))) + + ))) + diff --git a/backend/test/uxbox/tests/test_pages.clj b/backend/test/uxbox/tests/test_pages.clj new file mode 100644 index 000000000..98964b9f3 --- /dev/null +++ b/backend/test/uxbox/tests/test_pages.clj @@ -0,0 +1,224 @@ +(ns uxbox.tests.test-pages + (:require [clojure.test :as t] + [promesa.core :as p] + [suricatta.core :as sc] + [catacumba.testing :refer (with-server)] + [buddy.core.codecs :as codecs] + [uxbox.db :as db] + [uxbox.util.uuid :as uuid] + [uxbox.frontend :as uft] + [uxbox.services.projects :as uspr] + [uxbox.services.pages :as uspg] + [uxbox.services :as usv] + [uxbox.tests.helpers :as th])) + +(t/use-fixtures :each th/database-reset) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Frontend Test +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(t/deftest test-http-page-create + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"})] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/pages") + params {:body {:project (:id proj) + :name "page1" + :data "1" + :metadata "1" + :width 200 + :height 200 + :layout "mobile"}} + [status data] (th/http-post user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 201 status)) + (t/is (= (:data (:body params)) (:data data))) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "page1"))))))) + +(t/deftest test-http-page-update + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"}) + data {:id (uuid/random) + :user (:id user) + :project (:id proj) + :version 0 + :data "1" + :metadata "2" + :name "page1" + :width 200 + :height 200 + :layout "mobil"} + page (uspg/create-page conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ (str "/api/pages/" (:id page))) + params {:body (assoc page :data "3")} + [status page'] (th/http-put user uri params)] + ;; (println "RESPONSE:" status page') + (t/is (= 200 status)) + (t/is (= "3" (:data page'))) + (t/is (= 1 (:version page'))) + (t/is (= (:user page') (:id user))) + (t/is (= (:name data) "page1"))))))) + +(t/deftest test-http-page-update-metadata + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"}) + data {:id (uuid/random) + :user (:id user) + :project (:id proj) + :version 0 + :data "1" + :metadata "2" + :name "page1" + :width 200 + :height 200 + :layout "mobil"} + page (uspg/create-page conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ (str "/api/pages/" (:id page) "/metadata")) + params {:body (assoc page :data "3")} + [status page'] (th/http-put user uri params)] + ;; (println "RESPONSE:" status page') + (t/is (= 200 status)) + (t/is (= "1" (:data page'))) + (t/is (= 1 (:version page'))) + (t/is (= (:user page') (:id user))) + (t/is (= (:name data) "page1"))))))) + +(t/deftest test-http-page-delete + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"}) + data {:id (uuid/random) + :user (:id user) + :project (:id proj) + :version 0 + :data "1" + :metadata "2" + :name "page1" + :width 200 + :height 200 + :layout "mobil"} + page (uspg/create-page conn data)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ (str "/api/pages/" (:id page))) + [status response] (th/http-delete user uri)] + ;; (println "RESPONSE:" status response) + (t/is (= 204 status)) + (let [sqlv ["SELECT * FROM pages WHERE \"user\"=? AND deleted_at is null" + (:id user)] + result (sc/fetch conn sqlv)] + (t/is (empty? result)))))))) + +(t/deftest test-http-page-list-by-project + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj1 (uspr/create-project conn {:user (:id user) :name "proj1"}) + proj2 (uspr/create-project conn {:user (:id user) :name "proj2"}) + data {:user (:id user) + :version 0 + :data "1" + :metadata "2" + :name "page1" + :width 200 + :height 200 + :layout "mobil"} + page1 (uspg/create-page conn (assoc data :project (:id proj1))) + page2 (uspg/create-page conn (assoc data :project (:id proj2)))] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ (str "/api/projects/" (:id proj1) "/pages")) + [status response] (th/http-get user uri)] + ;; (println "RESPONSE:" status response) + (t/is (= 200 status)) + (t/is (= 1 (count response))) + (t/is (= (:id (first response)) (:id page1)))))))) + +(t/deftest test-http-page-history-retrieve + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"}) + data {:id (uuid/random) + :user (:id user) + :project (:id proj) + :version 0 + :data "1" + :metadata "2" + :name "page1" + :width 200 + :height 200 + :layout "mobil"} + page (uspg/create-page conn data)] + (dotimes [i 100] + (let [page (uspg/get-page-by-id conn (:id data))] + (uspg/update-page conn (assoc page :data (str i))))) + + ;; Check inserted history + (let [sqlv ["SELECT * FROM pages_history WHERE page=?" (:id data)] + result (sc/fetch conn sqlv)] + (t/is (= (count result) 100))) + + ;; Check retrieve all items + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/pages/" (:id page) "/history") + [status result] (th/http-get user uri nil)] + ;; (println "RESPONSE:" status result) + (t/is (= (count result) 10)) + (t/is (= 200 status)) + (t/is (= 99 (:version (first result)))) + + (let [params {:query {:since (:version (last result)) + :max 20}} + [status result] (th/http-get user uri params)] + ;; (println "RESPONSE:" status result) + (t/is (= (count result) 20)) + (t/is (= 200 status)) + (t/is (= 89 (:version (first result)))))) + )))) + +(t/deftest test-http-page-history-update + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"}) + data {:id (uuid/random) + :user (:id user) + :project (:id proj) + :version 0 + :data "1" + :metadata "2" + :name "page1" + :width 200 + :height 200 + :layout "mobil"} + page (uspg/create-page conn data)] + + (dotimes [i 10] + (let [page (uspg/get-page-by-id conn (:id data))] + (uspg/update-page conn (assoc page :data (str i))))) + + ;; Check inserted history + (let [sql (str "SELECT * FROM pages_history " + " WHERE page=? ORDER BY created_at DESC") + result (sc/fetch conn [sql (:id data)]) + item (first result)] + + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ + "/api/pages/" (:id page) + "/history/" (:id item)) + params {:body {:label "test" :pinned true}} + [status data] (th/http-put user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:id data) (:id item)))))) + + (let [sql (str "SELECT * FROM pages_history " + " WHERE page=? AND pinned = true " + " ORDER BY created_at DESC") + result (sc/fetch-one conn [sql (:id data)])] + (t/is (= "test" (:label result))) + (t/is (= true (:pinned result))))))) diff --git a/backend/test/uxbox/tests/test_projects.clj b/backend/test/uxbox/tests/test_projects.clj new file mode 100644 index 000000000..963e158aa --- /dev/null +++ b/backend/test/uxbox/tests/test_projects.clj @@ -0,0 +1,92 @@ +(ns uxbox.tests.test-projects + (:require [clojure.test :as t] + [promesa.core :as p] + [suricatta.core :as sc] + [clj-uuid :as uuid] + [catacumba.testing :refer (with-server)] + [catacumba.serializers :as sz] + [uxbox.db :as db] + [uxbox.frontend :as uft] + [uxbox.services.projects :as uspr] + [uxbox.services.pages :as uspg] + [uxbox.services :as usv] + [uxbox.tests.helpers :as th])) + +(t/use-fixtures :each th/database-reset) + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +;; Frontend Test +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +(t/deftest test-http-project-list + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"})] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/projects") + [status data] (th/http-get user uri)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= 1 (count data)))))))) + +(t/deftest test-http-project-create + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/projects") + params {:body {:name "proj1"}} + [status data] (th/http-post user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 201 status)) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "proj1"))))))) + +(t/deftest test-http-project-update + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"})] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/projects/" (:id proj)) + params {:body (assoc proj :name "proj2")} + [status data] (th/http-put user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:user data) (:id user))) + (t/is (= (:name data) "proj2"))))))) + +(t/deftest test-http-project-delete + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"})] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/projects/" (:id proj)) + [status data] (th/http-delete user uri)] + (t/is (= 204 status)) + (let [sqlv ["SELECT * FROM projects WHERE \"user\"=? AND deleted_at is null" + (:id user)] + result (sc/fetch conn sqlv)] + (t/is (empty? result)))))))) + +(t/deftest test-http-project-retrieve-by-share-token + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + proj (uspr/create-project conn {:user (:id user) :name "proj1"}) + page (uspg/create-page conn {:id (uuid/v4) + :user (:id user) + :project (:id proj) + :version 0 + :data "1" + :options "2" + :name "page1" + :width 200 + :height 200 + :layout "mobil"}) + shares (uspr/get-share-tokens-for-project conn (:id proj))] + (with-server {:handler (uft/routes)} + (let [token (:token (first shares)) + uri (str th/+base-url+ "/api/projects-by-token/" token) + [status data] (th/http-get user uri)] + ;; (println "RESPONSE:" status data) + (t/is (= status 200)) + (t/is (vector? (:pages data))) + (t/is (= 1 (count (:pages data))))))))) diff --git a/backend/test/uxbox/tests/test_svgparse.clj b/backend/test/uxbox/tests/test_svgparse.clj new file mode 100644 index 000000000..4c3f7277e --- /dev/null +++ b/backend/test/uxbox/tests/test_svgparse.clj @@ -0,0 +1,89 @@ +(ns uxbox.tests.test-svgparse + (:require [clojure.test :as t] + [clojure.java.io :as io] + [catacumba.testing :refer [with-server]] + [uxbox.frontend :as uft] + [uxbox.services :as usv] + [uxbox.services.svgparse :as svg] + [uxbox.tests.helpers :as th])) + +(t/use-fixtures :each th/state-init) + +(t/deftest parse-svg-test + (t/testing "parsing valid svg 1" + (let [image (slurp (io/resource "uxbox/tests/_files/sample1.svg")) + result (svg/parse-string image)] + (t/is (contains? result :width)) + (t/is (contains? result :height)) + (t/is (contains? result :view-box)) + (t/is (contains? result :name)) + (t/is (contains? result :content)) + (t/is (= 500.0 (:width result))) + (t/is (= 500.0 (:height result))) + (t/is (= [0.0 0.0 500.00001 500.00001] (:view-box result))) + (t/is (= "lock.svg" (:name result))))) + + (t/testing "parsing valid svg 2" + (let [image (slurp (io/resource "uxbox/tests/_files/sample2.svg")) + result (svg/parse-string image)] + (t/is (contains? result :width)) + (t/is (contains? result :height)) + (t/is (contains? result :view-box)) + (t/is (contains? result :name)) + (t/is (contains? result :content)) + (t/is (= 500.0 (:width result))) + (t/is (= 500.0 (:height result))) + (t/is (= [0.0 0.0 500.0 500.00001] (:view-box result))) + (t/is (= "play.svg" (:name result))))) + + (t/testing "parsing invalid data 1" + (let [image (slurp (io/resource "uxbox/tests/_files/sample.jpg")) + [e result] (th/try-on (svg/parse-string image))] + (t/is (th/exception? e)) + (t/is (th/ex-info? e)) + (t/is (th/ex-with-code? e :uxbox.services.svgparse/invalid-input)))) + + (t/testing "parsing invalid data 2" + (let [[e result] (th/try-on (svg/parse-string ""))] + (t/is (th/exception? e)) + (t/is (th/ex-info? e)) + (t/is (th/ex-with-code? e :uxbox.services.svgparse/invalid-input)))) + + (t/testing "parsing invalid data 3" + (let [[e result] (th/try-on (svg/parse-string ""))] + (t/is (th/exception? e)) + (t/is (th/ex-info? e)) + (t/is (th/ex-with-code? e :uxbox.services.svgparse/invalid-result)))) + + (t/testing "valid http request" + (let [image (slurp (io/resource "uxbox/tests/_files/sample2.svg")) + path "/api/svg/parse"] + (with-server {:handler (uft/routes)} + (let [rsp (th/request {:method :post + :path path + :body image + :raw? true})] + (t/is (= 200 (:status rsp))) + (t/is (contains? (:body rsp) :width)) + (t/is (contains? (:body rsp) :height)) + (t/is (contains? (:body rsp) :view-box)) + (t/is (contains? (:body rsp) :name)) + (t/is (contains? (:body rsp) :content)) + (t/is (= 500.0 (:width (:body rsp)))) + (t/is (= 500.0 (:height (:body rsp)))) + (t/is (= [0.0 0.0 500.0 500.00001] (:view-box (:body rsp)))) + (t/is (= "play.svg" (:name (:body rsp)))))))) + + (t/testing "invalid http request" + (let [path "/api/svg/parse" + image ""] + (with-server {:handler (uft/routes)} + (let [rsp (th/request {:method :post + :path path + :body image + :raw? true})] + (t/is (= 400 (:status rsp))) + (t/is (= :validation (get-in rsp [:body :type]))) + (t/is (= ::svg/invalid-result (get-in rsp [:body :code]))))))) + + ) diff --git a/backend/test/uxbox/tests/test_txlog.clj b/backend/test/uxbox/tests/test_txlog.clj new file mode 100644 index 000000000..d8099664c --- /dev/null +++ b/backend/test/uxbox/tests/test_txlog.clj @@ -0,0 +1,19 @@ +(ns uxbox.tests.test-txlog + "A txlog and services abstraction generic tests." + (:require [clojure.test :as t] + [promesa.core :as p] + [uxbox.services.core :as usc] + [uxbox.services :as usv] + [uxbox.tests.helpers :as th])) + +(t/use-fixtures :each th/database-reset) + +(defmethod usc/novelty ::testype1 + [data] + true) + +(t/deftest txlog-spec1 + (let [data {:type ::testype1 :foo 1 :bar "baz"} + response (usv/novelty data)] + (t/is (p/promise? response)) + (t/is (= true @response)))) diff --git a/backend/test/uxbox/tests/test_users.clj b/backend/test/uxbox/tests/test_users.clj new file mode 100644 index 000000000..4d1773ead --- /dev/null +++ b/backend/test/uxbox/tests/test_users.clj @@ -0,0 +1,120 @@ +(ns uxbox.tests.test-users + (:require [clojure.test :as t] + [clojure.java.io :as io] + [promesa.core :as p] + [buddy.hashers :as hashers] + [clj-http.client :as http] + [suricatta.core :as sc] + [catacumba.testing :refer (with-server)] + [uxbox.db :as db] + [uxbox.frontend :as uft] + [uxbox.services.users :as usu] + [uxbox.services :as usv] + [uxbox.tests.helpers :as th])) + +(t/use-fixtures :each th/database-reset) + +(t/deftest test-http-retrieve-profile + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/profile/me") + [status data] (th/http-get user uri)] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:fullname data) "User 1")) + (t/is (= (:username data) "user1")) + (t/is (= (:metadata data) "1")) + (t/is (= (:email data) "user1@uxbox.io")) + (t/is (not (contains? data :password)))))))) + +(t/deftest test-http-update-profile + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/profile/me") + data (assoc user + :fullname "Full Name" + :username "user222" + :metadata "222" + :email "user222@uxbox.io") + [status data] (th/http-put user uri {:body data})] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status)) + (t/is (= (:fullname data) "Full Name")) + (t/is (= (:username data) "user222")) + (t/is (= (:metadata data) "222")) + (t/is (= (:email data) "user222@uxbox.io")) + (t/is (not (contains? data :password)))))))) + +(t/deftest test-http-update-profile-photo + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1)] + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/profile/me/photo") + params [{:name "sample.jpg" + :part-name "file" + :content (io/input-stream + (io/resource "uxbox/tests/_files/sample.jpg"))}] + [status data] (th/http-multipart user uri params)] + ;; (println "RESPONSE:" status data) + (t/is (= 204 status))))))) + +(t/deftest test-http-register-user + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/auth/register") + data {:fullname "Full Name" + :username "user222" + :email "user222@uxbox.io" + :password "user222"} + [status data] (th/http-post uri {:body data})] + ;; (println "RESPONSE:" status data) + (t/is (= 200 status))))) + +(t/deftest test-http-validate-recovery-token + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1)] + (with-server {:handler (uft/routes)} + (let [token (#'usu/request-password-recovery conn "user1") + uri1 (str th/+base-url+ "/api/auth/recovery/not-existing") + uri2 (str th/+base-url+ "/api/auth/recovery/" token) + [status1 data1] (th/http-get user uri1) + [status2 data2] (th/http-get user uri2)] + ;; (println "RESPONSE:" status1 data1) + ;; (println "RESPONSE:" status2 data2) + (t/is (= 404 status1)) + (t/is (= 204 status2))))))) + +(t/deftest test-http-request-password-recovery + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1) + sql "select * from user_pswd_recovery" + res (sc/fetch-one conn sql)] + + ;; Initially no tokens exists + (t/is (nil? res)) + + (with-server {:handler (uft/routes)} + (let [uri (str th/+base-url+ "/api/auth/recovery") + data {:username "user1"} + [status data] (th/http-post user uri {:body data})] + ;; (println "RESPONSE:" status data) + (t/is (= 204 status))) + + (let [res (sc/fetch-one conn sql)] + (t/is (not (nil? res))) + (t/is (= (:user res) (:id user)))))))) + +(t/deftest test-http-validate-recovery-token + (with-open [conn (db/connection)] + (let [user (th/create-user conn 1)] + (with-server {:handler (uft/routes)} + (let [token (#'usu/request-password-recovery conn (:username user)) + uri (str th/+base-url+ "/api/auth/recovery") + data {:token token :password "mytestpassword"} + [status data] (th/http-put user uri {:body data}) + + user' (usu/find-full-user-by-id conn (:id user))] + (t/is (= status 204)) + (t/is (hashers/check "mytestpassword" (:password user')))))))) + diff --git a/backend/vendor/executors/core.clj b/backend/vendor/executors/core.clj new file mode 100644 index 000000000..4bffc365b --- /dev/null +++ b/backend/vendor/executors/core.clj @@ -0,0 +1,176 @@ +;; 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) 2016 Andrey Antukh + +(ns executors.core + "A executos service abstraction layer." + (:import java.util.function.Supplier + java.util.concurrent.ForkJoinPool + java.util.concurrent.Future + java.util.concurrent.CompletableFuture + java.util.concurrent.ExecutorService + java.util.concurrent.TimeoutException + java.util.concurrent.ThreadFactory + java.util.concurrent.TimeUnit + java.util.concurrent.ScheduledExecutorService + java.util.concurrent.Executors)) + +(def ^:const +max-priority+ Thread/MAX_PRIORITY) +(def ^:const +min-priority+ Thread/MIN_PRIORITY) +(def ^:const +norm-priority+ Thread/NORM_PRIORITY) + +;; --- Protocols + +(defprotocol IExecutor + (^:private -execute [_ task] "Execute a task in a executor.") + (^:private -submit [_ task] "Submit a task and return a promise.")) + +(defprotocol IScheduledExecutor + (^:provate -schedule [_ ms task] "Schedule a task to execute in a future.")) + +(defprotocol IScheduledTask + "A cancellation abstraction." + (-cancel [_]) + (-cancelled? [_])) + +;; --- Implementation + +(defn- thread-factory-adapter + "Adapt a simple clojure function into a + ThreadFactory instance." + [func] + (reify ThreadFactory + (^Thread newThread [_ ^Runnable runnable] + (func runnable)))) + +(defn- thread-factory + [{:keys [daemon priority] + :or {daemon true + priority Thread/NORM_PRIORITY}}] + (thread-factory-adapter + (fn [runnable] + (let [thread (Thread. ^Runnable runnable)] + (.setDaemon thread daemon) + (.setPriority thread priority) + thread)))) + +(defn- resolve-thread-factory + [opts] + (cond + (map? opts) (thread-factory opts) + (fn? opts) (thread-factory-adapter opts) + (instance? ThreadFactory opts) opts + :else (throw (ex-info "Invalid thread factory" {})))) + +(deftype ScheduledTask [^Future fut] + clojure.lang.IDeref + (deref [_] + (.get fut)) + + clojure.lang.IBlockingDeref + (deref [_ ms default] + (try + (.get fut ms TimeUnit/MILLISECONDS) + (catch TimeoutException e + default))) + + clojure.lang.IPending + (isRealized [_] (and (.isDone fut) + (not (.isCancelled fut)))) + + IScheduledTask + (-cancelled? [_] + (.isCancelled fut)) + + (-cancel [_] + (when-not (.isCancelled fut) + (.cancel fut true)))) + +(extend-type ExecutorService + IExecutor + (-execute [this task] + (CompletableFuture/runAsync ^Runnable task this)) + + (-submit [this task] + (let [supplier (reify Supplier (get [_] (task)))] + (CompletableFuture/supplyAsync supplier this)))) + +(extend-type ScheduledExecutorService + IScheduledExecutor + (-schedule [this ms func] + (let [fut (.schedule this func ms TimeUnit/MILLISECONDS)] + (ScheduledTask. fut)))) + +;; --- Public Api (Pool Constructors) + +(defn common-pool + "Get the common pool." + [] + (ForkJoinPool/commonPool)) + +(defn cached + "A cached thread pool constructor." + ([] + (Executors/newCachedThreadPool)) + ([opts] + (let [factory (resolve-thread-factory opts)] + (Executors/newCachedThreadPool factory)))) + +(defn fixed + "A fixed thread pool constructor." + ([n] + (Executors/newFixedThreadPool (int n))) + ([n opts] + (let [factory (resolve-thread-factory opts)] + (Executors/newFixedThreadPool (int n) factory)))) + +(defn single-thread + "A single thread pool constructor." + ([] + (Executors/newSingleThreadExecutor)) + ([opts] + (let [factory (resolve-thread-factory opts)] + (Executors/newSingleThreadExecutor factory)))) + +(defn scheduled + "A scheduled thread pool constructo." + ([] (Executors/newScheduledThreadPool (int 1))) + ([n] (Executors/newScheduledThreadPool (int n))) + ([n opts] + (let [factory (resolve-thread-factory opts)] + (Executors/newScheduledThreadPool (int n) factory)))) + +;; --- Public Api (Task Execution) + +(defn execute + "Execute a task in a provided executor. + + A task is a plain clojure function or + jvm Runnable instance." + ([task] + (-> (common-pool) + (-execute task))) + ([executor task] + (-execute executor task))) + +(defn submit + "Submit a task to be executed in a provided executor + and return a promise that will be completed with + the return value of a task. + + A task is a plain clojure function." + ([task] + (-> (common-pool) + (-submit task))) + ([executor task] + (-submit executor task))) + +(defn schedule + "Schedule task exection for some time in the future." + ([ms task] + (-> (common-pool) + (-schedule ms task))) + ([executor ms task] + (-schedule executor ms task))) diff --git a/backend/vendor/storages/core.clj b/backend/vendor/storages/core.clj new file mode 100644 index 000000000..01aad5fcf --- /dev/null +++ b/backend/vendor/storages/core.clj @@ -0,0 +1,59 @@ +;; 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) 2016 Andrey Antukh + +(ns storages.core + "A storages abstraction layer." + (:require [storages.proto :as pt] + [storages.impl])) + +(defn save + "Perists a file or bytes in the storage. This function + returns a relative path where file is saved. + + The final file path can be different to the one provided + to this function and the behavior is totally dependen on + the storage implementation." + [storage path content] + (pt/-save storage path content)) + +(defn lookup + "Resolve provided relative path in the storage and return + the local filesystem absolute path to it. + This method may be not implemented in all storages." + [storage path] + {:pre [(satisfies? pt/ILocalStorage storage)]} + (pt/-lookup storage path)) + +(defn exists? + "Check if a relative `path` exists in the storage." + [storage path] + (pt/-exists? storage path)) + +(defn delete + "Delete a file from the storage." + [storage path] + (pt/-delete storage path)) + +(defn clear! + "Clear all contents of the storage." + [storage] + (pt/-clear storage)) + +(defn path + "Create path from string or more than one string." + ([fst] + (pt/-path fst)) + ([fst & more] + (pt/-path (cons fst more)))) + +(defn public-url + [storage path] + (pt/-public-uri storage path)) + +(defn storage? + "Return `true` if `v` implements IStorage protocol" + [v] + (satisfies? pt/IStorage v)) diff --git a/backend/vendor/storages/fs/local.clj b/backend/vendor/storages/fs/local.clj new file mode 100644 index 000000000..a6d964cde --- /dev/null +++ b/backend/vendor/storages/fs/local.clj @@ -0,0 +1,104 @@ +;; 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) 2016 Andrey Antukh + +(ns storages.fs.local + "A local filesystem storage implementation." + (:require [promesa.core :as p] + [clojure.java.io :as io] + [executors.core :as exec] + [storages.proto :as pt] + [storages.impl :as impl] + [storages.util :as util]) + (:import java.io.InputStream + java.io.OutputStream + java.net.URI + java.nio.file.Path + java.nio.file.Files)) + +(defn normalize-path + [^Path base ^Path path] + (if (util/absolute? path) + (throw (ex-info "Suspicios operation: absolute path not allowed." + {:path (str path)})) + (let [^Path fullpath (.resolve base path) + ^Path fullpath (.normalize fullpath)] + (when-not (.startsWith fullpath base) + (throw (ex-info "Suspicios operation: go to parent dir is not allowed." + {:path (str path)}))) + fullpath))) + +(defn- save + [base path content] + (let [^Path path (pt/-path path) + ^Path fullpath (normalize-path base path)] + (when-not (util/exists? (.getParent fullpath)) + (util/create-dir! (.getParent fullpath))) + (with-open [^InputStream source (pt/-input-stream content) + ^OutputStream dest (Files/newOutputStream + fullpath util/write-open-opts)] + (io/copy source dest) + path))) + +(defn- delete + [base path] + (let [path (->> (pt/-path path) + (normalize-path base))] + (Files/deleteIfExists ^Path path))) + +(defrecord FileSystemStorage [^Path base ^URI baseuri] + pt/IPublicStorage + (-public-uri [_ path] + (.resolve baseuri (str path))) + + pt/IStorage + (-save [_ path content] + (exec/submit (partial save base path content))) + + (-delete [_ path] + (exec/submit (partial delete base path))) + + (-exists? [this path] + (try + (p/resolved + (let [path (->> (pt/-path path) + (normalize-path base))] + (util/exists? path))) + (catch Exception e + (p/rejected e)))) + + pt/IClearableStorage + (-clear [_] + (util/delete-dir! base) + (util/create-dir! base)) + + pt/ILocalStorage + (-lookup [_ path'] + (try + (p/resolved + (->> (pt/-path path') + (normalize-path base))) + (catch Exception e + (p/rejected e))))) + +(defn filesystem + "Create an instance of local FileSystem storage providing an + absolute base path. + + If that path does not exists it will be automatically created, + if it exists but is not a directory, an exception will be + raised." + [{:keys [basedir baseuri] :as keys}] + (let [^Path basepath (pt/-path basedir) + ^URI baseuri (pt/-uri baseuri)] + (when (and (util/exists? basepath) + (not (util/directory? basepath))) + (throw (ex-info "File already exists." {}))) + + (when-not (util/exists? basepath) + (util/create-dir! basepath)) + + (->FileSystemStorage basepath baseuri))) + diff --git a/backend/vendor/storages/fs/misc.clj b/backend/vendor/storages/fs/misc.clj new file mode 100644 index 000000000..4dab9bf1f --- /dev/null +++ b/backend/vendor/storages/fs/misc.clj @@ -0,0 +1,111 @@ +;; 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) 2016 Andrey Antukh + +(ns storages.fs.misc + "A local filesystem storage implementation." + (:require [promesa.core :as p] + [cuerdas.core :as str] + [buddy.core.codecs :as codecs] + [buddy.core.codecs.base64 :as b64] + [buddy.core.nonce :as nonce] + [buddy.core.hash :as hash] + [storages.proto :as pt] + [storages.impl :as impl] + [storages.fs.local :as localfs]) + (:import java.io.InputStream + java.io.OutputStream + java.nio.file.Path + java.nio.file.Files)) + +;; --- Scoped Storage + +(defrecord ScopedPathStorage [storage ^Path prefix] + pt/IPublicStorage + (-public-uri [_ path] + (let [^Path path (pt/-path [prefix path])] + (pt/-public-uri storage path))) + + pt/IStorage + (-save [_ path content] + (let [^Path path (pt/-path [prefix path])] + (->> (pt/-save storage path content) + (p/map (fn [^Path path] + (.relativize prefix path)))))) + + (-delete [_ path] + (let [^Path path (pt/-path [prefix path])] + (pt/-delete storage path))) + + (-exists? [this path] + (let [^Path path (pt/-path [prefix path])] + (pt/-exists? storage path))) + + pt/ILocalStorage + (-lookup [_ path] + (->> (pt/-lookup storage "") + (p/map (fn [^Path base] + (let [base (pt/-path [base prefix])] + (->> (pt/-path path) + (localfs/normalize-path base)))))))) + +(defn scoped + "Create a composed storage instance that automatically prefixes + the path when content is saved. For the rest of methods it just + relies to the underlying storage. + + This is usefull for atomatically add sertain prefix to some + uploads." + [storage prefix] + (let [prefix (pt/-path prefix)] + (->ScopedPathStorage storage prefix))) + +;; --- Hashed Storage + +(defn- generate-path + [^Path path] + (let [name (str (.getFileName path)) + hash (-> (nonce/random-nonce 128) + (hash/blake2b-256) + (b64/encode true) + (codecs/bytes->str)) + tokens (re-seq #"[\w\d\-\_]{3}" hash) + path-tokens (take 6 tokens) + rest-tokens (drop 6 tokens) + path (pt/-path path-tokens) + frest (apply str rest-tokens)] + (pt/-path (list path frest name)))) + +(defrecord HashedStorage [storage] + pt/IPublicStorage + (-public-uri [_ path] + (pt/-public-uri storage path)) + + pt/IStorage + (-save [_ path content] + (let [^Path path (pt/-path path) + ^Path path (generate-path path)] + (pt/-save storage path content))) + + (-delete [_ path] + (pt/-delete storage path)) + + (-exists? [this path] + (pt/-exists? storage path)) + + pt/ILocalStorage + (-lookup [_ path] + (pt/-lookup storage path))) + +(defn hashed + "Create a composed storage instance that uses random + hash based directory tree distribution for the final + file path. + + This is usefull when you want to store files with + not predictable uris." + [storage] + (->HashedStorage storage)) + diff --git a/backend/vendor/storages/impl.clj b/backend/vendor/storages/impl.clj new file mode 100644 index 000000000..2ffdc0d70 --- /dev/null +++ b/backend/vendor/storages/impl.clj @@ -0,0 +1,115 @@ +;; 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) 2016 Andrey Antukh + +(ns storages.impl + "Implementation details and helpers." + (:require [storages.proto :as pt] + [storages.util :as util] + [buddy.core.codecs :as codecs] + [clojure.java.io :as io]) + (:import java.io.File + java.io.ByteArrayInputStream + java.io.InputStream + java.net.URL + java.net.URI + java.nio.file.Path + java.nio.file.Paths + java.nio.file.Files)) + +(extend-protocol pt/IContent + String + (-input-stream [v] + (ByteArrayInputStream. (codecs/str->bytes v))) + + Path + (-input-stream [v] + (io/input-stream v)) + + File + (-input-stream [v] + (io/input-stream v)) + + URI + (-input-stream [v] + (io/input-stream v)) + + URL + (-input-stream [v] + (io/input-stream v)) + + InputStream + (-input-stream [v] + v) + + ratpack.http.TypedData + (-input-stream [this] + (.getInputStream this))) + +(extend-protocol pt/IUri + URI + (-uri [v] v) + + String + (-uri [v] (URI. v))) + +(def ^:private empty-string-array + (make-array String 0)) + +(extend-protocol pt/IPath + Path + (-path [v] v) + + URI + (-path [v] (Paths/get v)) + + URL + (-path [v] (Paths/get (.toURI v))) + + String + (-path [v] (Paths/get v empty-string-array)) + + clojure.lang.Sequential + (-path [v] + (reduce #(.resolve %1 %2) + (pt/-path (first v)) + (map pt/-path (rest v))))) + +(defn- path->input-stream + [^Path path] + (Files/newInputStream path util/read-open-opts)) + +(defn- path->output-stream + [^Path path] + (Files/newOutputStream path util/write-open-opts)) + +(extend-type Path + io/IOFactory + (make-reader [path opts] + (let [^InputStream is (path->input-stream path)] + (io/make-reader is opts))) + (make-writer [path opts] + (let [^OutputStream os (path->output-stream path)] + (io/make-writer os opts))) + (make-input-stream [path opts] + (let [^InputStream is (path->input-stream path)] + (io/make-input-stream is opts))) + (make-output-stream [path opts] + (let [^OutputStream os (path->output-stream path)] + (io/make-output-stream os opts)))) + +(extend-type ratpack.http.TypedData + io/IOFactory + (make-reader [td opts] + (let [^InputStream is (.getInputStream td)] + (io/make-reader is opts))) + (make-writer [path opts] + (throw (UnsupportedOperationException. "read only object"))) + (make-input-stream [td opts] + (let [^InputStream is (.getInputStream td)] + (io/make-input-stream is opts))) + (make-output-stream [path opts] + (throw (UnsupportedOperationException. "read only object")))) + diff --git a/backend/vendor/storages/proto.clj b/backend/vendor/storages/proto.clj new file mode 100644 index 000000000..c80e6548a --- /dev/null +++ b/backend/vendor/storages/proto.clj @@ -0,0 +1,38 @@ +;; 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) 2016 Andrey Antukh + +(ns storages.proto + "A storage abstraction definition.") + +(defprotocol IUri + (-uri [_] "Coerce to uri.")) + +(defprotocol IPath + (-path [_] "Coerce to path.")) + +(defprotocol IContent + (-input-stream [_] "Coerce to input stream.")) + +(defprotocol IStorage + "A basic abstraction for storage access." + (-save [_ path content] "Persist the content under specified path.") + (-delete [_ path] "Delete the file by its path.") + (-exists? [_ path] "Check if file exists by path.")) + +(defprotocol IClearableStorage + (-clear [_] "clear all contents of the storage")) + +(defprotocol IPublicStorage + (-public-uri [_ path] "Get a public accessible uri for path.")) + +(defprotocol ILocalStorage + (-lookup [_ path] "Resolves the path to the local filesystem.")) + +(defprotocol IStorageIntrospection + (-accessed-time [_ path] "Return the last accessed time of the file.") + (-created-time [_ path] "Return the creation time of the file.") + (-modified-time [_ path] "Return the last modified time of the file.")) + diff --git a/backend/vendor/storages/util.clj b/backend/vendor/storages/util.clj new file mode 100644 index 000000000..0d367a398 --- /dev/null +++ b/backend/vendor/storages/util.clj @@ -0,0 +1,140 @@ +;; 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) 2016 Andrey Antukh + +(ns storages.util + "FileSystem related utils." + (:refer-clojure :exclude [name]) + (:require [storages.proto :as pt]) + (:import java.nio.file.Path + java.nio.file.Files + java.nio.file.LinkOption + java.nio.file.OpenOption + java.nio.file.StandardOpenOption + java.nio.file.SimpleFileVisitor + java.nio.file.FileVisitResult + java.nio.file.attribute.FileAttribute + java.nio.file.attribute.PosixFilePermissions + ratpack.form.UploadedFile)) + +;; --- Constants + +(def write-open-opts + (->> [#_StandardOpenOption/CREATE_NEW + StandardOpenOption/CREATE + StandardOpenOption/WRITE] + (into-array OpenOption))) + +(def read-open-opts + (->> [StandardOpenOption/READ] + (into-array OpenOption))) + +(def follow-link-opts + (into-array LinkOption [LinkOption/NOFOLLOW_LINKS])) + +;; --- Path Helpers + +(defn path + "Create path from string or more than one string." + ([fst] + (pt/-path fst)) + ([fst & more] + (pt/-path (cons fst more)))) + +(defn make-file-attrs + "Generate a array of `FileAttribute` instances + generated from `rwxr-xr-x` kind of expressions." + [^String expr] + (let [perms (PosixFilePermissions/fromString expr) + attr (PosixFilePermissions/asFileAttribute perms)] + (into-array FileAttribute [attr]))) + +(defn path? + "Return `true` if provided value is an instance of Path." + [v] + (instance? Path v)) + +(defn absolute? + "Return `true` if the provided path is absolute, `else` in case contrary. + The `path` parameter can be anything convertible to path instance." + [path] + (let [^Path path (pt/-path path)] + (.isAbsolute path))) + +(defn exists? + "Return `true` if the provided path exists, `else` in case contrary. + The `path` parameter can be anything convertible to path instance." + [path] + (let [^Path path (pt/-path path)] + (Files/exists path follow-link-opts))) + +(defn directory? + "Return `true` if the provided path is a directory, `else` in case contrary. + The `path` parameter can be anything convertible to path instance." + [path] + (let [^Path path (pt/-path path)] + (Files/isDirectory path follow-link-opts))) + +(defn parent + "Get parent path if it exists." + [path] + (.getParent ^Path (pt/-path path))) + +(defn base-name + "Get the file name." + [path] + (if (instance? UploadedFile path) + (.getFileName ^UploadedFile path) + (str (.getFileName ^Path (pt/-path path))))) + +(defn split-ext + "Returns a vector of `[name extension]`." + [path] + (let [base (base-name path) + i (.lastIndexOf base ".")] + (if (pos? i) + [(subs base 0 i) (subs base i)] + [base nil]))) + +(defn extension + "Return the extension part of a file." + [path] + (last (split-ext path))) + +(defn name + "Return the name part of a file." + [path] + (first (split-ext path))) + +(defn list-directory + [path] + (let [path (pt/-path path)] + (with-open [stream (Files/newDirectoryStream path)] + (vec stream)))) + +(defn list-files + [path] + (filter (complement directory?) (list-directory path))) + +;; --- Side-Effectfull Operations + +(defn create-dir! + "Create a new directory." + [path] + (let [^Path path (pt/-path path) + attrs (make-file-attrs "rwxr-xr-x")] + (Files/createDirectories path attrs))) + +(defn delete-dir! + [path] + (let [path (pt/-path path) + visitor (proxy [SimpleFileVisitor] [] + (visitFile [file attrs] + (Files/delete file) + FileVisitResult/CONTINUE) + (postVisitDirectory [dir exc] + (Files/delete dir) + FileVisitResult/CONTINUE))] + (Files/walkFileTree path visitor)))