diff --git a/src/metabase/models/permissions.clj b/src/metabase/models/permissions.clj index 45cc5304907fae38f748171a084ec1e70c3a70f4..a63c2c9f7272810446d6bb0b6b19f4db777b0e60 100644 --- a/src/metabase/models/permissions.clj +++ b/src/metabase/models/permissions.clj @@ -222,7 +222,8 @@ "Regex for a valid permissions path. The [[metabase.util.regex/rx]] macro is used to make the big-and-hairy regex somewhat readable." (u.regex/rx "^/" - ;; any path starting with /db/ is a DATA PERMISSIONS path + ;; any path containing /db/ is a DATA permissions path + ;; any path starting with /db/ is a DATA ACCESS permissions path (or ;; /db/:id/ -> permissions for the entire DB -- native and all schemas (and #"db/\d+/" @@ -243,6 +244,17 @@ ;; .../segmented/ -> Permissions to run a query against ;; a Table using GTAP (opt "segmented/")))))))))))) + ;; any path starting with /download/ is a DOWNLOAD permissions path + ;; /download/db/:id/ -> permissions to download 1M rows in query results + ;; /download/limited/db/:id/ -> permissions to download 1k rows in query results + (and "download/" + (opt "limited/") + (and #"db/\d+/" + (opt (or + "native/" + (and "schema/" + (opt (and path-char "*/" + (opt #"table/\d+/")))))))) ;; any path starting with /collection/ is a COLLECTION permissions path (and "collection/" (or @@ -606,6 +618,7 @@ [:or [:= :object (hx/literal "/")] [:like :object (hx/literal "/db/%")] + [:like :object (hx/literal "/download/%")] [:like :object (hx/literal "/block/db/%")]]]}) db-ids (delay (db/select-ids 'Database)) group-id->paths (reduce diff --git a/src/metabase/models/permissions/parse.clj b/src/metabase/models/permissions/parse.clj index 7ca143ed819ca0cf57a4688d7b81734fe0323e9e..54f599239c1ac07241fe60325cea3be53748be0e 100644 --- a/src/metabase/models/permissions/parse.clj +++ b/src/metabase/models/permissions/parse.clj @@ -12,7 +12,7 @@ (def ^:private grammar "Describes permission strings like /db/3/ or /collection/root/read/" - "permission = ( all | db | collection | block ) + "permission = ( all | db | download | collection | block ) all = <'/'> db = <'/db/'> #'\\d+' <'/'> ( native | schemas )? native = <'native/'> @@ -21,6 +21,14 @@ table = <'table/'> #'\\d+' <'/'> (table-perm <'/'>)? table-perm = ('read'|'query'|'query/segmented') + download = <'/download'> ( dl-limited | dl-db ) + dl-limited = <'/limited'> dl-db + dl-db = <'/db/'> #'\\d+' <'/'> ( dl-native | dl-schemas )? + dl-native = <'native/'> + dl-schemas = <'schema/'> dl-schema? + dl-schema = #'[^/]*' <'/'> dl-table? + dl-table = <'table/'> #'\\d+' <'/'> + collection = <'/collection/'> #'[^/]*' <'/'> ('read' <'/'>)? block = <'/block/db/'> #'\\d+' <'/'>") @@ -33,34 +41,56 @@ [id] (if (= id "root") :root (Long/parseUnsignedLong id))) +(defn- append-to-all + "If `path-or-paths` is a single path, append `x` to the end of it. If it's a vector of paths, append `x` to each path." + [path-or-paths x] + (if (seqable? (first path-or-paths)) + (map (fn [path] (append-to-all path x)) (seq path-or-paths)) + (into path-or-paths [x]))) + (defn- path "Recursively build permission path from parse tree" [tree] (match/match tree - (_ :guard insta/failure?) (log/error (trs "Error parsing permissions tree {0}" (pr-str tree))) - [:permission t] (path t) - [:all] [:all] ; admin permissions - [:db db-id] (let [db-id (Long/parseUnsignedLong db-id)] - [[:db db-id :data :native :write] - [:db db-id :data :schemas :all]]) - [:db db-id db-node] (let [db-id (Long/parseUnsignedLong db-id)] - (into [:db db-id] (path db-node))) - [:schemas] [:data :schemas :all] - [:schemas schema] (into [:data :schemas] (path schema)) - [:schema schema-name] [schema-name :all] - [:schema schema-name table] (into [schema-name] (path table)) - [:table table-id] [(Long/parseUnsignedLong table-id) :all] - [:table table-id table-perm] (into [(Long/parseUnsignedLong table-id)] (path table-perm)) - [:table-perm perm] (case perm - "read" [:read :all] - "query" [:query :all] - "query/segmented" [:query :segmented]) - [:native] [:data :native :write] + (_ :guard insta/failure?) (log/error (trs "Error parsing permissions tree {0}" (pr-str tree))) + [:permission t] (path t) + [:all] [:all] ; admin permissions + [:db db-id] (let [db-id (Long/parseUnsignedLong db-id)] + [[:db db-id :data :native :write] + [:db db-id :data :schemas :all]]) + [:db db-id db-node] (let [db-id (Long/parseUnsignedLong db-id)] + (into [:db db-id] (path db-node))) + [:schemas] [:data :schemas :all] + [:schemas schema] (into [:data :schemas] (path schema)) + [:schema schema-name] [schema-name :all] + [:schema schema-name table] (into [schema-name] (path table)) + [:table table-id] [(Long/parseUnsignedLong table-id) :all] + [:table table-id table-perm] (into [(Long/parseUnsignedLong table-id)] (path table-perm)) + [:table-perm perm] (case perm + "read" [:read :all] + "query" [:query :all] + "query/segmented" [:query :segmented]) + [:native] [:data :native :write] + ;; download perms + [:download + [:dl-limited db-node]] (append-to-all (path db-node) :limited) + [:download db-node] (append-to-all (path db-node) :full) + [:dl-db db-id] (let [db-id (Long/parseUnsignedLong db-id)] + #{[:db db-id :download :native] + [:db db-id :download :schemas]}) + [:dl-db db-id db-node] (let [db-id (Long/parseUnsignedLong db-id)] + (into [:db db-id] (path db-node))) + [:dl-schemas] [:download :schemas] + [:dl-schemas schema] (into [:download :schemas] (path schema)) + [:dl-schema schema-name] [schema-name] + [:dl-schema schema-name table] (into [schema-name] (path table)) + [:dl-table table-id] [(Long/parseUnsignedLong table-id)] + [:dl-native] [:download :native] ;; collection perms - [:collection id] [:collection (collection-id id) :write] - [:collection id "read"] [:collection (collection-id id) :read] + [:collection id] [:collection (collection-id id) :write] + [:collection id "read"] [:collection (collection-id id) :read] ;; block perms. Parse something like /block/db/1/ to {:db {1 {:schemas :block}}} - [:block db-id] [:db (Long/parseUnsignedLong db-id) :data :schemas :block])) + [:block db-id] [:db (Long/parseUnsignedLong db-id) :data :schemas :block])) (defn- graph "Given a set of permission paths, return a graph that expresses the most permissions possible for the set @@ -95,7 +125,7 @@ (walk/prewalk (fn [x] (or (when (map? x) (some #(and (= (% x) '()) %) - [:block :all :some :write :read :segmented])) + [:block :all :some :write :read :segmented :full :limited])) x))))) (defn permissions->graph diff --git a/test/metabase/models/permissions/parse_test.clj b/test/metabase/models/permissions/parse_test.clj index 13c3203151cb468dde544aa74e2920d4380a9618..fcc17d86b7af85c0e1085ef074c4737b4c78b2c3 100644 --- a/test/metabase/models/permissions/parse_test.clj +++ b/test/metabase/models/permissions/parse_test.clj @@ -5,15 +5,27 @@ (deftest permissions->graph (testing "Parses each permission string to the correct graph" (are [x y] (= y (parse/permissions->graph [x])) - "/db/3/" {:db {3 {:data {:native :write - :schemas :all}}}} - "/db/3/native/" {:db {3 {:data {:native :write}}}} - "/db/3/schema/" {:db {3 {:data {:schemas :all}}}} - "/db/3/schema/PUBLIC/" {:db {3 {:data {:schemas {"PUBLIC" :all}}}}} - "/db/3/schema/PUBLIC/table/4/" {:db {3 {:data {:schemas {"PUBLIC" {4 :all}}}}}} - "/db/3/schema/PUBLIC/table/4/read/" {:db {3 {:data {:schemas {"PUBLIC" {4 {:read :all}}}}}}} - "/db/3/schema/PUBLIC/table/4/query/" {:db {3 {:data {:schemas {"PUBLIC" {4 {:query :all}}}}}}} - "/db/3/schema/PUBLIC/table/4/query/segmented/" {:db {3 {:data {:schemas {"PUBLIC" {4 {:query :segmented}}}}}}}))) + "/db/3/" {:db {3 {:data {:native :write + :schemas :all}}}} + "/db/3/native/" {:db {3 {:data {:native :write}}}} + "/db/3/schema/" {:db {3 {:data {:schemas :all}}}} + "/db/3/schema/PUBLIC/" {:db {3 {:data {:schemas {"PUBLIC" :all}}}}} + "/db/3/schema/PUBLIC/table/4/" {:db {3 {:data {:schemas {"PUBLIC" {4 :all}}}}}} + "/db/3/schema/PUBLIC/table/4/read/" {:db {3 {:data {:schemas {"PUBLIC" {4 {:read :all}}}}}}} + "/db/3/schema/PUBLIC/table/4/query/" {:db {3 {:data {:schemas {"PUBLIC" {4 {:query :all}}}}}}} + "/db/3/schema/PUBLIC/table/4/query/segmented/" {:db {3 {:data {:schemas {"PUBLIC" {4 {:query :segmented}}}}}}} + "/download/db/3/" {:db {3 {:download {:native :full + :schemas :full}}}} + "/download/limited/db/3/" {:db {3 {:download {:native :limited + :schemas :limited}}}} + "/download/db/3/native/" {:db {3 {:download {:native :full}}}} + "/download/limited/db/3/native/" {:db {3 {:download {:native :limited}}}} + "/download/db/3/schema/" {:db {3 {:download {:schemas :full}}}} + "/download/limited/db/3/schema/" {:db {3 {:download {:schemas :limited}}}} + "/download/db/3/schema/PUBLIC/" {:db {3 {:download {:schemas {"PUBLIC" :full}}}}} + "/download/limited/db/3/schema/PUBLIC/" {:db {3 {:download {:schemas {"PUBLIC" :limited}}}}} + "/download/db/3/schema/PUBLIC/table/4/" {:db {3 {:download {:schemas {"PUBLIC" {4 :full}}}}}} + "/download/limited/db/3/schema/PUBLIC/table/4/" {:db {3 {:download {:schemas {"PUBLIC" {4 :limited}}}}}}))) (deftest combines-permissions-for-graph diff --git a/test/metabase/models/permissions_test.clj b/test/metabase/models/permissions_test.clj index e1d55699ff538549c3956ca11a19f360d331b1b9..f9ae2ca6b2cf30488cc0ac80ed352444f7b8bc53 100644 --- a/test/metabase/models/permissions_test.clj +++ b/test/metabase/models/permissions_test.clj @@ -33,6 +33,15 @@ "/db/1/schema/1234/table/1/" "/db/1/schema/PUBLIC/table/1/query/" "/db/1/schema/PUBLIC/table/1/query/segmented/" + ;; download permissions + "/download/db/1/" + "/download/limited/db/1/" + "/download/db/1/native/" + "/download/limited/db/1/native/" + "/download/db/1/schema/PUBLIC/" + "/download/limited/db/1/schema/PUBLIC/" + "/download/db/1/schema/PUBLIC/table/1/" + "/download/limited/db/1/schema/PUBLIC/table/1/" ;; block permissions "/block/db/1/" "/block/db/1000/" @@ -43,7 +52,7 @@ (perms/valid-path? path))))) (testing "\nWe should allow slashes in permissions paths? (#8693, #13263)\n" - (doseq [path [ ;; COMPANY-NET\ should get escaped to COMPANY-NET\\ + (doseq [path [;; COMPANY-NET\ should get escaped to COMPANY-NET\\ "/db/16/schema/COMPANY-NET\\\\john.doe/" ;; COMPANY-NET/ should get escaped to COMPANY-NET\/ "/db/16/schema/COMPANY-NET\\/john.doe/" @@ -156,7 +165,14 @@ "/block/db/1/schema/PUBLIC/" "/block/db/1/schema/PUBLIC/table/" "/block/db/1/schema/PUBLIC/table/2/" - "/block/collection/1/"]}] + "/block/collection/1/"] + + "invalid download permissions" + ["/download/" + "/download/limited/" + "/download/db/1/schema/PUBLIC/table/1/query/" + "/download/db/1/schema/PUBLIC/table/1/query/segmented/"]}] + (testing reason (doseq [path paths] (testing (str "\n" (pr-str path))