Compare commits
38 Commits
bc70fb8dd0
...
v0.2.1
Author | SHA1 | Date | |
---|---|---|---|
39f95575da | |||
6e1ab89209 | |||
9cd6c78741 | |||
5b730df803 | |||
0ce4a3ecee | |||
12ece12394 | |||
05c20b5811 | |||
bd822664b0 | |||
3f0efce0d8 | |||
eeedc57cd3 | |||
164eea1bde | |||
85fe3b0b38 | |||
519955e57a | |||
96b746822c | |||
eac833fd6d | |||
ad391c84a4 | |||
641c9315bc | |||
3a136865b0 | |||
|
f40d69a98d | ||
|
40cfdcc2a3 | ||
|
260d0d1268 | ||
|
e1f5bd3950 | ||
|
1f0c481105 | ||
|
dd8e419e52 | ||
|
c74ca114d8 | ||
0a1037e79a | |||
|
bc5ae4962f | ||
|
ec2bf45a6e | ||
57163b10e4 | |||
25d394627d | |||
85dfdb163a | |||
b593ef7593 | |||
26698082f4 | |||
42a0b95015 | |||
5cc3b8f824 | |||
d59c75281e | |||
55e4f397f8 | |||
49d1276031 |
3
.gitignore
vendored
3
.gitignore
vendored
@@ -65,3 +65,6 @@ node_modules/
|
||||
# Compiled assets
|
||||
public/css/
|
||||
public/js/
|
||||
|
||||
# Ignore production configuration files to protect against credential leaks
|
||||
config/production.yaml
|
||||
|
@@ -1,4 +1,6 @@
|
||||
require: rubocop-sequel
|
||||
plugins:
|
||||
- rubocop-rspec
|
||||
- rubocop-sequel
|
||||
|
||||
AllCops:
|
||||
NewCops: enable
|
||||
|
@@ -1,11 +1,32 @@
|
||||
pipeline:
|
||||
style:
|
||||
image: ruby:3.0
|
||||
steps:
|
||||
setup:
|
||||
image: ruby:3.4
|
||||
env:
|
||||
RACK_ENV: testing
|
||||
commands:
|
||||
- 'gem install rake'
|
||||
- 'bundle config set --local path "vendor/bundle"'
|
||||
- 'bundle install'
|
||||
- 'rake test:rubocop'
|
||||
- gem install rake
|
||||
- bundle config set --local path "vendor/bundle"
|
||||
- bundle install
|
||||
- rake db:migrate
|
||||
|
||||
test_ruby34:
|
||||
image: ruby:3.4
|
||||
env:
|
||||
RACK_ENV: testing
|
||||
commands:
|
||||
- gem install rake
|
||||
- bundle config set --local path "vendor/bundle"
|
||||
- rake test:unit
|
||||
group: tests
|
||||
|
||||
style:
|
||||
image: ruby:3.4
|
||||
env:
|
||||
RACK_ENV: testing
|
||||
commands:
|
||||
- gem install rake
|
||||
- bundle config set --local path "vendor/bundle"
|
||||
- rake test:lint
|
||||
|
||||
gitea_release:
|
||||
image: plugins/gitea-release
|
||||
@@ -15,5 +36,5 @@ pipeline:
|
||||
base_url: https://git.metaunix.net
|
||||
title: "${CI_COMMIT_TAG}"
|
||||
when:
|
||||
event: tag
|
||||
|
||||
event:
|
||||
- tag
|
||||
|
11
Gemfile
11
Gemfile
@@ -1,7 +1,7 @@
|
||||
source 'https://rubygems.org'
|
||||
|
||||
gem 'sinatra', '~> 3.2'
|
||||
gem 'sinatra-contrib', '~> 3.2'
|
||||
gem 'sinatra', '~> 4.1'
|
||||
gem 'sinatra-contrib', '~> 4.1'
|
||||
gem 'puma', '~> 6.6'
|
||||
|
||||
gem 'sequel', '~> 5.92'
|
||||
@@ -15,5 +15,12 @@ group :development, :test do
|
||||
|
||||
# rubocop and extensions for code style
|
||||
gem 'rubocop'
|
||||
gem 'rubocop-rspec'
|
||||
gem 'rubocop-sequel'
|
||||
end
|
||||
|
||||
group :test do
|
||||
gem 'rspec'
|
||||
gem 'rack-test'
|
||||
gem 'database_cleaner-sequel'
|
||||
end
|
||||
|
55
Gemfile.lock
55
Gemfile.lock
@@ -4,6 +4,11 @@ GEM
|
||||
ast (2.4.3)
|
||||
base64 (0.3.0)
|
||||
bigdecimal (3.2.2)
|
||||
database_cleaner-core (2.0.1)
|
||||
database_cleaner-sequel (2.0.2)
|
||||
database_cleaner-core (~> 2.0.0)
|
||||
sequel
|
||||
diff-lcs (1.6.2)
|
||||
ffi (1.17.2-aarch64-linux-gnu)
|
||||
ffi (1.17.2-aarch64-linux-musl)
|
||||
ffi (1.17.2-arm-linux-gnu)
|
||||
@@ -33,10 +38,16 @@ GEM
|
||||
puma (6.6.0)
|
||||
nio4r (~> 2.0)
|
||||
racc (1.8.1)
|
||||
rack (2.2.17)
|
||||
rack-protection (3.2.0)
|
||||
rack (3.2.0)
|
||||
rack-protection (4.1.1)
|
||||
base64 (>= 0.1.0)
|
||||
rack (~> 2.2, >= 2.2.4)
|
||||
logger (>= 1.6.0)
|
||||
rack (>= 3.0.0, < 4)
|
||||
rack-session (2.1.1)
|
||||
base64 (>= 0.1.0)
|
||||
rack (>= 3.0.0)
|
||||
rack-test (2.2.0)
|
||||
rack (>= 1.3)
|
||||
rainbow (3.1.1)
|
||||
rb-fsevent (0.11.2)
|
||||
rb-inotify (0.11.1)
|
||||
@@ -44,6 +55,19 @@ GEM
|
||||
regexp_parser (2.10.0)
|
||||
rerun (0.14.0)
|
||||
listen (~> 3.0)
|
||||
rspec (3.13.1)
|
||||
rspec-core (~> 3.13.0)
|
||||
rspec-expectations (~> 3.13.0)
|
||||
rspec-mocks (~> 3.13.0)
|
||||
rspec-core (3.13.5)
|
||||
rspec-support (~> 3.13.0)
|
||||
rspec-expectations (3.13.5)
|
||||
diff-lcs (>= 1.2.0, < 2.0)
|
||||
rspec-support (~> 3.13.0)
|
||||
rspec-mocks (3.13.5)
|
||||
diff-lcs (>= 1.2.0, < 2.0)
|
||||
rspec-support (~> 3.13.0)
|
||||
rspec-support (3.13.4)
|
||||
rubocop (1.76.1)
|
||||
json (~> 2.3)
|
||||
language_server-protocol (~> 3.17.0.2)
|
||||
@@ -58,6 +82,9 @@ GEM
|
||||
rubocop-ast (1.45.1)
|
||||
parser (>= 3.3.7.2)
|
||||
prism (~> 1.4)
|
||||
rubocop-rspec (3.6.0)
|
||||
lint_roller (~> 1.1)
|
||||
rubocop (~> 1.72, >= 1.72.1)
|
||||
rubocop-sequel (0.4.1)
|
||||
lint_roller (~> 1.1)
|
||||
rubocop (>= 1.72.1, < 2)
|
||||
@@ -65,16 +92,18 @@ GEM
|
||||
ruby2_keywords (0.0.5)
|
||||
sequel (5.93.0)
|
||||
bigdecimal
|
||||
sinatra (3.2.0)
|
||||
sinatra (4.1.1)
|
||||
logger (>= 1.6.0)
|
||||
mustermann (~> 3.0)
|
||||
rack (~> 2.2, >= 2.2.4)
|
||||
rack-protection (= 3.2.0)
|
||||
rack (>= 3.0.0, < 4)
|
||||
rack-protection (= 4.1.1)
|
||||
rack-session (>= 2.0.0, < 3)
|
||||
tilt (~> 2.0)
|
||||
sinatra-contrib (3.2.0)
|
||||
sinatra-contrib (4.1.1)
|
||||
multi_json (>= 0.0.2)
|
||||
mustermann (~> 3.0)
|
||||
rack-protection (= 3.2.0)
|
||||
sinatra (= 3.2.0)
|
||||
rack-protection (= 4.1.1)
|
||||
sinatra (= 4.1.1)
|
||||
tilt (~> 2.0)
|
||||
sqlite3 (2.7.0-aarch64-linux-gnu)
|
||||
sqlite3 (2.7.0-aarch64-linux-musl)
|
||||
@@ -104,14 +133,18 @@ PLATFORMS
|
||||
x86_64-linux-musl
|
||||
|
||||
DEPENDENCIES
|
||||
database_cleaner-sequel
|
||||
logger
|
||||
puma (~> 6.6)
|
||||
rack-test
|
||||
rerun
|
||||
rspec
|
||||
rubocop
|
||||
rubocop-rspec
|
||||
rubocop-sequel
|
||||
sequel (~> 5.92)
|
||||
sinatra (~> 3.2)
|
||||
sinatra-contrib (~> 3.2)
|
||||
sinatra (~> 4.1)
|
||||
sinatra-contrib (~> 4.1)
|
||||
sqlite3 (~> 2.6)
|
||||
|
||||
BUNDLED WITH
|
||||
|
44
README.md
44
README.md
@@ -1,5 +1,7 @@
|
||||
# Game Data
|
||||
|
||||

