@atomist/sdm-pack-aspect
Version:
an Atomist SDM Extension Pack for visualizing drift across an organization
200 lines (192 loc) • 9.08 kB
text/typescript
/*
* Copyright © 2019 Atomist, Inc.
*
* 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.
*/
import { logger } from "@atomist/automation-client";
import * as camelcaseKeys from "camelcase-keys";
import { PlantedTree } from "../../../tree/sunburst";
import { validatePlantedTree } from "../../../tree/treeUtils";
import {
ClientFactory,
doWithClient,
} from "./pgUtils";
import { TreeQuery } from "./ProjectAnalysisResultStore";
/**
* Return results for non-matching fingerprints
*/
function nonMatchingRepos(tq: TreeQuery): string {
return `SELECT null as id, $4 as name, null as sha, null as data, $1 as type,
(
SELECT json_agg(row_to_json(repo))
FROM (
SELECT
repo_snapshots.id, repo_snapshots.owner, repo_snapshots.name, repo_snapshots.url, 1 as size
FROM repo_snapshots
WHERE workspace_id ${tq.workspaceId === "*" ? "<>" : "="} $1
AND repo_snapshots.id not in (select repo_fingerprints.repo_snapshot_id
FROM repo_fingerprints WHERE repo_fingerprints.fingerprint_id in
(SELECT id from fingerprints where fingerprints.feature_name = $2
AND fingerprints.name ${tq.byName ? "=" : "<>"} $3))
) repo
)
children`;
}
function fingerprintsToReposQuery(tq: TreeQuery): string {
// We always select by aspect (aka feature_name, aka type), and sometimes also by fingerprint name.
const sql = `
SELECT row_to_json(fingerprint_groups) FROM (
SELECT json_agg(fp) as children FROM (
SELECT
fingerprints.id as id, fingerprints.name as name, fingerprints.sha as sha,
fingerprints.data as data, fingerprints.feature_name as type,
fingerprints.display_name, fingerprints.display_value,
(
SELECT json_agg(row_to_json(repo)) FROM (
SELECT
repo_snapshots.id, repo_snapshots.owner, repo_snapshots.name, repo_snapshots.url, repo_snapshots.provider_id, 1 as size, repo_fingerprints.path
FROM repo_fingerprints, repo_snapshots
WHERE repo_fingerprints.fingerprint_id = fingerprints.id
AND repo_snapshots.id = repo_fingerprints.repo_snapshot_id
AND workspace_id ${tq.workspaceId === "*" ? "<>" : "="} $1
) repo
) as children FROM fingerprints
WHERE fingerprints.feature_name = $2 and fingerprints.name ${tq.byName ? "=" : "<>"} $3
${tq.otherLabel ? ("UNION ALL " + nonMatchingRepos(tq)) : ""}
) fp WHERE children is not NULL) as fingerprint_groups
`;
logger.debug("Running fingerprintsToRepos SQL\n%s", sql);
return sql;
}
/**
* Tree where children is one of a range of values, leaves individual repos with one of those values
*/
export async function fingerprintsToReposTreeQuery(tq: TreeQuery, clientFactory: ClientFactory): Promise<PlantedTree> {
const sql = fingerprintsToReposQuery(tq);
const children = await doWithClient(sql, clientFactory, async client => {
try {
const bindParams = [tq.workspaceId, tq.aspectName, tq.rootName];
if (tq.otherLabel) {
bindParams.push(tq.otherLabel);
}
const results = await client.query(sql, bindParams);
const data = results.rows[0];
return data.row_to_json.children;
} catch (err) {
logger.error("Error running SQL %s: %s", sql, err);
throw err;
}
}, []);
const result = {
tree: {
name: tq.rootName,
children,
},
circles: [
{ meaning: tq.byName ? "fingerprint name" : "aspect" },
{ meaning: "fingerprint value" },
{ meaning: "repo" },
],
};
validatePlantedTree(result);
return camelcaseKeys(result, { deep: true }) as any;
}
export async function driftTreeForAllAspects(workspaceId: string,
percentile: number,
clientFactory: ClientFactory): Promise<PlantedTree> {
const sql = driftTreeSql(workspaceId, { repos: false });
const circles = [
{ meaning: "report" },
{ meaning: "aspect name" },
{ meaning: "fingerprint name" },
];
return doWithClient(sql, clientFactory, async client => {
const result = await client.query(sql,
[workspaceId, percentile / 100]);
const tree: PlantedTree = {
circles,
tree: {
name: "drift",
children: result.rows.map(r => r.children),
},
};
return tree;
}, err => {
return {
circles,
tree: { name: "failed drift report", children: [] },
errors: [{ message: err.message }],
};
});
}
export async function driftTreeForSingleAspect(workspaceId: string,
percentile: number,
options: { repos?: boolean, type?: string },
clientFactory: ClientFactory): Promise<PlantedTree> {
const sql = driftTreeSql(workspaceId, options);
return doWithClient(sql, clientFactory, async client => {
const result = await client.query(sql,
[workspaceId, percentile / 100, options.type]);
const tree: PlantedTree = {
circles: !options.repos ? [
{ meaning: "type" },
{ meaning: "fingerprint entropy" },
] : [
{ meaning: "type" },
{ meaning: "fingerprint entropy" },
{ meaning: "repos" },
],
tree: {
name: options.type,
children: result.rows[0].children.children,
},
};
return tree;
});
}
function driftTreeSql(workspaceId: string, options: { repos?: boolean, type?: string }): string {
if (!options.repos) {
return `SELECT row_to_json(data) as children
FROM (SELECT f0.type as name, f0.type as type, json_agg(aspects) as children
FROM (SELECT distinct feature_name as type from fingerprint_analytics) f0, (
SELECT name, name as fingerprint_name, feature_name as type, variants, count, entropy, variants as size
FROM fingerprint_analytics f1
WHERE workspace_id ${workspaceId === "*" ? "<>" : "="} $1
AND entropy >=
(SELECT percentile_disc($2) within group (order by entropy)
FROM fingerprint_analytics
WHERE workspace_id ${workspaceId === "*" ? "<>" : "="} $1)
ORDER BY entropy DESC, fingerprint_name ASC) as aspects
WHERE aspects.type = f0.type ${options.type ? `AND aspects.type = $3` : ""}
GROUP by f0.type) as data`;
} else {
// tslint:disable:max-line-length
return `SELECT row_to_json(data) as children
FROM (SELECT f0.type as name, f0.type as type, json_agg(aspects) as children
FROM (SELECT distinct feature_name as type from fingerprint_analytics) f0, (
SELECT f1.name, f1.name as fingerprint_name, f1.feature_name as type, f1.variants, f1.count, f1.entropy, f1.variants as size, json_agg(repos) as children
FROM fingerprint_analytics f1, (SELECT distinct _rs1.url, _rs1.owner, _rs1.name, _rs1.url, _f1.feature_name as type, _f1.name as fingerprint_name, 1 as size
FROM repo_snapshots _rs1, repo_fingerprints _rf1, fingerprints _f1 WHERE _rs1.id = _rf1.repo_snapshot_id AND _rf1.fingerprint_id = _f1.id) as repos
WHERE workspace_id ${workspaceId === "*" ? "<>" : "="} $1
AND entropy >=
(SELECT percentile_disc($2) within group (order by entropy)
FROM fingerprint_analytics
WHERE workspace_id ${workspaceId === "*" ? "<>" : "="} $1)
AND repos.type = f1.feature_name AND repos.fingerprint_name = f1.name
GROUP BY f1.name, f1.feature_name, f1.variants, f1.count, f1.entropy
ORDER BY entropy DESC, fingerprint_name ASC) as aspects
WHERE aspects.type = f0.type ${options.type ? `AND aspects.type = $3` : ""}
GROUP by f0.type) as data`;
// tslint:enable:max-line-length
}
}