-
Notifications
You must be signed in to change notification settings - Fork 10
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
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
9abce33
chore: cleaning up
aec5e26
chore: update version numbers to match the rest of best
20a6df8
wip: begin work to integrate api-db into the cli
fabeaa6
chore: remove eslint in api-db
a5d6d83
feat: initial implementation of api-db into cli
ec96612
fix: wrong boolean for temporary on snapshot
4af65bd
feat: add ability to configure api-db by best config file
05d493e
chore: remove testing username from best.config.js
f59cf6a
fix: metrics not get properly normalized for the database
7eee3e4
fix: get commit date directly from git
c7e1514
feat: you can now select commits when viewing a benchmark
2cab84a
wip: begin implementation of selecting commits to show more info
a3e83ba
wip: designing commit info popover
2d92caf
chore: clean up some old comments/todos
0f42aa9
fix: clean up graphs with better ticks
384a937
fix: improve the format that view.zoom is being stored in urlstorage
c4649b8
fix: changing selected metric now re-renders benchmarks
0c2a55a
feat: commit info now shows up in the correct position
15b2e74
fix: close button for commit info
0a68e19
feat: refactor benchmarks and create new graph component
e56a031
fix: relayout handler now always get attached
23baca6
fix: graphs now get properly resized when window resizes
12a4aca
fix: non-temporary snapshots now must be unique
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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" | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
export { saveBenchmarkSummaryInDB } from './store'; | ||
|
||
export { loadDbFromConfig } from './utils'; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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; | ||
} | ||
} |
17 changes: 17 additions & 0 deletions
17
packages/@best/api-db/src/postgres/migrations/1559599488979_setup-projects.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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'); | ||
}; |
34 changes: 34 additions & 0 deletions
34
packages/@best/api-db/src/postgres/migrations/1559600258932_benchmark-snapshot.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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'); | ||
}; |
11 changes: 11 additions & 0 deletions
11
packages/@best/api-db/src/postgres/migrations/1559603549891_add-updated-at.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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'), | ||
}, | ||
}); | ||
}; |
9 changes: 9 additions & 0 deletions
9
packages/@best/api-db/src/postgres/migrations/1559770417215_project-last-release.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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', | ||
}, | ||
}); | ||
}; |
5 changes: 5 additions & 0 deletions
5
packages/@best/api-db/src/postgres/migrations/1560444988970_unique-snapshots.js
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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'` }) | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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 | ||
})); | ||
} | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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); | ||
}), | ||
); | ||
} |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Where this comes from?
There was a problem hiding this comment.
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.