2016-01-19 20:16:45 +00:00
|
|
|
// 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.
|
|
|
|
|
2015-12-28 20:03:29 +00:00
|
|
|
package pgsql
|
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
import "strconv"
|
2015-12-28 20:03:29 +00:00
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
const (
|
|
|
|
lockVulnerabilityAffects = `LOCK Vulnerability_Affects_FeatureVersion IN SHARE ROW EXCLUSIVE MODE`
|
|
|
|
disableHashJoin = `SET LOCAL enable_hashjoin = off`
|
|
|
|
disableMergeJoin = `SET LOCAL enable_mergejoin = off`
|
2016-01-12 15:40:46 +00:00
|
|
|
|
2015-12-28 20:03:29 +00:00
|
|
|
// keyvalue.go
|
2016-02-23 00:15:38 +00:00
|
|
|
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`
|
2015-12-28 20:03:29 +00:00
|
|
|
|
|
|
|
// namespace.go
|
2016-02-23 00:15:38 +00:00
|
|
|
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`
|
|
|
|
|
2016-03-08 02:24:29 +00:00
|
|
|
searchNamespace = `SELECT id FROM Namespace WHERE name = $1`
|
|
|
|
listNamespace = `SELECT id, name FROM Namespace`
|
2016-01-28 16:26:34 +00:00
|
|
|
|
2015-12-28 20:03:29 +00:00
|
|
|
// feature.go
|
2016-02-23 00:15:38 +00:00
|
|
|
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`
|
|
|
|
|
2016-03-03 19:15:06 +00:00
|
|
|
searchFeatureVersion = `
|
|
|
|
SELECT id FROM FeatureVersion WHERE feature_id = $1 AND version = $2`
|
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
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
|
|
|
|
)
|
2016-11-12 14:42:59 +00:00
|
|
|
SELECT false, id FROM FeatureVersion WHERE feature_id = $1 AND version = $2
|
2016-02-23 00:15:38 +00:00
|
|
|
UNION
|
2016-11-12 14:42:59 +00:00
|
|
|
SELECT true, id FROM new_featureversion`
|
2016-02-23 00:15:38 +00:00
|
|
|
|
|
|
|
searchVulnerabilityFixedInFeature = `
|
|
|
|
SELECT id, vulnerability_id, version FROM Vulnerability_FixedIn_Feature
|
2016-01-08 15:27:30 +00:00
|
|
|
WHERE feature_id = $1`
|
2015-12-28 20:03:29 +00:00
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
insertVulnerabilityAffectsFeatureVersion = `
|
|
|
|
INSERT INTO Vulnerability_Affects_FeatureVersion(vulnerability_id,
|
2015-12-28 20:03:29 +00:00
|
|
|
featureversion_id, fixedin_id) VALUES($1, $2, $3)`
|
|
|
|
|
|
|
|
// layer.go
|
2016-02-23 00:15:38 +00:00
|
|
|
searchLayer = `
|
|
|
|
SELECT l.id, l.name, l.engineversion, p.id, p.name, n.id, n.name
|
|
|
|
FROM Layer l
|
|
|
|
LEFT JOIN Layer p ON l.parent_id = p.id
|
|
|
|
LEFT JOIN Namespace n ON l.namespace_id = n.id
|
|
|
|
WHERE l.name = $1;`
|
|
|
|
|
|
|
|
searchLayerFeatureVersion = `
|
2016-02-18 16:04:16 +00:00
|
|
|
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`
|
2015-12-28 20:03:29 +00:00
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
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`
|
|
|
|
|
|
|
|
insertLayer = `
|
|
|
|
INSERT INTO Layer(name, engineversion, parent_id, namespace_id, created_at)
|
2016-02-04 16:37:04 +00:00
|
|
|
VALUES($1, $2, $3, $4, CURRENT_TIMESTAMP)
|
|
|
|
RETURNING id`
|
2016-01-08 15:27:30 +00:00
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
updateLayer = `UPDATE LAYER SET engineversion = $2, namespace_id = $3 WHERE id = $1`
|
2016-01-08 15:27:30 +00:00
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
removeLayerDiffFeatureVersion = `
|
|
|
|
DELETE FROM Layer_diff_FeatureVersion
|
|
|
|
WHERE layer_id = $1`
|
2015-12-28 20:03:29 +00:00
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
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[])`
|
2016-01-08 19:42:07 +00:00
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
removeLayer = `DELETE FROM Layer WHERE name = $1`
|
2016-01-20 18:57:51 +00:00
|
|
|
|
2016-01-08 19:42:07 +00:00
|
|
|
// lock.go
|
2016-02-23 00:15:38 +00:00
|
|
|
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`
|
2016-01-12 15:40:46 +00:00
|
|
|
|
|
|
|
// vulnerability.go
|
2016-02-23 00:15:38 +00:00
|
|
|
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`
|
2016-02-26 10:18:45 +00:00
|
|
|
searchVulnerabilityByNamespace = ` WHERE n.name = $1 AND v.deleted_at IS NULL
|
2016-02-29 08:29:40 +00:00
|
|
|
AND v.id >= $2
|
|
|
|
ORDER BY v.id
|
|
|
|
LIMIT $3`
|
2016-02-23 00:15:38 +00:00
|
|
|
|
|
|
|
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`
|
|
|
|
|
2016-11-11 18:16:40 +00:00
|
|
|
soiVulnerabilityFixedInFeature = `
|
|
|
|
WITH new_fixedinfeature AS (
|
|
|
|
INSERT INTO Vulnerability_FixedIn_Feature(vulnerability_id, feature_id, version)
|
|
|
|
SELECT CAST($1 AS INTEGER), CAST($2 AS INTEGER), CAST($3 AS VARCHAR)
|
|
|
|
WHERE NOT EXISTS (SELECT id FROM Vulnerability_FixedIn_Feature WHERE vulnerability_id = $1 AND feature_id = $2)
|
|
|
|
RETURNING id
|
|
|
|
)
|
2016-11-12 14:42:59 +00:00
|
|
|
SELECT false, id FROM Vulnerability_FixedIn_Feature WHERE vulnerability_id = $1 AND feature_id = $2
|
2016-11-11 18:16:40 +00:00
|
|
|
UNION
|
2016-11-12 14:42:59 +00:00
|
|
|
SELECT true, id FROM new_fixedinfeature`
|
2016-02-23 00:15:38 +00:00
|
|
|
|
|
|
|
searchFeatureVersionByFeature = `SELECT id, version FROM FeatureVersion WHERE feature_id = $1`
|
|
|
|
|
|
|
|
removeVulnerability = `
|
|
|
|
UPDATE Vulnerability
|
2016-02-04 16:34:01 +00:00
|
|
|
SET deleted_at = CURRENT_TIMESTAMP
|
2016-01-28 18:35:07 +00:00
|
|
|
WHERE namespace_id = (SELECT id FROM Namespace WHERE name = $1)
|
2016-02-04 16:34:01 +00:00
|
|
|
AND name = $2
|
|
|
|
AND deleted_at IS NULL
|
|
|
|
RETURNING id`
|
2016-01-28 18:35:07 +00:00
|
|
|
|
2016-01-21 23:09:23 +00:00
|
|
|
// notification.go
|
2016-02-23 00:15:38 +00:00
|
|
|
insertNotification = `
|
|
|
|
INSERT INTO Vulnerability_Notification(name, created_at, old_vulnerability_id, new_vulnerability_id)
|
2016-01-22 20:59:46 +00:00
|
|
|
VALUES($1, CURRENT_TIMESTAMP, $2, $3)`
|
2016-01-21 23:09:23 +00:00
|
|
|
|
2016-02-23 00:15:38 +00:00
|
|
|
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 = `
|
2016-12-04 11:54:09 +00:00
|
|
|
WITH LDFV AS (
|
|
|
|
SELECT ldfv.layer_id
|
|
|
|
FROM Vulnerability_Affects_FeatureVersion vafv, FeatureVersion fv, Layer_diff_FeatureVersion ldfv
|
|
|
|
WHERE ldfv.layer_id >= $2
|
|
|
|
AND vafv.vulnerability_id = $1
|
|
|
|
AND vafv.featureversion_id = fv.id
|
|
|
|
AND ldfv.featureversion_id = fv.id
|
|
|
|
AND ldfv.modification = 'add'
|
|
|
|
ORDER BY ldfv.ID
|
|
|
|
)
|
|
|
|
SELECT l.id, l.name
|
|
|
|
FROM LDFV, Layer l
|
|
|
|
WHERE LDFV.layer_id = l.id
|
|
|
|
LIMIT $3`
|
|
|
|
|
2016-01-18 23:52:16 +00:00
|
|
|
// complex_test.go
|
2016-02-23 00:15:38 +00:00
|
|
|
searchComplexTestFeatureVersionAffects = `
|
|
|
|
SELECT v.name
|
2016-01-18 23:52:16 +00:00
|
|
|
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`
|
2016-02-23 00:15:38 +00:00
|
|
|
)
|
2016-01-08 15:27:30 +00:00
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|