// Copyright 2015 clair authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package pgsql import "strconv" const ( lockVulnerabilityAffects = `LOCK Vulnerability_Affects_FeatureVersion IN SHARE ROW EXCLUSIVE MODE` disableHashJoin = `SET LOCAL enable_hashjoin = off` disableMergeJoin = `SET LOCAL enable_mergejoin = off` // keyvalue.go updateKeyValue = `UPDATE KeyValue SET value = $1 WHERE key = $2` insertKeyValue = `INSERT INTO KeyValue(key, value) VALUES($1, $2)` searchKeyValue = `SELECT value FROM KeyValue WHERE key = $1` // namespace.go soiNamespace = ` WITH new_namespace AS ( INSERT INTO Namespace(name) SELECT CAST($1 AS VARCHAR) WHERE NOT EXISTS (SELECT name FROM Namespace WHERE name = $1) RETURNING id ) SELECT id FROM Namespace WHERE name = $1 UNION SELECT id FROM new_namespace` searchNamespace = `SELECT id FROM Namespace WHERE name = $1` listNamespace = `SELECT id, name FROM Namespace` // feature.go soiFeature = ` WITH new_feature AS ( INSERT INTO Feature(name, namespace_id) SELECT CAST($1 AS VARCHAR), CAST($2 AS INTEGER) WHERE NOT EXISTS (SELECT id FROM Feature WHERE name = $1 AND namespace_id = $2) RETURNING id ) SELECT id FROM Feature WHERE name = $1 AND namespace_id = $2 UNION SELECT id FROM new_feature` searchFeatureVersion = ` SELECT id FROM FeatureVersion WHERE feature_id = $1 AND version = $2` soiFeatureVersion = ` WITH new_featureversion AS ( INSERT INTO FeatureVersion(feature_id, version) SELECT CAST($1 AS INTEGER), CAST($2 AS VARCHAR) WHERE NOT EXISTS (SELECT id FROM FeatureVersion WHERE feature_id = $1 AND version = $2) RETURNING id ) SELECT 'exi', id FROM FeatureVersion WHERE feature_id = $1 AND version = $2 UNION SELECT 'new', id FROM new_featureversion` searchVulnerabilityFixedInFeature = ` SELECT id, vulnerability_id, version FROM Vulnerability_FixedIn_Feature WHERE feature_id = $1` insertVulnerabilityAffectsFeatureVersion = ` INSERT INTO Vulnerability_Affects_FeatureVersion(vulnerability_id, featureversion_id, fixedin_id) VALUES($1, $2, $3)` // layer.go searchLayer = ` SELECT l.id, l.name, l.engineversion, p.id, p.name FROM Layer l LEFT JOIN Layer p ON l.parent_id = p.id WHERE l.name = $1;` searchLayerFeatureVersion = ` WITH RECURSIVE layer_tree(id, name, parent_id, depth, path, cycle) AS( SELECT l.id, l.name, l.parent_id, 1, ARRAY[l.id], false FROM Layer l WHERE l.id = $1 UNION ALL SELECT l.id, l.name, l.parent_id, lt.depth + 1, path || l.id, l.id = ANY(path) FROM Layer l, layer_tree lt WHERE l.id = lt.parent_id ) SELECT ldf.featureversion_id, ldf.modification, fn.id, fn.name, f.id, f.name, fv.id, fv.version, ltree.id, ltree.name FROM Layer_diff_FeatureVersion ldf JOIN ( SELECT row_number() over (ORDER BY depth DESC), id, name FROM layer_tree ) AS ltree (ordering, id, name) ON ldf.layer_id = ltree.id, FeatureVersion fv, Feature f, Namespace fn WHERE ldf.featureversion_id = fv.id AND fv.feature_id = f.id AND f.namespace_id = fn.id ORDER BY ltree.ordering` searchFeatureVersionVulnerability = ` SELECT vafv.featureversion_id, v.id, v.name, v.description, v.link, v.severity, v.metadata, vn.name, vfif.version FROM Vulnerability_Affects_FeatureVersion vafv, Vulnerability v, Namespace vn, Vulnerability_FixedIn_Feature vfif WHERE vafv.featureversion_id = ANY($1::integer[]) AND vfif.vulnerability_id = v.id AND vafv.fixedin_id = vfif.id AND v.namespace_id = vn.id AND v.deleted_at IS NULL` searchLayerNamespace = ` SELECT n.id, n.name From LayerNamespace ln, Namespace n WHERE ln.layer_id = $1 AND ln.namespace_id = n.id` insertLayer = ` INSERT INTO Layer(name, engineversion, parent_id, created_at) VALUES($1, $2, $3, CURRENT_TIMESTAMP) RETURNING id` updateLayer = `UPDATE LAYER SET engineversion = $2 WHERE id = $1` removeLayerDiffFeatureVersion = ` DELETE FROM Layer_diff_FeatureVersion WHERE layer_id = $1` insertLayerDiffFeatureVersion = ` INSERT INTO Layer_diff_FeatureVersion(layer_id, featureversion_id, modification) SELECT $1, fv.id, $2 FROM FeatureVersion fv WHERE fv.id = ANY($3::integer[])` removeLayer = `DELETE FROM Layer WHERE name = $1` removeLayerNamespace = ` DELETE FROM LayerNamespace WHERE layer_id = $1` soiLayerNamespace = ` WITH new_layernamespace AS ( INSERT INTO LayerNamespace(layer_id, namespace_id) SELECT CAST($1 AS INTEGER), CAST($2 AS INTEGER) WHERE NOT EXISTS (SELECT id FROM LayerNamespace WHERE layer_id = $1 AND namespace_id = $2) RETURNING id ) SELECT id FROM LayerNamespace WHERE layer_id = $1 AND namespace_id = $2 UNION SELECT id FROM new_layernamespace` // lock.go insertLock = `INSERT INTO Lock(name, owner, until) VALUES($1, $2, $3)` searchLock = `SELECT owner, until FROM Lock WHERE name = $1` updateLock = `UPDATE Lock SET until = $3 WHERE name = $1 AND owner = $2` removeLock = `DELETE FROM Lock WHERE name = $1 AND owner = $2` removeLockExpired = `DELETE FROM LOCK WHERE until < CURRENT_TIMESTAMP` // vulnerability.go searchVulnerabilityBase = ` SELECT v.id, v.name, n.id, n.name, v.description, v.link, v.severity, v.metadata FROM Vulnerability v JOIN Namespace n ON v.namespace_id = n.id` searchVulnerabilityForUpdate = ` FOR UPDATE OF v` searchVulnerabilityByNamespaceAndName = ` WHERE n.name = $1 AND v.name = $2 AND v.deleted_at IS NULL` searchVulnerabilityByID = ` WHERE v.id = $1` searchVulnerabilityByNamespace = ` WHERE n.name = $1 AND v.deleted_at IS NULL AND v.id >= $2 ORDER BY v.id LIMIT $3` searchVulnerabilityFixedIn = ` SELECT vfif.version, f.id, f.Name FROM Vulnerability_FixedIn_Feature vfif JOIN Feature f ON vfif.feature_id = f.id WHERE vfif.vulnerability_id = $1` insertVulnerability = ` INSERT INTO Vulnerability(namespace_id, name, description, link, severity, metadata, created_at) VALUES($1, $2, $3, $4, $5, $6, CURRENT_TIMESTAMP) RETURNING id` insertVulnerabilityFixedInFeature = ` INSERT INTO Vulnerability_FixedIn_Feature(vulnerability_id, feature_id, version) VALUES($1, $2, $3) RETURNING id` searchFeatureVersionByFeature = `SELECT id, version FROM FeatureVersion WHERE feature_id = $1` removeVulnerability = ` UPDATE Vulnerability SET deleted_at = CURRENT_TIMESTAMP WHERE namespace_id = (SELECT id FROM Namespace WHERE name = $1) AND name = $2 AND deleted_at IS NULL RETURNING id` // notification.go insertNotification = ` INSERT INTO Vulnerability_Notification(name, created_at, old_vulnerability_id, new_vulnerability_id) VALUES($1, CURRENT_TIMESTAMP, $2, $3)` updatedNotificationNotified = ` UPDATE Vulnerability_Notification SET notified_at = CURRENT_TIMESTAMP WHERE name = $1` removeNotification = ` UPDATE Vulnerability_Notification SET deleted_at = CURRENT_TIMESTAMP WHERE name = $1` searchNotificationAvailable = ` SELECT id, name, created_at, notified_at, deleted_at FROM Vulnerability_Notification WHERE (notified_at IS NULL OR notified_at < $1) AND deleted_at IS NULL AND name NOT IN (SELECT name FROM Lock) ORDER BY Random() LIMIT 1` searchNotification = ` SELECT id, name, created_at, notified_at, deleted_at, old_vulnerability_id, new_vulnerability_id FROM Vulnerability_Notification WHERE name = $1` searchNotificationLayerIntroducingVulnerability = ` SELECT l.ID, l.name FROM Vulnerability v, Vulnerability_Affects_FeatureVersion vafv, FeatureVersion fv, Layer_diff_FeatureVersion ldfv, Layer l WHERE v.id = $1 AND v.id = vafv.vulnerability_id AND vafv.featureversion_id = fv.id AND fv.id = ldfv.featureversion_id AND ldfv.modification = 'add' AND ldfv.layer_id = l.id AND l.id >= $2 ORDER BY l.ID LIMIT $3` // complex_test.go searchComplexTestFeatureVersionAffects = ` SELECT v.name FROM FeatureVersion fv LEFT JOIN Vulnerability_Affects_FeatureVersion vaf ON fv.id = vaf.featureversion_id JOIN Vulnerability v ON vaf.vulnerability_id = v.id WHERE featureversion_id = $1` ) // buildInputArray constructs a PostgreSQL input array from the specified integers. // Useful to use the `= ANY($1::integer[])` syntax that let us use a IN clause while using // a single placeholder. func buildInputArray(ints []int) string { str := "{" for i := 0; i < len(ints)-1; i++ { str = str + strconv.Itoa(ints[i]) + "," } str = str + strconv.Itoa(ints[len(ints)-1]) + "}" return str }