Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: New Frontend Dashboard #155

Merged
merged 24 commits into from
Jun 13, 2019
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
12296d2
feat: add @best/frontend and @best/api-db packages
Jun 7, 2019
9abce33
chore: cleaning up
Jun 7, 2019
aec5e26
chore: update version numbers to match the rest of best
Jun 7, 2019
20a6df8
wip: begin work to integrate api-db into the cli
Jun 8, 2019
fabeaa6
chore: remove eslint in api-db
Jun 10, 2019
a5d6d83
feat: initial implementation of api-db into cli
Jun 10, 2019
ec96612
fix: wrong boolean for temporary on snapshot
Jun 10, 2019
4af65bd
feat: add ability to configure api-db by best config file
Jun 10, 2019
05d493e
chore: remove testing username from best.config.js
Jun 10, 2019
f59cf6a
fix: metrics not get properly normalized for the database
Jun 10, 2019
7eee3e4
fix: get commit date directly from git
Jun 11, 2019
c7e1514
feat: you can now select commits when viewing a benchmark
Jun 11, 2019
2cab84a
wip: begin implementation of selecting commits to show more info
Jun 11, 2019
a3e83ba
wip: designing commit info popover
Jun 11, 2019
2d92caf
chore: clean up some old comments/todos
Jun 11, 2019
0f42aa9
fix: clean up graphs with better ticks
Jun 12, 2019
384a937
fix: improve the format that view.zoom is being stored in urlstorage
Jun 12, 2019
c4649b8
fix: changing selected metric now re-renders benchmarks
Jun 12, 2019
0c2a55a
feat: commit info now shows up in the correct position
Jun 12, 2019
15b2e74
fix: close button for commit info
Jun 12, 2019
0a68e19
feat: refactor benchmarks and create new graph component
Jun 12, 2019
e56a031
fix: relayout handler now always get attached
Jun 12, 2019
23baca6
fix: graphs now get properly resized when window resizes
Jun 13, 2019
12a4aca
fix: non-temporary snapshots now must be unique
Jun 13, 2019
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions .eslintrc
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,14 @@
"env": {
"browser": true
}
},
{
"files": [
"**/@best/api-db/**/migrations/**"
],
"rules": {
"@typescript-eslint/camelcase": "off"
}
}
]
}
5 changes: 3 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
"@commitlint/cli": "^8.0.0",
"@commitlint/config-conventional": "^8.0.0",
"@types/express": "^4.17.0",
"@types/globby": "^9.1.0",
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Where this comes from?

Copy link
Contributor Author

@jasonsilberman jasonsilberman Jun 10, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

My build was failing because TypeScript couldn't find the types for the globby module which is used in @best/cli and @best/store-fs. I would happily remove it if TypeScript would build without it.