|
||||
|
||||
Web-based tool to store and organize PC hardware gaming benchmarks.
|
||||
|
||||
## Project Goals
|
||||
@@ -7,28 +9,58 @@ Web-based tool to store and organize PC hardware gaming benchmarks.
|
||||
The goals of this project are to:
|
||||
|
||||
* Record benchmarking results from multiple devices - e.g. log from a laptop or a phone.
|
||||
* Group results into tests - it's good practice to run a benchmark multiple times for accuracy.
|
||||
* Group results into tests to keep track of different testing configurations.
|
||||
* Encourage running tests multiple times - it's good practice to run a benchmark multiple times for accuracy.
|
||||
* Create comparisons of hardware tests to compare performance.
|
||||
* Generate graphs of hardware comparisons for usage in videos and articles.
|
||||
|
||||
## Requirements
|
||||
|
||||
Game Data runs on Ruby, and takes advantage of Bundler to manage code dependencies and Rake to run various tasks for maintaining the app. You can install them globally like so:
|
||||
Game Data runs on Ruby, and takes advantage of [Bundler](https://bundler.io/) to manage code dependencies and [Rake](https://ruby.github.io/rake/) to run various tasks for maintaining the app. You can install them globally like so:
|
||||
|
||||
Debian/Ubuntu: `apt install -y ruby ruby-bundler rake`
|
||||
RedHat and clones: `dnf install -y ruby rubygem-bundler rubygem-rake`
|
||||
|
||||
## Production Deployment
|
||||
|
||||
**TBD**
|
||||
|
||||
## Development
|
||||
|
||||
Install dependencies via bundler:
|
||||
### Via Docker
|
||||
|
||||
If you'd prefer not to install dependencies and such to your local OS, you can do the development via Docker. The scripts provided in `bin/` will build Docker images for running the Ruby app and building the front-end assets via Gulp. *Both containers will automatically watch for changes.*
|
||||
|
||||
**Note:** Using the scripts below, the Docker images will remove themselves when stopped. This is to make clean up a bit more streamlined.
|
||||
|
||||
1. [Install Docker](https://docs.docker.com/engine/install/) for your OS.
|
||||
|
||||
2. Build the docker images for Ruby and Gulp:
|
||||
|
||||
`bin/docker-build.sh`
|
||||
|
||||
3. Run the docker images:
|
||||
|
||||
`bin/docker-run.sh`
|
||||
|
||||
4. If everything is running successfully you can open your browser and go to https://localhost:9292.
|
||||
|
||||
### Local/Native Development
|
||||
|
||||
1. Install dependencies via bundler:
|
||||
|
||||
`bundle install`
|
||||
|
||||
Perform database migrations:
|
||||
2. Perform database migrations:
|
||||
|
||||
`rake db:migrate`
|
||||
|
||||
Run the server in development with auto-reloading:
|
||||
3. Run the server in development with auto-reloading:
|
||||
|
||||
`rake server:dev`
|
||||
|
||||
If everything is running successfully you can open your browser and go to https://localhost:9292.
|
||||
4. If everything is running successfully you can open your browser and go to https://localhost:9292.
|
||||
|
||||
## License
|
||||
|
||||
This project is available under the BSD 2-Clause license.
|
||||
|
22
Rakefile
22
Rakefile
@@ -3,18 +3,22 @@ require 'bundler/setup'
|
||||
namespace :db do
|
||||
desc 'Run migrations'
|
||||
task :migrate, [:version] do |t, args|
|
||||
require "sequel/core"
|
||||
require 'sequel/core'
|
||||
# load configuration
|
||||
require_relative 'src/config'
|
||||
conf = Config.new()
|
||||
|
||||
Sequel.extension :migration
|
||||
version = args[:version].to_i if args[:version]
|
||||
Sequel.connect('sqlite://data/gamedata.db') do |db|
|
||||
Sequel::Migrator.run(db, "db/migrations", target: version)
|
||||
Sequel.connect(adapter: conf.get('database.adapter'), database: conf.get('database.database')) do |db|
|
||||
Sequel::Migrator.run(db, 'db/migrations', target: version)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
namespace :server do
|
||||
task :start do
|
||||
ENV['APP_ENV'] = 'production'
|
||||
ENV['RACK_ENV'] = 'production'
|
||||
system("puma")
|
||||
end
|
||||
|
||||
@@ -24,8 +28,12 @@ namespace :server do
|
||||
end
|
||||
|
||||
namespace :test do
|
||||
task :rubocop do
|
||||
system("rubocop src/")
|
||||
task :unit do
|
||||
ENV['RACK_ENV'] = 'testing'
|
||||
system("rspec")
|
||||
end
|
||||
|
||||
task :lint do
|
||||
system("rubocop src/ spec/")
|
||||
end
|
||||
end
|
||||
|
||||
|
@@ -1,3 +1,21 @@
|
||||
$ ->
|
||||
# run foundation scripts
|
||||
$(document).foundation()
|
||||
console.log('Ready.')
|
||||
|
||||
averageResults = (results, decimals = 2) ->
|
||||
avgScore = 0
|
||||
minScore = Infinity
|
||||
maxScore = -Infinity
|
||||
|
||||
factor = (10 ^ decimals)
|
||||
|
||||
for result in results
|
||||
avgScore += result.avg_score
|
||||
minScore = Math.min(minScore, result.min_score)
|
||||
maxScore = Math.max(maxScore, result.max_score)
|
||||
|
||||
return {
|
||||
avgScore: Math.round((avgScore / results.length) * factor) / factor,
|
||||
minScore: minScore,
|
||||
maxScore: maxScore,
|
||||
}
|
||||
|
@@ -1,55 +1,109 @@
|
||||
$(document).ready ->
|
||||
$('#generate_button').on 'click', (e) ->
|
||||
e.preventDefault()
|
||||
$.ajax(
|
||||
method: 'POST'
|
||||
url: '/reports'
|
||||
data:
|
||||
type: $('#report_type').val()
|
||||
choice: $('#report_choice').val()
|
||||
compare: $('#report_compare').val()
|
||||
).done (data) ->
|
||||
benchChart.options.title.text = data.choice
|
||||
benchChart.data.labels = data.names
|
||||
benchChart.data.datasets[0].data = data.avg_results
|
||||
benchChart.data.datasets[1].data = data.min_results
|
||||
benchChart.update()
|
||||
return
|
||||
return
|
||||
return
|
||||
$ ->
|
||||
chartInstance = null
|
||||
|
||||
benchChart = new Chart(document.getElementById('chart_canvas').getContext('2d'),
|
||||
type: 'horizontalBar'
|
||||
data:
|
||||
$('#report-benchmarks').on 'change', (e) ->
|
||||
$('#report-tests option').prop('selected', false)
|
||||
|
||||
$('#reports-download').on 'click', (e) ->
|
||||
e.preventDefault()
|
||||
canvas = $('#benchmark-chart')[0]
|
||||
a = document.createElement 'a'
|
||||
a.href = canvas.toDataURL 'image/png'
|
||||
a.download = 'chart.png'
|
||||
a.click()
|
||||
|
||||
$('#reports-button').on 'click', (e) ->
|
||||
e.preventDefault()
|
||||
$('#reports-download').attr('disabled', true)
|
||||
chartInstance.destroy() if chartInstance
|
||||
|
||||
benchmarkId = $('#report-benchmarks').val()
|
||||
testIds = $('#report-tests').val()
|
||||
|
||||
benchmarkSearchParams = new URLSearchParams
|
||||
benchmark_id: benchmarkId
|
||||
benchmarkRes = await fetch("/api/v1/benchmark/details?#{benchmarkSearchParams}")
|
||||
benchmarkData = await benchmarkRes.json()
|
||||
|
||||
data =
|
||||
labels: []
|
||||
datasets: [
|
||||
{
|
||||
datasets: []
|
||||
|
||||
switch benchmarkData.scoring
|
||||
when 'pts'
|
||||
data.datasets.push({
|
||||
label: 'Average Score'
|
||||
data: []
|
||||
})
|
||||
when 'fps'
|
||||
data.datasets.push({
|
||||
label: 'Average FPS'
|
||||
data: []
|
||||
backgroundColor: 'hotpink'
|
||||
borderColor: '#212121'
|
||||
borderWidth: 1
|
||||
}
|
||||
{
|
||||
})
|
||||
data.datasets.push({
|
||||
label: 'Minimum FPS'
|
||||
data: []
|
||||
backgroundColor: 'cornflowerblue'
|
||||
borderColor: '#212121'
|
||||
borderWidth: 1
|
||||
}
|
||||
]
|
||||
options:
|
||||
})
|
||||
when 'ms'
|
||||
data.datasets.push({
|
||||
label: 'Average Frame Time'
|
||||
data: []
|
||||
})
|
||||
data.datasets.push({
|
||||
label: 'Minimum Frame Time'
|
||||
data: []
|
||||
})
|
||||
|
||||
for testId in testIds
|
||||
try
|
||||
testSearchParams = new URLSearchParams
|
||||
test_id: testId
|
||||
testRes = await fetch("/api/v1/test/details?#{testSearchParams}")
|
||||
testData = await testRes.json()
|
||||
|
||||
resultSearchParams = new URLSearchParams
|
||||
test_id: testId
|
||||
benchmark_id: benchmarkId
|
||||
resultRes = await fetch("/api/v1/result/list?#{resultSearchParams}")
|
||||
resultData = await resultRes.json()
|
||||
|
||||
resultAverage = averageResults(resultData)
|
||||
|
||||
data.labels.push(testData.name)
|
||||
data.datasets[0].data.push(resultAverage.avgScore)
|
||||
console.log(data.datasets[0].data)
|
||||
switch benchmarkData.scoring
|
||||
when 'fps', 'ms'
|
||||
data.datasets[1].data.push(resultAverage.minScore)
|
||||
catch error
|
||||
console.error 'An error occurred while fetching benchmark results.', error
|
||||
|
||||
ctx = $('#benchmark-chart')[0].getContext('2d')
|
||||
|
||||
options =
|
||||
indexAxis: 'y'
|
||||
plugins:
|
||||
title:
|
||||
display: true
|
||||
text: 'N/a'
|
||||
scales: xAxes: [ {
|
||||
display: true
|
||||
ticks: beginAtZero: true
|
||||
} ]
|
||||
animation: onComplete: ->
|
||||
dwnbtn = $('#download_button')
|
||||
dwnbtn.attr 'href', benchChart.toBase64Image()
|
||||
dwnbtn.attr 'download', 'benchmark_chart.png'
|
||||
dwnbtn.attr 'disabled', false
|
||||
return
|
||||
)
|
||||
text: benchmarkData.name
|
||||
font:
|
||||
size: '24'
|
||||
datalabels:
|
||||
anchor: 'end'
|
||||
align: 'left'
|
||||
color: 'black'
|
||||
font:
|
||||
weight: 'bold'
|
||||
formatter: (value) -> value
|
||||
scales:
|
||||
y:
|
||||
beginAtZero: true
|
||||
|
||||
chartInstance = new Chart ctx,
|
||||
type: 'bar'
|
||||
data: data
|
||||
options: options
|
||||
plugins: [ChartDataLabels]
|
||||
|
||||
$('#reports-download').attr('disabled', false)
|
||||
$('#benchmark-chart').removeClass('disabled')
|
||||
|
@@ -10,17 +10,10 @@ fetchTestBenchmarkResults = (testId, benchmarkId) ->
|
||||
resultSearchParams = new URLSearchParams
|
||||
test_id: testId
|
||||
benchmark_id: benchmarkId
|
||||
resultRes = await fetch("/api/v1/results?#{resultSearchParams}")
|
||||
resultRes = await fetch("/api/v1/result/list?#{resultSearchParams}")
|
||||
resultData = await resultRes.json()
|
||||
|
||||
avg_total = 0
|
||||
min_total = 0
|
||||
max_total = 0
|
||||
|
||||
for result in resultData
|
||||
avg_total += result.avg_score
|
||||
min_total += result.min_score
|
||||
max_total += result.max_score
|
||||
resultAverage = averageResults(resultData)
|
||||
|
||||
tableRow = $("#results-table tr[data-benchmark-id=#{benchmarkId}]")
|
||||
|
||||
@@ -29,16 +22,18 @@ fetchTestBenchmarkResults = (testId, benchmarkId) ->
|
||||
tableRow.append('<td>' + resultData.length + '</td>')
|
||||
|
||||
if resultData.length != 0
|
||||
tableRow.append('<td>' + (avg_total / resultData.length) + '</td>')
|
||||
tableRow.append('<td>' + resultAverage.avgScore + '</td>')
|
||||
|
||||
if benchmarkData.scoring == 'fps'
|
||||
tableRow.append('<td>' + resultAverage.minScore + '</td>')
|
||||
tableRow.append('<td>' + resultAverage.maxScore + '</td>')
|
||||
else
|
||||
tableRow.append('<td>N/a</td>')
|
||||
|
||||
if min_total != 0
|
||||
tableRow.append('<td>' + (min_total / resultData.length) + '</td>')
|
||||
tableRow.append('<td>' + (max_total / resultData.length) + '</td>')
|
||||
tableRow.append('<td>N/a</td>')
|
||||
else
|
||||
tableRow.append('<td>N/a</td>')
|
||||
tableRow.append('<td>N/a</td>')
|
||||
tableRow.append('<td>N/a</td>')
|
||||
catch error
|
||||
console.error 'An error occurred while fetching benchmark results.', error
|
||||
|
||||
|
@@ -4,6 +4,10 @@
|
||||
$primary-color: cornflowerblue
|
||||
$primary-color-highlight: color.adjust($primary-color, $lightness: -10%)
|
||||
|
||||
html
|
||||
width: 100%
|
||||
height: 100%
|
||||
|
||||
body
|
||||
background: rgb(240, 235, 248)
|
||||
|
||||
|
@@ -1,6 +1,4 @@
|
||||
# Load application config
|
||||
require_relative 'src/config.rb'
|
||||
$conf = Config.new(File.join(__dir__, 'config/defaults.yaml'))
|
||||
|
||||
root = ::File.dirname(__FILE__)
|
||||
require ::File.join( root, 'src', 'server' )
|
||||
|
@@ -1,6 +1,6 @@
|
||||
database:
|
||||
adapter: 'sqlite'
|
||||
database: 'data/gamedata.db'
|
||||
server:
|
||||
host: '0.0.0.0'
|
||||
port: '9292'
|
||||
|
||||
testing:
|
||||
minimum_results_required: 3
|
||||
|
3
config/development.yaml
Normal file
3
config/development.yaml
Normal file
@@ -0,0 +1,3 @@
|
||||
database:
|
||||
adapter: 'sqlite'
|
||||
database: 'data/gamedata.db'
|
@@ -1,14 +1,11 @@
|
||||
app_dir = File.expand_path("..", __dir__)
|
||||
log_dir = File.join(app_dir, "log")
|
||||
pid_dir = "/var/run/game-data" # or use app_dir + '/tmp/pids' if not using system-wide PID dir
|
||||
|
||||
app_dir = File.expand_path('..', __dir__)
|
||||
directory app_dir
|
||||
pidfile File.join(pid_dir, "puma.pid")
|
||||
stdout_redirect File.join(log_dir, "puma.log"), File.join(log_dir, "puma.err.log"), true
|
||||
|
||||
environment 'production'
|
||||
environment ENV.fetch('RACK_ENV', 'development')
|
||||
|
||||
bind 'tcp://0.0.0.0:9292'
|
||||
require_relative '../src/config'
|
||||
conf = Config.new()
|
||||
|
||||
bind "tcp://#{conf.get('server.host')}:#{conf.get('server.port')}"
|
||||
workers 2
|
||||
threads 1, 5
|
||||
daemonize true
|
||||
|
3
config/testing.yaml
Normal file
3
config/testing.yaml
Normal file
@@ -0,0 +1,3 @@
|
||||
database:
|
||||
adapter: 'sqlite'
|
||||
database: 'data/gamedata_testing.db'
|
44
spec/controllers/benchmark_controller_spec.rb
Normal file
44
spec/controllers/benchmark_controller_spec.rb
Normal file
@@ -0,0 +1,44 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative '../spec_helper'
|
||||
|
||||
RSpec.describe(BenchmarkController) do
|
||||
# /benchmark - redirects to /benchmark/list
|
||||
describe 'GET /benchmark' do
|
||||
before { get '/benchmark' }
|
||||
|
||||
it 'Benchmark base route is a redirect' do
|
||||
expect(last_response).to(be_redirect)
|
||||
end
|
||||
|
||||
it 'Benchmark base route Location header points to /benchmark/list' do
|
||||
expect(last_response['Location']).to(eq("#{BASE_URL}/benchmark/list"))
|
||||
end
|
||||
end
|
||||
|
||||
# /benchmark/list - displays a table of benchmarks
|
||||
describe 'GET /benchmark/list' do
|
||||
before { get '/benchmark/list' }
|
||||
|
||||
it 'Benchmark list page returns 200.' do
|
||||
expect(last_response).to(be_ok)
|
||||
end
|
||||
|
||||
it "Benchmark list page contains 'List of benchmarks' on page." do
|
||||
expect(last_response.body).to(include('List of benchmarks'))
|
||||
end
|
||||
end
|
||||
|
||||
# /benchmark/add - form for adding benchmark
|
||||
describe 'GET /benchmark/add' do
|
||||
before { get '/benchmark/add' }
|
||||
|
||||
it 'Benchmark add page returns 200.' do
|
||||
expect(last_response).to(be_ok)
|
||||
end
|
||||
|
||||
it "Benchmark add page contains 'Add new benchmark' on page." do
|
||||
expect(last_response.body).to(include('Add new benchmark'))
|
||||
end
|
||||
end
|
||||
end
|
44
spec/controllers/hardware_controller_spec.rb
Normal file
44
spec/controllers/hardware_controller_spec.rb
Normal file
@@ -0,0 +1,44 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative '../spec_helper'
|
||||
|
||||
RSpec.describe(HardwareController) do
|
||||
# /hardware - redirects to /hardware/list
|
||||
describe 'GET /hardware' do
|
||||
before { get '/hardware' }
|
||||
|
||||
it 'Hardware base route is a redirect' do
|
||||
expect(last_response).to(be_redirect)
|
||||
end
|
||||
|
||||
it 'Hardware base route Location header points to /hardware/list' do
|
||||
expect(last_response['Location']).to(eq("#{BASE_URL}/hardware/list"))
|
||||
end
|
||||
end
|
||||
|
||||
# /hardware/list - displays a table of hardwares
|
||||
describe 'GET /hardware/list' do
|
||||
before { get '/hardware/list' }
|
||||
|
||||
it 'Hardware list page returns 200.' do
|
||||
expect(last_response).to(be_ok)
|
||||
end
|
||||
|
||||
it "Hardware list page contains 'List of hardware' on page." do
|
||||
expect(last_response.body).to(include('List of hardware'))
|
||||
end
|
||||
end
|
||||
|
||||
# /hardware/add - form for adding hardware
|
||||
describe 'GET /hardware/add' do
|
||||
before { get '/hardware/add' }
|
||||
|
||||
it 'Hardware add page returns 200.' do
|
||||
expect(last_response).to(be_ok)
|
||||
end
|
||||
|
||||
it "Hardware add page contains 'Add new hardware' on page." do
|
||||
expect(last_response.body).to(include('Add new hardware'))
|
||||
end
|
||||
end
|
||||
end
|
21
spec/controllers/index_controller_spec.rb
Normal file
21
spec/controllers/index_controller_spec.rb
Normal file
@@ -0,0 +1,21 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative '../spec_helper'
|
||||
|
||||
RSpec.describe(IndexController) do
|
||||
describe 'GET /' do
|
||||
before { get '/' }
|
||||
|
||||
it 'Dashboard returns 200.' do
|
||||
expect(last_response).to(be_ok)
|
||||
end
|
||||
|
||||
it "Dashboard contains 'Game Data' on page (nav bar should be loaded)." do
|
||||
expect(last_response.body).to(include('Game Data'))
|
||||
end
|
||||
|
||||
it "Dashboard contains 'Ruby version' on page (footer should be loaded)." do
|
||||
expect(last_response.body).to(include('Ruby version'))
|
||||
end
|
||||
end
|
||||
end
|
17
spec/controllers/reports_controller_spec.rb
Normal file
17
spec/controllers/reports_controller_spec.rb
Normal file
@@ -0,0 +1,17 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative '../spec_helper'
|
||||
|
||||
RSpec.describe(ReportsController) do
|
||||
describe 'GET /report' do
|
||||
before { get '/report' }
|
||||
|
||||
it 'Reports page returns 200.' do
|
||||
expect(last_response).to(be_ok)
|
||||
end
|
||||
|
||||
it "Reports page contains 'Generate report' on page." do
|
||||
expect(last_response.body).to(include('Generate report'))
|
||||
end
|
||||
end
|
||||
end
|
44
spec/controllers/test_controller_spec.rb
Normal file
44
spec/controllers/test_controller_spec.rb
Normal file
@@ -0,0 +1,44 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative '../spec_helper'
|
||||
|
||||
RSpec.describe(TestController) do
|
||||
# /test - redirects to /test/list
|
||||
describe 'GET /test' do
|
||||
before { get '/test' }
|
||||
|
||||
it 'Test base route is a redirect' do
|
||||
expect(last_response).to(be_redirect)
|
||||
end
|
||||
|
||||
it 'Test base route Location header points to /test/list' do
|
||||
expect(last_response['Location']).to(eq("#{BASE_URL}/test/list"))
|
||||
end
|
||||
end
|
||||
|
||||
# /test/list - displays a table of tests
|
||||
describe 'GET /test/list' do
|
||||
before { get '/test/list' }
|
||||
|
||||
it 'Test list page returns 200.' do
|
||||
expect(last_response).to(be_ok)
|
||||
end
|
||||
|
||||
it "Test list page contains 'List of tests' on page." do
|
||||
expect(last_response.body).to(include('List of tests'))
|
||||
end
|
||||
end
|
||||
|
||||
# /test/add - form for adding test
|
||||
describe 'GET /test/add' do
|
||||
before { get '/test/add' }
|
||||
|
||||
it 'Test add page returns 200.' do
|
||||
expect(last_response).to(be_ok)
|
||||
end
|
||||
|
||||
it "Test add page contains 'Add new test' on page." do
|
||||
expect(last_response.body).to(include('Add new test'))
|
||||
end
|
||||
end
|
||||
end
|
27
spec/models/benchmark_model_spec.rb
Normal file
27
spec/models/benchmark_model_spec.rb
Normal file
@@ -0,0 +1,27 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative '../spec_helper'
|
||||
|
||||
RSpec.describe(Benchmark) do
|
||||
describe 'Benchmark Creation' do
|
||||
it 'Benchmark creation updates model count.' do
|
||||
expect do
|
||||
described_class.create(name: 'Test Benchmark', scoring: 'fps')
|
||||
end.to(change(described_class, :count).by(1))
|
||||
end
|
||||
end
|
||||
|
||||
describe 'Benchmark Read' do
|
||||
before { described_class.create(name: 'Test Benchmark', scoring: 'fps') }
|
||||
|
||||
it 'Benchmark model has name.' do
|
||||
bench = described_class.first()
|
||||
expect(bench.name).to(eq('Test Benchmark'))
|
||||
end
|
||||
|
||||
it 'Benchmark model has scoring.' do
|
||||
bench = described_class.first()
|
||||
expect(bench.scoring).to(eq('fps'))
|
||||
end
|
||||
end
|
||||
end
|
25
spec/models/hardware_model_spec.rb
Normal file
25
spec/models/hardware_model_spec.rb
Normal file
@@ -0,0 +1,25 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative '../spec_helper'
|
||||
|
||||
RSpec.describe(Hardware) do
|
||||
describe 'Hardware Creation' do
|
||||
it 'Hardware creation updates model count.' do
|
||||
expect { described_class.create(name: 'Test Hardware', type: 'gpu') }.to(change(described_class, :count).by(1))
|
||||
end
|
||||
end
|
||||
|
||||
describe 'Hardware Read' do
|
||||
before { described_class.create(name: 'Test Hardware', type: 'gpu') }
|
||||
|
||||
it 'Hardware model has name.' do
|
||||
hardware = described_class.first()
|
||||
expect(hardware.name).to(eq('Test Hardware'))
|
||||
end
|
||||
|
||||
it 'Hardware model has scoring.' do
|
||||
hardware = described_class.first()
|
||||
expect(hardware.type).to(eq('gpu'))
|
||||
end
|
||||
end
|
||||
end
|
36
spec/models/test_model_spec.rb
Normal file
36
spec/models/test_model_spec.rb
Normal file
@@ -0,0 +1,36 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative '../spec_helper'
|
||||
|
||||
RSpec.describe(Test) do
|
||||
describe 'Test Creation' do
|
||||
it 'Test creation updates model count.' do
|
||||
expect { described_class.create(name: 'Test Test') }.to(change(described_class, :count).by(1))
|
||||
end
|
||||
end
|
||||
|
||||
describe 'Test Read' do
|
||||
before do
|
||||
described_class.create(name: 'Test Test')
|
||||
end
|
||||
|
||||
it 'Test model has name.' do
|
||||
tst = described_class.first()
|
||||
expect(tst.name).to(eq('Test Test'))
|
||||
end
|
||||
end
|
||||
|
||||
describe 'Test one-to-many association with Hardware' do
|
||||
it 'Test model has Hardware associated with it.' do
|
||||
hardware = Hardware.create(name: 'Test Hardware', type: 'gpu')
|
||||
tst = described_class.create(name: 'Test Test', hardware_id: hardware.id)
|
||||
expect(tst.hardware).to(eq(hardware))
|
||||
end
|
||||
|
||||
it 'Test model\'s hardware has name set.' do
|
||||
hardware = Hardware.create(name: 'Test Hardware', type: 'gpu')
|
||||
tst = described_class.create(name: 'Test Test', hardware_id: hardware.id)
|
||||
expect(tst.hardware.name).to(eq('Test Hardware'))
|
||||
end
|
||||
end
|
||||
end
|
35
spec/spec_helper.rb
Normal file
35
spec/spec_helper.rb
Normal file
@@ -0,0 +1,35 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
ENV['APP_ENV'] = 'test'
|
||||
|
||||
require_relative '../src/server'
|
||||
require 'rspec'
|
||||
require 'rack/test'
|
||||
require 'database_cleaner/sequel'
|
||||
|
||||
# setting this here so all redirect tests can reference the same base URL
|
||||
BASE_URL = 'http://example.org'
|
||||
|
||||
module RSpecMixin
|
||||
|
||||
include Rack::Test::Methods
|
||||
|
||||
def app
|
||||
GameData
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
RSpec.configure do |config|
|
||||
config.include(RSpecMixin)
|
||||
|
||||
config.before(:suite) do
|
||||
DatabaseCleaner.strategy = :transaction
|
||||
end
|
||||
|
||||
config.around do |suite|
|
||||
DatabaseCleaner.cleaning do
|
||||
suite.run
|
||||
end
|
||||
end
|
||||
end
|
7
src/appinfo.rb
Normal file
7
src/appinfo.rb
Normal file
@@ -0,0 +1,7 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
module AppInfo
|
||||
|
||||
VERSION = '0.2.1'
|
||||
|
||||
end
|
@@ -6,8 +6,9 @@ require 'yaml'
|
||||
class Config
|
||||
|
||||
DEFAULT_CONFIG = 'config/defaults.yaml'
|
||||
ENVIRONMENT_CONFIG = ENV.fetch('RACK_ENV', 'development')
|
||||
|
||||
def initialize(config_path)
|
||||
def initialize(config_path = "config/#{ENVIRONMENT_CONFIG}.yaml")
|
||||
@data = YAML.load_file(DEFAULT_CONFIG)
|
||||
|
||||
# merge in user-defined configuration if it exists
|
||||
|
@@ -1,7 +1,14 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require 'sinatra/json'
|
||||
|
||||
require_relative 'base_controller'
|
||||
require_relative '../models/benchmark'
|
||||
require_relative '../models/test'
|
||||
require_relative '../models/result'
|
||||
|
||||
# /api/v1 routes
|
||||
class GameData < Sinatra::Base
|
||||
class APIv1Controller < BaseController
|
||||
|
||||
get '/api/v1/benchmark/details' do
|
||||
benchmark_id = params[:benchmark_id]
|
||||
@@ -11,7 +18,7 @@ class GameData < Sinatra::Base
|
||||
json benchmark.values()
|
||||
end
|
||||
|
||||
get '/api/v1/results' do
|
||||
get '/api/v1/result/list' do
|
||||
test_id = params[:test_id]
|
||||
benchmark_id = params[:benchmark_id]
|
||||
|
||||
@@ -20,4 +27,12 @@ class GameData < Sinatra::Base
|
||||
json results.map(&:values)
|
||||
end
|
||||
|
||||
get '/api/v1/test/details' do
|
||||
test_id = params[:test_id]
|
||||
|
||||
tst = Test.where(id: test_id).first()
|
||||
|
||||
json tst.values()
|
||||
end
|
||||
|
||||
end
|
15
src/controllers/base_controller.rb
Normal file
15
src/controllers/base_controller.rb
Normal file
@@ -0,0 +1,15 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require 'sinatra/base'
|
||||
|
||||
# BaseController - base modular Sinatra app class
|
||||
class BaseController < Sinatra::Base
|
||||
|
||||
# Register view helpers
|
||||
require_relative '../helpers'
|
||||
helpers Helpers
|
||||
|
||||
# Set up our view engine
|
||||
set :views, File.join(settings.root, '/../../views')
|
||||
|
||||
end
|
@@ -1,9 +1,16 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative 'base_controller'
|
||||
require_relative '../models/benchmark'
|
||||
|
||||
# /benchmark routes
|
||||
class GameData < Sinatra::Base
|
||||
class BenchmarkController < BaseController
|
||||
|
||||
get '/benchmark' do
|
||||
redirect('/benchmark/list')
|
||||
end
|
||||
|
||||
get '/benchmark/list' do
|
||||
benchmarks = Benchmark.reverse(:updated_at).limit(10).all()
|
||||
|
||||
erb :'benchmark/index', locals: {
|
@@ -1,9 +1,17 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative 'base_controller'
|
||||
require_relative '../models/hardware'
|
||||
require_relative '../models/benchmark'
|
||||
|
||||
# /hardware routes
|
||||
class GameData < Sinatra::Base
|
||||
class HardwareController < BaseController
|
||||
|
||||
get '/hardware' do
|
||||
redirect('/hardware/list')
|
||||
end
|
||||
|
||||
get '/hardware/list' do
|
||||
hardware = Hardware.reverse(:updated_at).limit(10).all()
|
||||
|
||||
erb :'hardware/index', locals: {
|
@@ -1,7 +1,10 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative 'base_controller'
|
||||
require_relative '../models/test'
|
||||
|
||||
# / (top-level) routes
|
||||
class GameData < Sinatra::Base
|
||||
class IndexController < BaseController
|
||||
|
||||
get '/' do
|
||||
tests = Test.reverse(:updated_at).limit(10).all()
|
@@ -1,7 +1,14 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require 'sinatra/json'
|
||||
|
||||
require_relative 'base_controller'
|
||||
require_relative '../models/benchmark'
|
||||
require_relative '../models/result'
|
||||
require_relative '../models/test'
|
||||
|
||||
# /reports routes
|
||||
class GameData < Sinatra::Base
|
||||
class ReportsController < BaseController
|
||||
|
||||
get '/report' do
|
||||
benchmarks = Benchmark.order(:name).all()
|
@@ -1,7 +1,10 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative 'base_controller'
|
||||
require_relative '../models/result'
|
||||
|
||||
# /result routes
|
||||
class GameData < Sinatra::Base
|
||||
class ResultController < BaseController
|
||||
|
||||
post '/result/add' do
|
||||
result_minimum = params[:result_minimum] if params.key?(:result_minimum)
|
96
src/controllers/test.rb
Normal file
96
src/controllers/test.rb
Normal file
@@ -0,0 +1,96 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative 'base_controller'
|
||||
require_relative '../models/benchmark'
|
||||
require_relative '../models/hardware'
|
||||
require_relative '../models/test'
|
||||
|
||||
# /test routes
|
||||
class TestController < BaseController
|
||||
|
||||
get '/test' do
|
||||
redirect('/test/list')
|
||||
end
|
||||
|
||||
get '/test/list' do
|
||||
tests = Test.reverse(:updated_at).limit(10).all()
|
||||
|
||||
erb :'test/index', locals: {
|
||||
title: 'List of Tests',
|
||||
tests: tests
|
||||
}
|
||||
end
|
||||
|
||||
get '/test/add' do
|
||||
hardware = Hardware.order(:name).all()
|
||||
benchmarks = Benchmark.order(:name).all()
|
||||
|
||||
erb :'test/add', locals: {
|
||||
title: 'Add Test',
|
||||
hardware: hardware,
|
||||
benchmarks: benchmarks
|
||||
}
|
||||
end
|
||||
post '/test/add' do
|
||||
tst = Test.create(
|
||||
name: params[:test_name],
|
||||
hardware_id: params[:test_hardware],
|
||||
description: params[:test_description]
|
||||
)
|
||||
|
||||
# create an array of the selected benchmarks
|
||||
benchmarks = Array(params[:test_benchmarks])
|
||||
# associate the benchmarks to the test
|
||||
benchmarks.each do |b|
|
||||
tst.add_benchmark(b)
|
||||
end
|
||||
|
||||
redirect "/test/#{tst.id}"
|
||||
end
|
||||
|
||||
get '/test/:test_id' do
|
||||
tst = Test.where(id: params[:test_id]).first()
|
||||
erb :'test/view', locals: {
|
||||
title: tst.name,
|
||||
test: tst
|
||||
}
|
||||
end
|
||||
|
||||
get '/test/:test_id/edit' do
|
||||
tst = Test.where(id: params[:test_id]).first()
|
||||
hardware = Hardware.order(:name).all()
|
||||
benchmarks = Benchmark.order(:name).all()
|
||||
|
||||
erb :'test/edit', locals: {
|
||||
title: "Editing: #{tst.name}",
|
||||
test: tst,
|
||||
hardware: hardware,
|
||||
benchmarks: benchmarks
|
||||
}
|
||||
end
|
||||
|
||||
post '/test/:test_id/edit' do
|
||||
tst = Test.where(id: params[:test_id]).first()
|
||||
|
||||
tst.update(
|
||||
name: params[:test_name],
|
||||
hardware_id: params[:test_hardware],
|
||||
description: params[:test_description]
|
||||
)
|
||||
|
||||
selected_benchmarks = Array(params[:test_benchmarks])
|
||||
|
||||
# remove benchmarks no longer associated with the test
|
||||
tst.benchmarks.dup.each do |b|
|
||||
tst.remove_benchmark(b.id) unless selected_benchmarks.include?(b.id)
|
||||
end
|
||||
|
||||
# associate the benchmarks to the test
|
||||
selected_benchmarks.each do |b|
|
||||
tst.add_benchmark(b) unless tst.benchmark?(b)
|
||||
end
|
||||
|
||||
redirect "/test/#{tst.id}"
|
||||
end
|
||||
|
||||
end
|
@@ -1,14 +1,16 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative 'appinfo'
|
||||
|
||||
# Helpers - view helper functions
|
||||
module Helpers
|
||||
|
||||
def ruby_version()
|
||||
def ruby_version
|
||||
return RUBY_VERSION
|
||||
end
|
||||
|
||||
def app_version()
|
||||
`git describe --tags`.strip
|
||||
def app_version
|
||||
return AppInfo::VERSION
|
||||
end
|
||||
|
||||
def date_format(date)
|
||||
|
@@ -1,6 +0,0 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative 'hardware'
|
||||
require_relative 'benchmark'
|
||||
require_relative 'result'
|
||||
require_relative 'test'
|
@@ -7,4 +7,8 @@ class Test < Sequel::Model
|
||||
many_to_one :hardware
|
||||
many_to_many :benchmarks
|
||||
|
||||
def benchmark?(benchmark_id)
|
||||
return benchmarks_dataset.where(Sequel[:benchmarks][:id] => benchmark_id).any?
|
||||
end
|
||||
|
||||
end
|
||||
|
@@ -1,10 +0,0 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require_relative 'index'
|
||||
require_relative 'hardware'
|
||||
require_relative 'benchmark'
|
||||
require_relative 'reports'
|
||||
require_relative 'result'
|
||||
require_relative 'test'
|
||||
|
||||
require_relative 'api1'
|
@@ -1,68 +0,0 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
# /test routes
|
||||
class GameData < Sinatra::Base
|
||||
|
||||
get '/test' do
|
||||
tests = Test.reverse(:updated_at).limit(10).all()
|
||||
|
||||
erb :'test/index', locals: {
|
||||
title: 'List of Tests',
|
||||
tests: tests
|
||||
}
|
||||
end
|
||||
|
||||
get '/test/add' do
|
||||
hardware = Hardware.order(:name).all()
|
||||
benchmarks = Benchmark.order(:name).all()
|
||||
erb :'test/add', locals: {
|
||||
title: 'Add Test',
|
||||
hardware: hardware,
|
||||
benchmarks: benchmarks
|
||||
}
|
||||
end
|
||||
post '/test/add' do
|
||||
tst = Test.create(
|
||||
name: params[:test_name],
|
||||
hardware_id: params[:test_hardware],
|
||||
description: params[:test_description]
|
||||
)
|
||||
|
||||
# create an array of the selected benchmarks
|
||||
benchmarks = Array(params[:test_benchmarks])
|
||||
# associate the benchmarks to the test
|
||||
benchmarks.each do |b|
|
||||
tst.add_benchmark(b)
|
||||
end
|
||||
|
||||
redirect "/test/#{tst.id}"
|
||||
end
|
||||
|
||||
get '/test/:test_id' do
|
||||
tst = Test.where(id: params[:test_id]).first()
|
||||
erb :'test/view', locals: {
|
||||
title: tst.name,
|
||||
test: tst
|
||||
}
|
||||
end
|
||||
|
||||
get '/test/:hardware_id/edit' do
|
||||
hardware = Hardware.where(id: params[:hardware_id]).first()
|
||||
erb :'test/edit', locals: {
|
||||
title: "Editing: #{hardware.name}",
|
||||
hardware: hardware
|
||||
}
|
||||
end
|
||||
|
||||
post '/test/:hardware_id/edit' do
|
||||
hardware = Hardware.where(id: params[:hardware_id]).first()
|
||||
|
||||
hardware.update(
|
||||
name: params[:hardware_name],
|
||||
type: params[:hardware_type]
|
||||
)
|
||||
|
||||
redirect "/hardware/#{hardware.id}"
|
||||
end
|
||||
|
||||
end
|
38
src/server.rb
Executable file → Normal file
38
src/server.rb
Executable file → Normal file
@@ -1,16 +1,32 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require 'sinatra/base'
|
||||
require 'sinatra/json'
|
||||
require 'sequel'
|
||||
require 'sqlite3'
|
||||
|
||||
require_relative 'config'
|
||||
|
||||
# Load configuration from environment config file
|
||||
$conf = Config.new()
|
||||
|
||||
# Load the Sequel timestamps plugin
|
||||
Sequel::Model.plugin(:timestamps)
|
||||
# Initialize Sequel gem for database actions
|
||||
DB = Sequel.connect(adapter: $conf.get('database.adapter'), database: $conf.get('database.database'))
|
||||
|
||||
# Base app
|
||||
# Load in routes (must happen after Sequel is loaded!)
|
||||
require_relative 'controllers/api1'
|
||||
require_relative 'controllers/benchmark'
|
||||
require_relative 'controllers/hardware'
|
||||
require_relative 'controllers/index'
|
||||
require_relative 'controllers/reports'
|
||||
require_relative 'controllers/result'
|
||||
require_relative 'controllers/test'
|
||||
|
||||
# GameData - main app that gets launched
|
||||
# - inherits from Sinatra::Base to instantiate the server
|
||||
# - sets up some base app configuration
|
||||
# - registers route classes with the base app
|
||||
class GameData < Sinatra::Base
|
||||
|
||||
enable :sessions
|
||||
@@ -19,16 +35,12 @@ class GameData < Sinatra::Base
|
||||
enable :static
|
||||
set :public_folder, File.join(__dir__, '/../public')
|
||||
|
||||
# Register view helpers
|
||||
require_relative 'helpers'
|
||||
helpers Helpers
|
||||
|
||||
# Set up our view engine
|
||||
set :views, File.join(settings.root, '/../views')
|
||||
use IndexController
|
||||
use HardwareController
|
||||
use BenchmarkController
|
||||
use TestController
|
||||
use ResultController
|
||||
use ReportsController
|
||||
use APIv1Controller
|
||||
|
||||
end
|
||||
|
||||
# Load routes
|
||||
require_relative 'routes/init'
|
||||
# Load models
|
||||
require_relative 'models/init'
|
||||
|
@@ -26,7 +26,7 @@
|
||||
<div class="row mb-3">
|
||||
<div class="col-12">
|
||||
<label for="benchmark_description">Benchmark description</label>
|
||||
<textarea id="benchmark_description" class="form-control" name="benchmark_description">Enter a description/notes here.</textarea>
|
||||
<textarea id="benchmark_description" class="form-control" name="benchmark_description" placeholder="Enter a description/notes here."></textarea>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
@@ -26,7 +26,7 @@
|
||||
<div class="row mb-3">
|
||||
<div class="col-12">
|
||||
<label for="benchmark_description">Benchmark description</label>
|
||||
<textarea id="benchmark_description" class="form-control" name="benchmark_description"><%= benchmark.description %></textarea>
|
||||
<textarea id="benchmark_description" class="form-control" name="benchmark_description" placeholder="Enter a description/notes here."><%= benchmark.description %></textarea>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
|
@@ -17,7 +17,7 @@
|
||||
<% tests.each do |t| %>
|
||||
<tr>
|
||||
<td><a href="/test/<%= t.id %>"><%= t.name %></a></td>
|
||||
<td><%= t.benchmark.length %></td>
|
||||
<td><%= t.benchmarks.length %></td>
|
||||
<td><%= t.updated_at %></td>
|
||||
</tr>
|
||||
<% end %>
|
||||
|
@@ -10,16 +10,17 @@
|
||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.7.1/jquery.min.js" charset="utf-8"></script>
|
||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.3.7/js/bootstrap.min.js" charset="utf-8"></script>
|
||||
<script src="/js/edgeville.js" charset="utf-8"></script>
|
||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.9.4/Chart.min.js" charset="utf-8"></script>
|
||||
</head>
|
||||
<body>
|
||||
<body class="d-flex flex-column min-vh-100">
|
||||
<!-- main navigation -->
|
||||
<%= erb :'partials/navbar', :locals => locals %>
|
||||
|
||||
<!-- main content -->
|
||||
<main class="flex-grow-1 py-4">
|
||||
<div id="wrapper" class="container mb-4">
|
||||
<%= yield %>
|
||||
</div>
|
||||
</main>
|
||||
|
||||
<!-- site footer -->
|
||||
<%= erb :'partials/footer', :locals => locals %>
|
||||
|
@@ -1,8 +1,10 @@
|
||||
<nav id="main-footer" class="container">
|
||||
<footer id="main-footer" class="bg-light border-top py-3 text-center">
|
||||
<div class="container">
|
||||
<div class="row">
|
||||
<div class="col-12">
|
||||
<p>Game Data version v<%= app_version() %></p>
|
||||
<p>Running Ruby version v<%= ruby_version() %></p>
|
||||
<p class="mb-0">Running Ruby version v<%= ruby_version() %></p>
|
||||
</div>
|
||||
</div>
|
||||
</nav>
|
||||
</div>
|
||||
</footer>
|
@@ -1,15 +1,21 @@
|
||||
<div class="row">
|
||||
<div class="col-12">
|
||||
<h1>Generate report</h1>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
<form class="col-12" action="/reports" method="post">
|
||||
<div class="row mb-3">
|
||||
<div class="col-12 col-md-6">
|
||||
<select id="report_type" class="form-select" name="report_type" disabled>
|
||||
<select id="report-type" class="form-select" name="report_type" disabled>
|
||||
<option value="benchmark">Benchmark</option>
|
||||
<option value="test">Test</option>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-6">
|
||||
<select id="report_choice" class="form-select" name="report_choice">
|
||||
<select id="report-benchmarks" class="form-select" name="report_choice">
|
||||
<% benchmarks.each do |b| %>
|
||||
<option value="<%= b.id %>"><%= b.name %></option>
|
||||
<% end %>
|
||||
@@ -18,7 +24,7 @@
|
||||
</div>
|
||||
|
||||
<div class="col-12 mb-3">
|
||||
<select id="report_compare" class="col-12 form-select" name="report_compare[]" multiple>
|
||||
<select id="report-tests" class="col-12 form-select" name="report_compare[]" multiple>
|
||||
<% tests.each do |t| %>
|
||||
<option value="<%= t.id %>"><%= t.name %></option>
|
||||
<% end %>
|
||||
@@ -26,16 +32,18 @@
|
||||
</div>
|
||||
|
||||
<div class="col-12 mb-3">
|
||||
<input type="submit" class="btn btn-primary" id="generate_button" value="Generate">
|
||||
<a href="#" class="btn btn-primary" id="download_button" disabled>Download</a>
|
||||
<button id="reports-button" class="btn btn-primary">Generate</button>
|
||||
<button id="reports-download" class="btn btn-primary" disabled>Download</button>
|
||||
</div>
|
||||
|
||||
<div class="col-12">
|
||||
<canvas id="chart_canvas" width="100%" height="25"></canvas>
|
||||
<canvas id="benchmark-chart" width="100%" height="25"></canvas>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<!-- load the chart.js library -->
|
||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/4.4.1/chart.umd.min.js" charset="utf-8"></script>
|
||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/chartjs-plugin-datalabels/2.2.0/chartjs-plugin-datalabels.min.js" charset="utf-8"></script>
|
||||
<!-- load chart functionality -->
|
||||
<script src="/js/reports.js" charset="utf-8"></script>
|
||||
|
@@ -1,27 +1,44 @@
|
||||
<div class="row">
|
||||
<div class="row mb-3">
|
||||
<div class="col-12">
|
||||
<h1>Editing: <%= hardware.name %></h1>
|
||||
<h1>Editing: <%= test.name %></h1>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
|
||||
<form class="col-12" action="/hardware/<%= hardware.id %>/edit" method="post">
|
||||
<div class="row mb-3">
|
||||
<div class="col-12 col-md-9">
|
||||
<label for="hardware_name">Hardware name</label>
|
||||
<input id="hardware_name" class="form-control" type="text" name="hardware_name" placeholder="Example hardware" value="<%= hardware.name %>">
|
||||
<form class="col-12" action="/test/<%= test.id %>/edit" method="post">
|
||||
<div class="row mb-0 mb-md-3">
|
||||
<div class="col-12 col-md-6 mb-3 mb-md-0">
|
||||
<label for="test_name">Test name</label>
|
||||
<input id="test_name" class="form-control" type="text" name="test_name" placeholder="My hardware test (01/99)" value="<%= test.name %>">
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-3">
|
||||
<label for="hardware_type">Type</label>
|
||||
<select id="hardware_type" class="form-control" name="hardware_type">
|
||||
<option value="gpu" <% if hardware.type == 'gpu' %>selected<% end %>>Graphics card</option>
|
||||
<option value="cpu" <% if hardware.type == 'cpu' %>selected<% end %>>Processor</option>
|
||||
<div class="col-12 col-md-6 mb-3 mb-md-0">
|
||||
<label for="test_hardware">Hardware to test</label>
|
||||
<select id="test_hardware" class="form-select" name="test_hardware">
|
||||
<% for h in hardware %>
|
||||
<option value="<%= h.id %>" <% if h.id == test.hardware.id %>selected<% end %>><%= h.name %></option>
|
||||
<% end %>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row mb-0 mb-md-3">
|
||||
<div class="col-12 col-md-4 mb-3 mb-md-0">
|
||||
<label for="test_benchmarks">Benchmarks</label>
|
||||
<select id="test_benchmarks" class="form-select" name="test_benchmarks[]" multiple>
|
||||
<% for b in benchmarks %>
|
||||
<option value="<%= b.id %>" <% if test.benchmark?(b.id) %>selected<% end %>><%= b.name %></option>
|
||||
<% end %>
|
||||
</select>
|
||||
</div>
|
||||
|
||||
<div class="col-12 col-md-8 mb-3 mb-md-0">
|
||||
<label for="test_description">Test description</label>
|
||||
<textarea id="test_description" class="form-control" name="test_description" placeholder="This is my test for a hardware..."><%= test.description %></textarea>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
<div class="col-12">
|
||||
<input class="btn btn-primary w-100" type="submit" value="Submit Changes">
|
||||
|
@@ -8,7 +8,7 @@
|
||||
</div>
|
||||
|
||||
<div class="col-12">
|
||||
<p>Hardware tested: <%= test.hardware.name %></p>
|
||||
<p>Hardware tested: <a href="/hardware/<%= test.hardware.id %>"><%= test.hardware.name %></a></p>
|
||||
</div>
|
||||
|
||||
<div class="col-12">
|
||||
|
Reference in New Issue
Block a user