scinote-web/lib/tasks/db_fake_data.rake
2017-03-08 13:18:20 +01:00

1340 lines
45 KiB
Ruby

require "#{Rails.root}/app/utilities/users_generator"
require "#{Rails.root}/app/utilities/renaming_util"
require "#{Rails.root}/test/helpers/fake_test_helper"
include UsersGenerator
include RenamingUtil
include FakeTestHelper
namespace :db do
NR_TEAMS = 4
NR_USERS = 100
NR_SAMPLE_TYPES = 20
NR_SAMPLE_GROUPS = 20
NR_CUSTOM_FIELDS = 20
NR_SAMPLES = 100
NR_PROTOCOLS = 20
NR_PROTOCOL_KEYWORDS = 20
NR_PROJECTS = 3
NR_EXPERIMENTS = 4
NR_MODULE_GROUPS = 4
NR_MODULES = 4
NR_STEPS = 3
NR_RESULTS = 4
NR_REPORTS = 4
NR_COMMENTS = 10
RATIO_USER_TEAMS = 0.5
NR_MAX_USER_TEAMS = 20
RATIO_CUSTOM_FIELDS = 0.7
RATIO_SAMPLE_CUSTOM_FIELDS = 0.6
RATIO_PROTOCOL_KEYWORDS = 0.3
THRESHOLD_PROTOCOL_IN_MODULE_LINKED = 0.5
THRESHOLD_PROTOCOL_PUBLIC = 0.6
THRESHOLD_PROTOCOL_ARCHIVED = 0.2
NR_MAX_USER_PROJECTS = 15
RATIO_USER_PROJECTS = 0.5
RATIO_COMMENTS = 0.7
NR_MAX_USER_MODULES = 15
RATIO_USER_MODULES = 0.5
NR_MAX_SAMPLE_MODULES = 10
RATIO_SAMPLE_MODULES = 0.7
RATIO_MODULE_MODULE_GROUPS = 0.8
RATIO_EDGES = 0.7
RATIO_STEP_COMPLETED = 0.5
NR_MAX_STEP_ATTACHMENTS = 2
RATIO_STEP_ATTACHMENTS = 0.2
NR_MAX_CHECKLIST_ITEMS = 20
RATIO_CHECKLIST_ITEM = 0.2
RATIO_CHECKLIST_ITEM_CHECKED = 0.5
RATIO_RESULT_ARCHIVED = 0.2
RATIO_REPORT_ELEMENTS = 0.75
THRESHOLD_ARCHIVED = 0.2
THRESHOLD_RESTORED = 0.9
MIN_FILE_SIZE = 0.01
MAX_FILE_SIZE = 0.1
desc "Drops the database, sets it up and inserts fake data for " +
"the current RAILS_ENV. WARNING: THIS WILL ERASE ALL " +
"CURRENT DATA IN THE DATABASE."
task :fake => :environment do
Rake::Task["db:drop"].reenable
Rake::Task["db:drop"].invoke
Rake::Task["db:setup"].reenable
Rake::Task["db:setup"].invoke
Rake::Task["db:fake:generate"].reenable
Rake::Task["db:fake:generate"].invoke
end
namespace :fake do
desc "Generates fake data & inserts it into database for the " +
"current RAILS_ENV."
task :generate => :environment do
require 'rgl/base'
require 'rgl/adjacency'
require 'rgl/topsort'
puts "Verbose? (Y/n)"
res = $stdin.gets.to_s.downcase.strip
unless res.in?(["", "y", "n"]) then
puts "Invalid parameter, exiting"
return
end
verbose = res.in?(["", "y"])
puts "Simple seeding? (Y/n)"
res = $stdin.gets.to_s.downcase.strip
unless res.in?(["", "y", "n"]) then
puts "Invalid parameter, exiting"
return
end
simple = res.in?(["", "y"])
if simple
puts "Choose the size of generated dataset(T - tiny, " +
"s - small, m - medium, l - large, h -huge)"
res = $stdin.gets.to_s.downcase.strip
unless res.in?(["", "t", "s", "m", "l", "h"]) then
puts "Invalid parameter, exiting"
return
end
case res
when "", "t"
factor = 0.5
when "s"
factor = 1
when "m"
factor = 5
when "l"
factor = 20
when "h"
factor = 100
end
nr_team = NR_TEAMS * factor
nr_users = NR_USERS * factor
nr_sample_types = NR_SAMPLE_TYPES * factor
nr_sample_groups = NR_SAMPLE_GROUPS * factor
nr_custom_fields = NR_CUSTOM_FIELDS * factor
nr_samples = NR_SAMPLES * factor
nr_protocols = NR_PROTOCOLS * factor
nr_protocol_keywords = NR_PROTOCOL_KEYWORDS * factor
nr_projects = NR_PROJECTS * factor
nr_experiments = NR_EXPERIMENTS * factor
nr_module_groups = NR_MODULE_GROUPS * factor
nr_modules = NR_MODULES * factor
nr_steps = NR_STEPS * factor
nr_results = NR_RESULTS * factor
nr_reports = NR_REPORTS * factor
nr_comments = NR_COMMENTS * factor
else
puts "Type in the number of seeded teams"
nr_team = $stdin.gets.to_i
puts "Type in the number of seeded users"
nr_users = $stdin.gets.to_i
puts "Type in the number of seeded sample types " +
"for each team"
nr_sample_types = $stdin.gets.to_i
puts "Type in the number of seeded sample groups for " +
"each team"
nr_sample_groups = $stdin.gets.to_i
puts "Type in the max. number of seeded custom fields " +
"for each team"
nr_custom_fields = $stdin.gets.to_i
puts "Type in the number of seeded samples for each team"
nr_samples = $stdin.gets.to_i
puts "Type in the number of seeded protocols for each team"
nr_protocols = $stdin.gets.to_i
puts "Type in the number of seeded protocol keywords for each team"
nr_protocol_keywords = $stdin.gets.to_i
puts "Type in the number of seeded projects for each team"
nr_projects = $stdin.gets.to_i
puts "Type in the number of seeded experiments for each project"
nr_experiments = $stdin.gets.to_i
puts "Type in the number of seeded workflows for each experiment"
nr_module_groups = $stdin.gets.to_i
puts "Type in the number of seeded modules for each experiment"
nr_modules = $stdin.gets.to_i
puts "Type in the number of seeded steps for each module"
nr_steps = $stdin.gets.to_i
puts "Type in the number of seeded results for each module"
nr_results = $stdin.gets.to_i
puts "Type in the number of seeded reports for each project"
nr_reports = $stdin.gets.to_i
puts "Type in the max. number of seeded comments for each " +
"commentable item"
nr_comments = $stdin.gets.to_i
end
begin
ActiveRecord::Base.transaction do
puts "Generating fake teams..."
taken_team_names = []
for _ in 1..nr_team
begin
name = Faker::University.name
end while name.in? taken_team_names
taken_team_names << name
Team.create(
name: name,
description: rand >= 0.7 ? Faker::Lorem.sentence : nil
)
end
all_teams = Team.all
puts "Generating fake users..."
taken_emails = []
for _ in 1..nr_users
begin
if rand >= 0.8
name = generate_got_name
else
name = Faker::Name.name
end
email_name = name.downcase.remove(".").split(" ").join(".")
password = Faker::Internet.password(10, 20)
email = Faker::Internet.free_email(email_name)
end while email.in? taken_emails
taken_emails << email
user = create_user(
name,
email,
password,
true,
nil,
[]
)
user.update(
confirmed_at: Faker::Date.backward(30),
)
if verbose then
puts " Generated user #{name} (email: #{email}, " +
"password: #{password})"
end
# Randomly assign user to teams
taken_team_ids = []
for _ in 1..[NR_MAX_USER_TEAMS, all_teams.count].min
if rand <= RATIO_USER_TEAMS then
begin
team = pluck_random(all_teams)
end while team.id.in? taken_team_ids
taken_team_ids << team.id
UserTeam.create(
user: user,
team: team,
role: rand(0..2)
)
end
end
end
puts "Generating fake sample types..."
all_teams.find_each do |team|
for _ in 1..nr_sample_types
SampleType.create(
name: Faker::Commerce.department(4),
team: team
)
end
end
puts "Generating fake sample groups..."
all_teams.find_each do |team|
for _ in 1..nr_sample_groups
SampleGroup.create(
name: Faker::Commerce.color,
team: team,
color: generate_color
)
end
end
puts "Generating fake custom fields..."
all_teams.find_each do |team|
for _ in 1..nr_custom_fields
if rand <= RATIO_CUSTOM_FIELDS then
CustomField.create(
name: Faker::Team.state,
team: team,
user: pluck_random(team.users)
)
end
end
end
puts "Generating fake samples..."
all_teams.find_each do |team|
for _ in 1..nr_samples
sample = Sample.create(
name: Faker::Book.title,
team: team,
user: pluck_random(team.users),
sample_type: pluck_random(team.sample_types),
sample_group: pluck_random(team.sample_groups)
)
# Add some custom fields to sample
team.custom_fields.find_each do |cf|
if rand <= RATIO_SAMPLE_CUSTOM_FIELDS then
SampleCustomField.create(
sample: sample,
custom_field: cf,
value: Faker::Team.state
)
end
end
end
end
puts "Generating fake protocol keywords"
all_teams.find_each do |team|
taken_kw_names = []
for _ in 1..nr_protocol_keywords
begin
name = Faker::Book.genre
end while name.in? taken_kw_names
taken_kw_names << name
ProtocolKeyword.create(
team: team,
name: name
)
end
end
puts "Generating fake repository protocols..."
all_teams.find_each do |team|
for _ in 1..nr_protocols
protocol = generate_fake_protocol(team,
nil,
nr_steps,
nr_comments)
if verbose then
puts " Generated protocol #{protocol.name}"
end
end
end
puts "Generating fake projects..."
all_teams.find_each do |team|
taken_project_names = []
for _ in 1..nr_projects
begin
name = Faker::Company.name[0..29]
end while name.in? taken_project_names
taken_project_names << name
author = pluck_random(team.users)
created_at = Faker::Time.backward(500)
last_modified_by = pluck_random(team.users)
archived_by = pluck_random(team.users)
archived_on = Faker::Time.between(created_at, DateTime.now)
restored_by = pluck_random(team.users)
restored_on = Faker::Time.between(archived_on, DateTime.now)
status = random_status
project = Project.create(
visibility: rand(0..1),
name: name,
due_date: nil,
team: team,
created_by: author,
created_at: created_at,
last_modified_by: last_modified_by,
archived: status == :archived,
archived_by: status.in?([:archived, :restored]) ?
archived_by : nil,
archived_on: status.in?([:archived, :restored]) ?
archived_on : nil,
restored_by: status == :restored ? restored_by : nil,
restored_on: status == :restored ? restored_on : nil
)
# Automatically assign project author onto project
UserProject.create(
user: author,
project: project,
role: 0,
created_at: created_at
)
# Activities
Activity.create(
type_of: :create_project,
user: author,
project: project,
message: I18n.t(
"activities.create_project",
user: author.full_name,
project: project.name
),
created_at: created_at
)
if status.in?([:archived, :restored]) then
Activity.create(
type_of: :archive_project,
user: archived_by,
project: project,
message: I18n.t(
"activities.archive_project",
user: archived_by.full_name,
project: project.name
),
created_at: archived_on
)
end
if status == :restored then
Activity.create(
type_of: :restore_project,
user: restored_by,
project: project,
message: I18n.t(
"activities.restore_project",
user: restored_by.full_name,
project: project.name
),
created_at: restored_on
)
end
# Assign users onto the project
taken_user_ids = []
for _ in 2..[NR_MAX_USER_PROJECTS, team.users.count].min
if rand <= RATIO_USER_PROJECTS
begin
user = pluck_random(team.users)
end while user.id.in? taken_user_ids
taken_user_ids << user.id
assigned_on = Faker::Time.backward(500)
assigned_by = pluck_random(project.users)
up = UserProject.create(
user: user,
project: project,
role: rand(0..3),
created_at: assigned_on,
assigned_by: assigned_by
)
Activity.create(
type_of: :assign_user_to_project,
user: assigned_by,
project: project,
message: I18n.t(
"activities.assign_user_to_project",
assigned_user: user.full_name,
role: up.role_str,
project: project.name,
assigned_by_user: assigned_by.full_name
),
created_at: assigned_on
)
end
end
# Add some comments
for _ in 1..nr_comments
generate_fake_project_comment(project) if rand <= RATIO_COMMENTS
end
end
end
puts "Generating fake experiments..."
Project.find_each do |project|
for _ in 1..nr_experiments
status = random_status
created_at = Faker::Time.backward(500)
archived_by = pluck_random(project.users)
archived_on = Faker::Time.between(created_at, DateTime.now)
restored_by = pluck_random(project.users)
restored_on = Faker::Time.between(archived_on, DateTime.now)
author = pluck_random(project.users)
Experiment.create(
name: Faker::Hacker.noun,
description: Faker::Hipster.sentence,
project: project,
created_at: created_at,
created_by: author,
last_modified_by: author,
archived: status.in?([:active, :restored]),
archived_on: status.in?([:archived, :restored]) ?
archived_on : nil,
archived_by: status.in?([:archived, :restored]) ?
archived_by : nil,
restored_on: status == :restored ? restored_on : nil,
restored_by: status == :restored ? restored_by : nil
)
end
end
puts "Generating fake workflows..."
Experiment.find_each do |experiment|
for _ in 1..nr_module_groups
MyModuleGroup.create(
name: Faker::Hacker.noun,
experiment: experiment
)
end
end
puts "Generating fake modules..."
total_experiment = Experiment.count
Experiment.find_each.with_index do |experiment, i|
if verbose then
puts " Generating modules for experiment #{experiment.name} " +
"(#{i + 1} of #{total_experiment})..."
end
project = experiment.project
taken_pos = []
for _ in 1..nr_modules
begin
x = rand(0..nr_modules) * 32
y = rand(0..nr_modules) * 16
end while [x, y].in? taken_pos
taken_pos << [x, y]
status = random_status
author = pluck_random(team.users)
created_at = Faker::Time.backward(500)
archived_by = pluck_random(team.users)
archived_on = Faker::Time.between(created_at, DateTime.now)
restored_by = pluck_random(team.users)
restored_on = Faker::Time.between(archived_on, DateTime.now)
my_module = MyModule.create(
name: Faker::Hacker.verb,
created_by: author,
created_at: created_at,
due_date: rand <= 0.5 ?
Faker::Time.forward(500) : nil,
description: rand <= 0.5 ?
Faker::Hacker.say_something_smart : nil,
x: x,
y: y,
experiment: experiment,
my_module_group: status == :archived ?
nil :
(rand <= RATIO_MODULE_MODULE_GROUPS ?
pluck_random(experiment.my_module_groups) : nil
),
archived: status == :archived,
archived_on: status.in?([:archived, :restored]) ?
archived_on : nil,
archived_by: status.in?([:archived, :restored]) ?
archived_by : nil,
restored_on: status == :restored ? restored_on : nil,
restored_by: status == :restored ? restored_by : nil
)
# Activities
Activity.create(
type_of: :create_module,
user: author,
project: my_module.experiment.project,
my_module: my_module,
message: I18n.t(
"activities.create_module",
user: author.full_name,
module: my_module.name
),
created_at: created_at
)
if status.in?([:archived, :restored]) then
Activity.create(
type_of: :archive_module,
user: archived_by,
project: my_module.experiment.project,
my_module: my_module,
message: I18n.t(
"activities.archive_module",
user: archived_by.full_name,
module: my_module.name
),
created_at: archived_on
)
end
if status == :restored then
Activity.create(
type_of: :restore_module,
user: restored_by,
project: my_module.experiment.project,
my_module: my_module,
message: I18n.t(
"activities.restore_module",
user: restored_by.full_name,
module: my_module.name
),
created_at: restored_on
)
end
if verbose then
puts " Generated module #{my_module.name}"
end
# Assign some users onto module
taken_user_ids = []
for _ in 1..[NR_MAX_USER_MODULES, project.users.count].min
if rand <= RATIO_USER_MODULES then
begin
user = pluck_random(project.users)
end while user.id.in? taken_user_ids
taken_user_ids << user.id
assigned_on = Faker::Time.backward(500)
assigned_by = pluck_random(my_module.experiment.project.users)
UserMyModule.create(
user: user,
my_module: my_module,
assigned_by: pluck_random(project.users),
created_at: assigned_on
)
Activity.create(
type_of: :assign_user_to_module,
user: assigned_by,
project: my_module.experiment.project,
my_module: my_module,
message: I18n.t(
"activities.assign_user_to_module",
assigned_user: user.full_name,
module: my_module.name,
assigned_by_user: assigned_by.full_name
),
created_at: assigned_on
)
end
end
# Assign some samples onto module
taken_sample_ids = []
for _ in 1..[
NR_MAX_SAMPLE_MODULES,
project.team.samples.count
].min
if rand <= RATIO_SAMPLE_MODULES then
begin
sample = pluck_random(project.team.samples)
end while sample.id.in? taken_sample_ids
taken_sample_ids << sample.id
SampleMyModule.create(
sample: sample,
my_module: my_module
)
end
end
# Add some comments
for _ in 1..nr_comments
generate_fake_module_comment(my_module) if rand <= RATIO_COMMENTS
end
end
# Generate some connections between modules
experiment.my_module_groups.find_each do |my_module_group|
if my_module_group.my_modules.empty? or
my_module_group.my_modules.count == 1
# If any module group doesn't contain
# any modules (or has only 1 module), remove it
my_module_group.destroy
else
# Make connections between project modules,
# keeping in mind not to generate cycles
n = my_module_group.my_modules.count
max_edges = (n - 1) * n / 2
dg = RGL::DirectedAdjacencyGraph.new
for _ in 1..max_edges
if rand <= RATIO_EDGES
begin
m1 = pluck_random(my_module_group.my_modules)
m2 = pluck_random(my_module_group.my_modules)
end while (
m1 == m2 or
dg.has_edge?(m1.id, m2.id)
)
# Only add edge if it won't make graph cyclic
dg.add_edge(m1.id, m2.id)
if dg.acyclic?
Connection.create(
input_id: m1.id,
output_id: m2.id
)
else
dg.remove_edge(m1.id, m2.id)
end
end
end
# Set order number for each module in group
topsort = dg.topsort_iterator.to_a
my_module_group.my_modules.each do |mm|
if topsort.include? mm.id
mm.workflow_order = topsort.find_index(mm.id)
mm.save!
end
end
end
end
end
puts 'Generating fake module protocols...'
Experiment.find_each do |experiment|
experiment.my_modules.find_each do |my_module|
generate_fake_protocol(experiment.project.team,
my_module,
nr_steps,
nr_comments)
end
end
puts "Generating fake module results..."
Experiment.find_each do |experiment|
experiment.my_modules.find_each do |my_module|
for _ in 1..nr_results
user = pluck_random(my_module.users)
created_at = Faker::Time.backward(500)
archived_on = Faker::Time.between(created_at, DateTime.now)
restored_on = Faker::Time.between(archived_on, DateTime.now)
status = random_status
result = Result.new(
name: Faker::Hacker.abbreviation[0..49],
my_module: my_module,
user: user,
created_at: created_at,
archived: status == :archived,
archived_by: status.in?([:archived, :restored]) ?
user : nil,
archived_on: status.in?([:archived, :restored]) ?
archived_on : nil,
restored_by: status == :restored ? user : nil,
restored_on: status == :restored ? restored_on : nil
)
type = [:text, :asset, :table][rand(0..2)]
case type
when :text
result.result_text = ResultText.new(
text: Faker::Hipster.paragraph,
)
str = "activities.add_text_result"
str2 = "activities.archive_text_result"
when :asset
result.asset = Asset.new(
file: generate_file(rand(MIN_FILE_SIZE..MAX_FILE_SIZE)),
created_by: result.user
)
str = "activities.add_asset_result"
str2 = "activities.archive_asset_result"
when :table
result.table = Table.new(
contents: generate_table_contents(rand(30), rand(150)),
created_by: result.user
)
str = "activities.add_table_result"
str2 = "activities.archive_table_result"
end
result.save
# Add activities
Activity.create(
type_of: :add_result,
project: experiment.project,
my_module: my_module,
user: user,
created_at: created_at,
message: I18n.t(
str,
user: user.full_name,
result: result.name
)
)
if status.in?([:archived, :restored]) then
Activity.create(
type_of: :archive_result,
user: user,
project: experiment.project,
my_module: my_module,
message: I18n.t(
str2,
user: user.full_name,
result: result.name
),
created_at: archived_on
)
end
# Currently, there is no way to restore archived results
# Add some comments
for _ in 1..nr_comments
generate_fake_result_comment(result) if rand <= RATIO_COMMENTS
end
end
end
end
puts "Generating fake reports..."
Experiment.find_each do |experiment|
project = experiment.project
for _ in 1..nr_reports
taken_project_names = []
begin
name = Faker::Company.bs
user = pluck_random(project.users)
end while [user, name].in? taken_project_names
taken_project_names << [user, name]
report = Report.create(
name: name,
description: Faker::Hipster.sentence,
project: project,
user: user
)
# Generate the oh-so-many report elements
ReportElement.create(
sort_order: 0,
position: 0,
report: report,
type_of: :project_header
)
experiment.my_modules.each do |my_module|
if rand <= RATIO_REPORT_ELEMENTS then
re_my_module = ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: :my_module,
my_module: my_module
)
my_module.protocol.completed_steps.each do |step|
if rand <= RATIO_REPORT_ELEMENTS then
re_step = ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: :step,
step: step,
parent: re_my_module
)
step.checklists.each do |checklist|
if rand <= RATIO_REPORT_ELEMENTS then
ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: :step_checklist,
checklist: checklist,
parent: re_step
)
end
end
step.assets.each do |asset|
if rand <= RATIO_REPORT_ELEMENTS then
ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: :step_asset,
asset: asset,
parent: re_step
)
end
end
step.tables.each do |table|
if rand <= RATIO_REPORT_ELEMENTS then
ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: :step_table,
table: table,
parent: re_step
)
end
end
if rand <= RATIO_REPORT_ELEMENTS then
ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: :step_comments,
step: step,
parent: re_step
)
end
end
end
my_module.results.each do |result|
if rand <= RATIO_REPORT_ELEMENTS then
if result.is_asset
type_of = :result_asset
elsif result.is_table
type_of = :result_table
else
type_of = :result_text
end
re_result = ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: type_of,
result: result,
parent: re_my_module
)
if rand <= RATIO_REPORT_ELEMENTS then
ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: :result_comments,
result: result,
parent: re_result
)
end
end
end
if rand <= RATIO_REPORT_ELEMENTS then
ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: :my_module_activity,
my_module: my_module,
parent: re_my_module
)
end
if rand <= RATIO_REPORT_ELEMENTS then
ReportElement.create(
sort_order: rand <= 0.5 ? 0 : 1,
position: 0,
report: report,
type_of: :my_module_samples,
my_module: my_module,
parent: re_my_module
)
end
end
end
# Shuffle the report
shuffle_report_elements(
report.report_elements.where(parent: nil)
)
end
end
end
# Now, at the end, add additional "private" team
# to each user
User.find_each do |user|
create_private_user_team(user, Constants::DEFAULT_PRIVATE_TEAM_NAME)
end
# Generate thumbnails of all experiments
Experiment.find_each do |experiment|
experiment.generate_workflow_img
end
# Calculate space taken by each team; this must
# be done in a separate transaction because the estimated
# asset sizes are calculated in after_commit, which is done
# after the first transaction is completed
ActiveRecord::Base.transaction do
puts 'Calculating team sizes...'
Team.find_each do |team|
team.calculate_space_taken
team.save
end
end
rescue ActiveRecord::ActiveRecordError,
ArgumentError, ActiveRecord::RecordNotSaved => e
puts "Error seeding fake data, transaction reverted"
puts "Output: #{e.inspect}"
puts e.backtrace.join("\n")
end
end
end
def generate_fake_protocol(
team,
my_module,
nr_steps,
nr_comments
)
protocol = nil
if my_module.present?
protocol = my_module.protocol
users = my_module.experiment.project.users
author = pluck_random(users)
if rand <= THRESHOLD_PROTOCOL_IN_MODULE_LINKED &&
(parent = pluck_random(
team.protocols.where(protocol_type: [
Protocol.protocol_types[:in_repository_private],
Protocol.protocol_types[:in_repository_public]
]))
).present?
protocol.protocol_type = :linked
protocol.added_by = author
protocol.parent = parent
protocol.parent_updated_at = parent.updated_at
else
protocol.protocol_type = :unlinked
end
protocol.my_module = my_module
else
protocol = Protocol.new
users = team.users
author = pluck_random(users)
val = rand
if val > THRESHOLD_PROTOCOL_ARCHIVED
if val > THRESHOLD_PROTOCOL_PUBLIC
protocol.protocol_type = :in_repository_public
protocol.published_on = Faker::Time.backward(500)
else
protocol.protocol_type = :in_repository_private
end
else
protocol.protocol_type = :in_repository_archived
protocol.archived_by = author
protocol.archived_on = Faker::Time.backward(500)
end
protocol.added_by = author
protocol.authors = Faker::Book.author
protocol.description = Faker::Lorem.paragraph(2)
end
protocol.name = Faker::Hacker.ingverb
protocol.team = team
if protocol.invalid? then
rename_record(protocol, :name)
end
protocol.save!
team.protocol_keywords.find_each do |kw|
if rand <= RATIO_PROTOCOL_KEYWORDS
ProtocolProtocolKeyword.create(
protocol: protocol,
protocol_keyword: kw
)
end
end
protocol.reload
if protocol.linked?
# For linked protocols, simply copy their parents' contents
protocol.load_from_repository(protocol.parent, author)
else
# Generate fake protocol data
for i in 1..nr_steps
created_at = Faker::Time.backward(500)
completed = protocol.in_repository? ? false : (rand <= RATIO_STEP_COMPLETED)
completed_on = completed ?
Faker::Time.between(created_at, DateTime.now) : nil
step = Step.create(
created_at: created_at,
name: Faker::Hacker.ingverb,
description: Faker::Hacker.say_something_smart,
position: i - 1,
completed: completed,
user: pluck_random(users),
protocol: protocol,
completed_on: completed_on
)
if protocol.in_module?
Activity.create(
type_of: :create_step,
project: my_module.experiment.project,
my_module: my_module,
user: step.user,
created_at: created_at,
message: I18n.t(
"activities.create_step",
user: step.user.full_name,
step: i,
step_name: step.name
)
)
end
if completed then
Activity.create(
type_of: :complete_step,
project: my_module.experiment.project,
my_module: my_module,
user: step.user,
created_at: completed_on,
message: I18n.t(
"activities.complete_step",
user: step.user.full_name,
step: i,
step_name: step.name,
completed: i,
all: i
)
)
end
# Add checklists
for _ in 1..NR_MAX_STEP_ATTACHMENTS
if rand <= RATIO_STEP_ATTACHMENTS then
checklist = Checklist.create(
name: Faker::Hacker.noun,
step: step,
created_by: step.user,
)
# Add checklist items
for j in 1..NR_MAX_CHECKLIST_ITEMS
if rand <= RATIO_CHECKLIST_ITEM then
checked = protocol.in_repository? ? false : (rand <= RATIO_CHECKLIST_ITEM_CHECKED)
checked_on = Faker::Time.backward(500)
ci = ChecklistItem.create(
created_at: checked_on,
text: Faker::Hipster.sentence,
checklist: checklist,
checked: checked,
created_by: step.user
)
if checked then
Activity.create(
type_of: :check_step_checklist_item,
project: my_module.experiment.project,
my_module: my_module,
user: step.user,
created_at: checked_on,
message: I18n.t(
"activities.check_step_checklist_item",
user: step.user.full_name,
checkbox: ci.text,
completed: j,
all: j,
step: i,
step_name: step.name
)
)
end
end
end
end
end
# Add assets
for _ in 1..NR_MAX_STEP_ATTACHMENTS
if rand <= RATIO_STEP_ATTACHMENTS then
asset = Asset.create(
file: generate_file(rand(MIN_FILE_SIZE..MAX_FILE_SIZE)),
estimated_size: 0,
file_present: true,
created_by: step.user
)
StepAsset.create(
step: step,
asset: asset
)
end
end
# Add tables
for _ in 1..NR_MAX_STEP_ATTACHMENTS
if rand <= RATIO_STEP_ATTACHMENTS then
table = Table.create(
contents:
generate_table_contents(rand(30), rand(150)),
created_by: step.user
)
StepTable.create(
step: step,
table: table
)
end
end
# Add some comments (only on protocols on module)
if protocol.in_module? then
for _ in 1..nr_comments
if rand <= RATIO_COMMENTS
user = pluck_random(users)
created_at = Faker::Time.backward(500)
StepComment.create(
user: user,
message: Faker::Hipster.sentence,
created_at: created_at,
step: step
)
Activity.create(
type_of: :add_comment_to_step,
project: my_module.experiment.project,
my_module: my_module,
user: user,
created_at: created_at,
message: I18n.t(
"activities.add_comment_to_step",
user: user.full_name,
step: i,
step_name: step.name
)
)
end
end
end
end
end
return protocol.reload
end
def shuffle_report_elements(report_elements)
if report_elements.blank? or report_elements.count == 0
return
end
header = report_elements.find_by(type_of: :project_header)
if header.present?
header.position = 0
header.save
i = 1
else
i = 0
end
ids_map = (i..(report_elements.count - 1)).to_a.shuffle
for i in i..(report_elements.count - 1)
re = report_elements[i]
re.position = ids_map[i]
re.save
end
# Recursively shuffle children
report_elements.each do |re2|
if re2.children.count > 0
shuffle_report_elements(re2.children)
end
end
end
def generate_fake_project_comment(project)
user = pluck_random(project.users)
created_at = Faker::Time.backward(500)
ProjectComment.create(
user: user,
message: Faker::Hipster.sentence,
created_at: created_at,
project: project
)
Activity.create(
type_of: :add_comment_to_project,
user: user,
project: project,
created_at: created_at,
message: I18n.t('activities.add_comment_to_project',
user: user.full_name,
project: project.name)
)
end
def generate_fake_module_comment(my_module)
user = pluck_random(my_module.experiment.project.users)
created_at = Faker::Time.backward(500)
TaskComment.create(
user: user,
message: Faker::Hipster.sentence,
created_at: created_at,
my_module: my_module
)
Activity.create(
type_of: :add_comment_to_module,
user: user,
project: my_module.experiment.project,
my_module: my_module,
created_at: created_at,
message: I18n.t('activities.add_comment_to_module',
user: user.full_name,
module: my_module.name)
)
end
def generate_fake_result_comment(result)
user = pluck_random(result.my_module.experiment.project.users)
created_at = Faker::Time.backward(500)
ResultComment.create(
user: user,
message: Faker::Hipster.sentence,
created_at: created_at,
result: result
)
Activity.create(
type_of: :add_comment_to_result,
project: result.my_module.experiment.project,
my_module: result.my_module,
user: user,
created_at: created_at,
message: I18n.t(
'activities.add_comment_to_result',
user: user.full_name,
result: result.name
)
)
end
def generate_fake_step_comment(step)
user = pluck_random(step.protocol.my_module.experiment.project.users)
created_at = Faker::Time.backward(500)
StepComment.create(
user: user,
message: Faker::Hipster.sentence,
created_at: created_at,
step: step
)
Activity.create(
type_of: :add_comment_to_step,
project: step.protocol.my_module.experiment.project,
my_module: step.protocol.my_module,
user: user,
created_at: created_at,
message: I18n.t(
"activities.add_comment_to_step",
user: user.full_name,
step: step.position + 1,
step_name: step.name
)
)
end
# WARNING: This only works on PostgreSQL
def pluck_random(scope)
scope.order("RANDOM()").first
end
# Randomly determine whether project/module/result is active (0),
# archived (1), or already restored (2)
def random_status
val = rand
status = :active
if val > THRESHOLD_ARCHIVED
if val > THRESHOLD_RESTORED
status = :archived
end
else
status = :restored
end
status
end
end