"@types/jest": "^24.0.13",
"@types/json2md": "^1.5.0",
"@types/micromatch": "^3.1.0",
Expand All @@ -51,8 +52,8 @@
"lerna": "^3.14.1",
"prettier": "^1.17.1",
"rimraf": "^2.6.3",
"typescript": "^3.5.1",
"ts-jest": "^24.0.2"
"ts-jest": "^24.0.2",
"typescript": "^3.5.1"
},
"config": {
"commitizen": {
Expand Down
13 changes: 13 additions & 0 deletions packages/@best/api-db/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
# API DB

This is the database adapter that the frontend uses to display results. The results are stored whenever a benchmark is run.

There is an associated Postgres db which is the only type of database currently supported. In the future we could add more supported databases.

## Migrations

In order to run the migrations required for the database you can run the following command:

```
yarn migrate:postgres up
```
16 changes: 16 additions & 0 deletions packages/@best/api-db/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
{
"name": "@best/api-db",
"version": "0.7.1",
"author": "Jason Silberman",
"dependencies": {
"node-pg-migrate": "^3.21.1",
"pg": "^7.11.0"
},
"devDependencies": {
"@types/pg": "^7.4.14"
},
"main": "build/index.js",
"scripts": {
"migrate:postgres": "node-pg-migrate -m src/postgres/migrations"
}
}
3 changes: 3 additions & 0 deletions packages/@best/api-db/src/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
export { saveBenchmarkSummaryInDB } from './store';

export { loadDbFromConfig } from './utils';
51 changes: 51 additions & 0 deletions packages/@best/api-db/src/postgres/db.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
import { Pool, QueryResult } from 'pg'
import { TemporarySnapshot } from '../types'

const normalizeMetrics = (metrics: any) => {
const standardizedMetrics = metrics.reduce((acc: any, metric: any) => {
return {
...acc,
[metric.name]: [metric.duration, metric.stdDeviation]
}
}, {})

return JSON.stringify(standardizedMetrics);
}

export default class DB {
pool: Pool
constructor(config: any) {
this.pool = new Pool(config)
}

// eslint-disable-next-line @typescript-eslint/no-explicit-any
query(text: string, params: any[]): Promise<QueryResult> {
console.warn('[DB] Directly using db.query is discouraged.')
return this.pool.query(text, params)
}

fetchProjects(): Promise<QueryResult> {
return this.pool.query('SELECT * FROM projects')
}

fetchSnapshots(projectId: number, since: string): Promise<QueryResult> {
if (since) {
return this.pool.query(`SELECT * FROM snapshots WHERE "project_id" = $1 AND "temporary" = 'f' AND "commit_date" > $2 ORDER BY commit_date, name`, [projectId, since])
}

return this.pool.query(`SELECT * FROM snapshots WHERE "project_id" = $1 AND "temporary" = 'f' ORDER BY commit_date, name`, [projectId])
}

fetchProject(name: string): Promise<QueryResult> {
return this.pool.query('SELECT * FROM projects WHERE "name" = $1 LIMIT 1', [name])
}

createProject(name: string): Promise<QueryResult> {
return this.pool.query('INSERT INTO projects(name) VALUES ($1) RETURNING *', [name]);
}

createSnapshot(snapshot: TemporarySnapshot, projectId: number): Promise<QueryResult> {
const values = [snapshot.name, normalizeMetrics(snapshot.metrics), snapshot.environmentHash, snapshot.similarityHash, snapshot.commit, snapshot.commitDate, snapshot.temporary, projectId];
return this.pool.query('INSERT INTO snapshots(name, metrics, environment_hash, similarity_hash, commit, commit_date, temporary, project_id) VALUES ($1, $2, $3, $4, $5, $6, $7, $8) RETURNING *', values)
}
}
46 changes: 46 additions & 0 deletions packages/@best/api-db/src/postgres/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
import { ApiDB, Project, TemporarySnapshot, Snapshot } from '../types';
import DB from './db';
import transformer from './transformer';

export default class PostgresDB extends ApiDB {
db: DB;
constructor(config: any) {
super(config)
this.db = new DB(config);
}

async fetchProjects(): Promise<Project[]> {
const results = await this.db.fetchProjects()

return transformer.projects(results)
}

async fetchSnapshots(projectId: number, since: string): Promise<Snapshot[]> {
const results = await this.db.fetchSnapshots(projectId, since)

return transformer.snapshots(results)
}

async saveSnapshots(snapshots: TemporarySnapshot[], projectName: string): Promise<boolean> {
const projectResult = await this.db.fetchProject(projectName);

let projectId: number;
if (projectResult.rows.length > 0) {
projectId = projectResult.rows[0].id;
} else {
const newProject = await this.db.createProject(projectName);
projectId = newProject.rows[0].id;
}

try {
await Promise.all(snapshots.map(async (snapshot) => {
return this.db.createSnapshot(snapshot, projectId);
}));
} catch (err) {
console.error('[API-DB] Could not save results', err);
return false;
}

return true;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
exports.shorthands = undefined;

exports.up = pgm => {
pgm.createTable('projects', {
id: 'id',
name: { type: 'varchar(100)', notNull: true },
created_at: {
type: 'timestamp',
notNull: true,
default: pgm.func('current_timestamp'),
},
});
};

exports.down = pgm => {
pgm.dropTable('projects');
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
exports.shorthands = undefined;

exports.up = pgm => {
pgm.createTable('snapshots', {
id: 'id',
project_id: {
type: 'integer',
notNull: true,
references: '"projects"',
onDelete: 'cascade',
},
name: { type: 'varchar(200)', notNull: true },
metrics: { type: 'varchar(2000)', notNull: true },
environment_hash: { type: 'varchar(100)', notNull: true },
similarity_hash: { type: 'varchar(100)', notNull: true },
commit: { type: 'varchar(100)', notNull: true },
commit_date: {
type: 'timestamp',
notNull: true,
},
created_at: {
type: 'timestamp',
notNull: true,
default: pgm.func('current_timestamp'),
},
temporary: 'boolean',
});

pgm.createIndex('snapshots', 'project_id');
};

exports.down = pgm => {
pgm.dropTable('snapshots');
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
exports.shorthands = undefined;

exports.up = pgm => {
pgm.addColumns('snapshots', {
updated_at: {
type: 'timestamp',
notNull: true,
default: pgm.func('current_timestamp'),
},
});
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
exports.shorthands = undefined;

exports.up = pgm => {
pgm.addColumns('projects', {
last_release_date: {
type: 'timestamp',
},
});
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
exports.shorthands = undefined;

exports.up = (pgm) => {
pgm.createIndex('snapshots', ['project_id', 'commit', 'name'], { unique: true, where: `temporary = 'f'` })
};
36 changes: 36 additions & 0 deletions packages/@best/api-db/src/postgres/transformer.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import { QueryResult } from 'pg';
import { Project, Snapshot, Metric } from '../types';

const normalizeMetrics = (metrics: any): Metric[] => {
return Object.keys(metrics).map((key): Metric => ({
name: key,
duration: metrics[key][0],
stdDeviation: metrics[key][1]
}));
};

export default {
projects: (query: QueryResult): Project[] => {
return query.rows.map((row): Project => ({
id: row.id,
name: row.name,
createdAt: row.created_at,
lastReleaseDate: row.last_release_date
}));
},
snapshots: (query: QueryResult): Snapshot[] => {
return query.rows.map((row): Snapshot => ({
id: row.id,
projectId: row.project_id,
name: row.name,
metrics: normalizeMetrics(JSON.parse(row.metrics)),
environmentHash: row.environment_hash,
similarityHash: row.similarity_hash,
commit: row.commit,
commitDate: row.commit_date,
temporary: row.temporary,
createdAt: row.created_at,
updatedAt: row.updated_at
}));
}
};
61 changes: 61 additions & 0 deletions packages/@best/api-db/src/store.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
import crypto from 'crypto';
import { loadDbFromConfig } from './utils';
import { TemporarySnapshot } from './types';

function md5(data: string) {
return crypto
.createHash('md5')
.update(data)
.digest('hex');
}

export const saveBenchmarkSummaryInDB = (benchmarkResults: any, globalConfig: any) => {
const db = loadDbFromConfig(globalConfig);

return Promise.all(
benchmarkResults.map(async (benchmarkResult: any) => {
const { benchmarkSignature, projectConfig, environment, stats } = benchmarkResult;
const { projectName } = projectConfig;
const { gitCommit, gitCommitDate, gitLocalChanges } = globalConfig;

const snapshotEnvironment = {
hardware: environment.hardware,
browser: environment.browser
}

const environmentHash = md5(JSON.stringify(snapshotEnvironment));

const runSettings = {
similarityHash: benchmarkSignature,
commit: gitCommit,
commitDate: gitCommitDate,
environmentHash,
// TODO: not sure if this is exactly what we want to determine here
temporary: gitLocalChanges
}

const snapshotsToSave: TemporarySnapshot[] = [];

stats.benchmarks.forEach((element: any) => {
element.benchmarks.forEach((bench: any) => {
const metricKeys = Object.keys(bench).filter(key => key !== 'name')
const metrics = metricKeys.map(name => ({
name,
duration: bench[name].median,
stdDeviation: bench[name].medianAbsoluteDeviation,
}))

const snapshot = {
...runSettings,
name: `${element.name}/${bench.name}`,
metrics: metrics
}
snapshotsToSave.push(snapshot);

});
});

return db.saveSnapshots(snapshotsToSave, projectName);
}),
);
}
Loading