bonito 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.codeclimate.yml +8 -0
- data/.gitignore +17 -0
- data/.reek.yml +3 -0
- data/.rspec +2 -0
- data/.rubocop.yml +9 -0
- data/.ruby-gemset +1 -0
- data/.travis.yml +30 -0
- data/CODE_OF_CONDUCT.md +74 -0
- data/Gemfile +8 -0
- data/LICENSE.txt +21 -0
- data/README.md +159 -0
- data/Rakefile +15 -0
- data/_config.yml +1 -0
- data/bin/console +16 -0
- data/bin/setup +8 -0
- data/bonito.gemspec +45 -0
- data/lib/bonito.rb +8 -0
- data/lib/bonito/moment.rb +39 -0
- data/lib/bonito/parallel_timeline.rb +93 -0
- data/lib/bonito/progress.rb +91 -0
- data/lib/bonito/runner.rb +45 -0
- data/lib/bonito/scope.rb +73 -0
- data/lib/bonito/serial_timeline.rb +330 -0
- data/lib/bonito/timeline.rb +94 -0
- data/lib/bonito/version.rb +5 -0
- metadata +251 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 41888e2e81cc5c6f383b6fe3526984cd0d85bb874853509af626b880c824e669
|
4
|
+
data.tar.gz: ae72f04e929420aaa6698f0b89fb3ffb4dabc3827ab248d84f3b400ec5c72a95
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 769052dccffcd20cd2fd281217e872c5455c6bc2b26a9dc3b10945a8992767b06dcd91e736dfe723eba49c6d2e007e8fc550197da1e2b8ecb2dc4343338418a4
|
7
|
+
data.tar.gz: 606bfa014aa8ef709f8069a173529f91ef06328dd182242259983c8eaa74ab938be4fc87ffa7e59fc4f461e62e2f9dd3f2dae2167f2d1abd25c033f26c57a886
|
data/.codeclimate.yml
ADDED
data/.gitignore
ADDED
data/.reek.yml
ADDED
data/.rspec
ADDED
data/.rubocop.yml
ADDED
data/.ruby-gemset
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
bonito
|
data/.travis.yml
ADDED
@@ -0,0 +1,30 @@
|
|
1
|
+
env:
|
2
|
+
global:
|
3
|
+
- CC_TEST_REPORTER_ID=7288d4a38203cc17d30f9fb4e9d46399d6e7b6f017572bc833d8da3f38fea48b
|
4
|
+
language: ruby
|
5
|
+
rvm:
|
6
|
+
- 2.5.1
|
7
|
+
- 2.3.0
|
8
|
+
before_install:
|
9
|
+
- gem update --system
|
10
|
+
- gem install bundler
|
11
|
+
before_script:
|
12
|
+
- curl -L https://codeclimate.com/downloads/test-reporter/test-reporter-latest-linux-amd64 > ./cc-test-reporter
|
13
|
+
- chmod +x ./cc-test-reporter
|
14
|
+
- ./cc-test-reporter before-build
|
15
|
+
script:
|
16
|
+
- bundle exec rspec
|
17
|
+
- bundle exec rake rerdoc
|
18
|
+
after_script:
|
19
|
+
- ./cc-test-reporter after-build -t simplecov --exit-code $TRAVIS_TEST_RESULT
|
20
|
+
cache: bundler
|
21
|
+
|
22
|
+
deploy:
|
23
|
+
provider: pages
|
24
|
+
local_dir: docs
|
25
|
+
skip_cleanup: true
|
26
|
+
target_branch: gh-pages
|
27
|
+
github_token: $PERSONAL_ACCESS_TOKEN
|
28
|
+
keep_history: true
|
29
|
+
on:
|
30
|
+
branch: master
|
data/CODE_OF_CONDUCT.md
ADDED
@@ -0,0 +1,74 @@
|
|
1
|
+
# Contributor Covenant Code of Conduct
|
2
|
+
|
3
|
+
## Our Pledge
|
4
|
+
|
5
|
+
In the interest of fostering an open and welcoming environment, we as
|
6
|
+
contributors and maintainers pledge to making participation in our project and
|
7
|
+
our community a harassment-free experience for everyone, regardless of age, body
|
8
|
+
size, disability, ethnicity, gender identity and expression, level of experience,
|
9
|
+
nationality, personal appearance, race, religion, or sexual identity and
|
10
|
+
orientation.
|
11
|
+
|
12
|
+
## Our Standards
|
13
|
+
|
14
|
+
Examples of behavior that contributes to creating a positive environment
|
15
|
+
include:
|
16
|
+
|
17
|
+
* Using welcoming and inclusive language
|
18
|
+
* Being respectful of differing viewpoints and experiences
|
19
|
+
* Gracefully accepting constructive criticism
|
20
|
+
* Focusing on what is best for the community
|
21
|
+
* Showing empathy towards other community members
|
22
|
+
|
23
|
+
Examples of unacceptable behavior by participants include:
|
24
|
+
|
25
|
+
* The use of sexualized language or imagery and unwelcome sexual attention or
|
26
|
+
advances
|
27
|
+
* Trolling, insulting/derogatory comments, and personal or political attacks
|
28
|
+
* Public or private harassment
|
29
|
+
* Publishing others' private information, such as a physical or electronic
|
30
|
+
address, without explicit permission
|
31
|
+
* Other conduct which could reasonably be considered inappropriate in a
|
32
|
+
professional setting
|
33
|
+
|
34
|
+
## Our Responsibilities
|
35
|
+
|
36
|
+
Project maintainers are responsible for clarifying the standards of acceptable
|
37
|
+
behavior and are expected to take appropriate and fair corrective action in
|
38
|
+
response to any instances of unacceptable behavior.
|
39
|
+
|
40
|
+
Project maintainers have the right and responsibility to remove, edit, or
|
41
|
+
reject comments, commits, code, wiki edits, issues, and other contributions
|
42
|
+
that are not aligned to this Code of Conduct, or to ban temporarily or
|
43
|
+
permanently any contributor for other behaviors that they deem inappropriate,
|
44
|
+
threatening, offensive, or harmful.
|
45
|
+
|
46
|
+
## Scope
|
47
|
+
|
48
|
+
This Code of Conduct applies both within project spaces and in public spaces
|
49
|
+
when an individual is representing the project or its community. Examples of
|
50
|
+
representing a project or community include using an official project e-mail
|
51
|
+
address, posting via an official social media account, or acting as an appointed
|
52
|
+
representative at an online or offline event. Representation of a project may be
|
53
|
+
further defined and clarified by project maintainers.
|
54
|
+
|
55
|
+
## Enforcement
|
56
|
+
|
57
|
+
Instances of abusive, harassing, or otherwise unacceptable behavior may be
|
58
|
+
reported by contacting the project team at tomfinill@gmail.com. All
|
59
|
+
complaints will be reviewed and investigated and will result in a response that
|
60
|
+
is deemed necessary and appropriate to the circumstances. The project team is
|
61
|
+
obligated to maintain confidentiality with regard to the reporter of an incident.
|
62
|
+
Further details of specific enforcement policies may be posted separately.
|
63
|
+
|
64
|
+
Project maintainers who do not follow or enforce the Code of Conduct in good
|
65
|
+
faith may face temporary or permanent repercussions as determined by other
|
66
|
+
members of the project's leadership.
|
67
|
+
|
68
|
+
## Attribution
|
69
|
+
|
70
|
+
This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4,
|
71
|
+
available at [http://contributor-covenant.org/version/1/4][version]
|
72
|
+
|
73
|
+
[homepage]: http://contributor-covenant.org
|
74
|
+
[version]: http://contributor-covenant.org/version/1/4/
|
data/Gemfile
ADDED
data/LICENSE.txt
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
The MIT License (MIT)
|
2
|
+
|
3
|
+
Copyright (c) 2018 Tom Finill
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in
|
13
|
+
all copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
21
|
+
THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,159 @@
|
|
1
|
+
# Bonito
|
2
|
+
|
3
|
+
 [](https://codeclimate.com/github/TomFinill/bonito/maintainability) [](https://codeclimate.com/github/TomFinill/bonito/test_coverage)
|
4
|
+
|
5
|
+
## TL;DR
|
6
|
+
|
7
|
+
**Bonito** is a ruby DSL for generating canned data. It can simulate, by
|
8
|
+
_freezing time_, sequences of events happening in series and parallel in
|
9
|
+
order to approximate any kind of live data.
|
10
|
+
|
11
|
+
`Bonito` uses [Timecop](https://github.com/travisjeffery/timecop) in order to
|
12
|
+
perform this _freezing_.
|
13
|
+
|
14
|
+

|
15
|
+
|
16
|
+
### An example is worth a thousand theorems
|
17
|
+
|
18
|
+
#### Timeline Definition
|
19
|
+
|
20
|
+
Suppose you work for a small media startup and you wish to create a set of data
|
21
|
+
that can be loaded into a demo environment in order to then be used by the sales
|
22
|
+
department when presenting to potential customers.
|
23
|
+
|
24
|
+
This data could consist of models representing `Author`s and the `Article`s they
|
25
|
+
write, as well as readers (or `User`s) and `Comment`s they leave on the
|
26
|
+
aforementioned `Article`s.
|
27
|
+
|
28
|
+
Obviously, each `Article` should be created by an `Author` and the simulated
|
29
|
+
creation time of each `Comment` should be _after_ that of its associated
|
30
|
+
`Article`. In fact, we can consider the data to consist of a collection of
|
31
|
+
_timelines_ where each timeline includes the creation of an `Article` by an
|
32
|
+
`Author` with this being followed afterwards by a series of `Comment`s on the
|
33
|
+
`Article` being created by `User`s.
|
34
|
+
|
35
|
+
`Bonito` offers a `Window` object to model such timelines.
|
36
|
+
|
37
|
+
Each `Window` has a duration in which events occur. These events are referred
|
38
|
+
to in `Bonito` as `Moment`s.
|
39
|
+
|
40
|
+
For example, a `Window` representing the timeline described above could be
|
41
|
+
created as follows:
|
42
|
+
|
43
|
+
```ruby
|
44
|
+
example_window = Bonito.over 1.week do
|
45
|
+
please do
|
46
|
+
author = authors.sample
|
47
|
+
title = Faker::Company.bs
|
48
|
+
self.article = Article.new(title, author)
|
49
|
+
articles << article
|
50
|
+
end
|
51
|
+
|
52
|
+
repeat times: rand(10), over: 5.days do
|
53
|
+
please do
|
54
|
+
user = users.sample
|
55
|
+
content = Faker::Lorem.sentence
|
56
|
+
comments << Comment.new(content, article, user)
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
60
|
+
```
|
61
|
+
|
62
|
+
Here, via the `over` method on the `Bonito` module, a `Window` is defined. Within
|
63
|
+
this window a series of `Moment`s are defined. Firstly, the `Window#please` method
|
64
|
+
is invoked to define the event. In this case the event consists of creating an
|
65
|
+
`Article`.
|
66
|
+
|
67
|
+
After this, we wish to define `Moment`s in which many `Comment`s are created for
|
68
|
+
the `Article`.
|
69
|
+
|
70
|
+
To do this we use the `Window#repeat` method. This method accepts a block along
|
71
|
+
with a `times` parameter and an `over` parameter and inserts a new, child `Window`
|
72
|
+
into the current, parent window. The contents of the child window will be
|
73
|
+
that defined by the block repeated `times` times.
|
74
|
+
|
75
|
+
This means that the child `Window` defines up to 9 (`rand(10)`) `Moment`s
|
76
|
+
where each such `Moment` creates a `Comment` belonging to the previously
|
77
|
+
created `Article`.
|
78
|
+
|
79
|
+
_Now_, suppose we intend to simulate the creation of multiple `Article`s and
|
80
|
+
their associated `Comment`s. One way to achieve this would be to repeat the
|
81
|
+
previously defined `Window` via the `repeat` command:
|
82
|
+
|
83
|
+
```ruby
|
84
|
+
serial_window = Bonito.over 10.weeks do
|
85
|
+
repeat times: 5 do
|
86
|
+
use example_window
|
87
|
+
end
|
88
|
+
end
|
89
|
+
```
|
90
|
+
|
91
|
+
However this approach has a serious drawback: all events will occur _in series_.
|
92
|
+
The second `Article` will not be created until all the `Comment`s on the first
|
93
|
+
`Article` have been created and similarly the third `Article` will be preceded by
|
94
|
+
all `Comment`s on the second.
|
95
|
+
|
96
|
+
Ideally what we want is for `Articles` and `Comment`s to be _interleaved_.
|
97
|
+
|
98
|
+
We can achieve this using the `Window#simultaneously`
|
99
|
+
method to create a `Container` object, used to define parallel timelines. We
|
100
|
+
then fill that container with the same, `Window` five times, using the
|
101
|
+
`Container#use` method.
|
102
|
+
|
103
|
+
```ruby
|
104
|
+
parallel_window = Bonito.over 2.weeks do
|
105
|
+
simultaneously do
|
106
|
+
repeat times: 5 do
|
107
|
+
use example_window
|
108
|
+
end
|
109
|
+
end
|
110
|
+
end
|
111
|
+
```
|
112
|
+
|
113
|
+
The above will create 5 `Article`s, each having up to 9 `Comment`s where the
|
114
|
+
moment at which `Article` is created is independent of any other `Article` or
|
115
|
+
`Comment`. The times at which the `Comment`s are created, meanwhile, are
|
116
|
+
dependent _only_ on the `Article` to which they belong.
|
117
|
+
|
118
|
+
#### Execution
|
119
|
+
|
120
|
+
Now we have defined the _shape_ of the data we wish to create, it remains
|
121
|
+
to actually create it.
|
122
|
+
|
123
|
+
This is achieved via a `Runner` object that takes a `Window` and uses it to
|
124
|
+
evaluate `Moment`s.
|
125
|
+
|
126
|
+
```ruby
|
127
|
+
Bonito.run parallel_window, starting: 8.weeks_ago
|
128
|
+
```
|
129
|
+
|
130
|
+
This will take the `Window` object `parallel_window` and distribute the `Moment`s
|
131
|
+
it contains according to its configuration, mapping each `Moment` to a point
|
132
|
+
in time relative to the start time given by the `starting` parameter.
|
133
|
+
|
134
|
+
_However_, a typical use case may require different data set sizes for
|
135
|
+
different applications: For example, a large dataset to live in a staging
|
136
|
+
environment in order to sanity check releases and a small, easy to load dataset
|
137
|
+
that can be used locally while developing.
|
138
|
+
|
139
|
+
Suppose we have certain events that we wish to occur only once,
|
140
|
+
(using the above example, this may be the creation of some `Organisation` object
|
141
|
+
representing the news company for which articles are being written) as well
|
142
|
+
as events that we wish to be able to scale, such as the creation of `Article`
|
143
|
+
objects and their associated `Comment`s.
|
144
|
+
|
145
|
+
Using `Bonito`, we could define two windows: a `singleton_window` that is run once per
|
146
|
+
dataset and generates our `Organisation` model, as well as a `scalable_window`
|
147
|
+
that results in different sizes of data according to some size parameter.
|
148
|
+
|
149
|
+
These windows can then be combined as follows, where the size paramter `factor`
|
150
|
+
can be provided dynamically as, say, an argument provided to a `Rake` task.
|
151
|
+
|
152
|
+
```ruby
|
153
|
+
scaled_window = singleton_window + (scalable_window * factor)
|
154
|
+
```
|
155
|
+
|
156
|
+
The `scaled_window`, when run, will run the `scalable_window` `factor` times
|
157
|
+
in parallel.
|
158
|
+
|
159
|
+
|
data/Rakefile
ADDED
@@ -0,0 +1,15 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'bundler/gem_tasks'
|
4
|
+
require 'rspec/core/rake_task'
|
5
|
+
require 'rdoc/task'
|
6
|
+
|
7
|
+
RSpec::Core::RakeTask.new(:spec)
|
8
|
+
|
9
|
+
task default: :spec
|
10
|
+
|
11
|
+
RDoc::Task.new do |rdoc|
|
12
|
+
rdoc.main = 'README.md'
|
13
|
+
rdoc.rdoc_dir = 'docs'
|
14
|
+
rdoc.rdoc_files.include('README.md', 'lib')
|
15
|
+
end
|
data/_config.yml
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
theme: jekyll-theme-cayman
|
data/bin/console
ADDED
@@ -0,0 +1,16 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
# frozen_string_literal: true
|
3
|
+
|
4
|
+
require 'bundler/setup'
|
5
|
+
require 'bonito'
|
6
|
+
require 'active_support/core_ext/numeric/time'
|
7
|
+
|
8
|
+
# You can add fixtures and/or initialization code here to make experimenting
|
9
|
+
# with your gem easier. You can also use a different console, if you like.
|
10
|
+
|
11
|
+
# (If you use this, don't forget to add pry to your Gemfile!)
|
12
|
+
# require "pry"
|
13
|
+
# Pry.start
|
14
|
+
|
15
|
+
require 'irb'
|
16
|
+
IRB.start(__FILE__)
|
data/bin/setup
ADDED
data/bonito.gemspec
ADDED
@@ -0,0 +1,45 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
lib = File.expand_path('lib', __dir__)
|
4
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
5
|
+
require 'bonito/version'
|
6
|
+
|
7
|
+
Gem::Specification.new do |spec|
|
8
|
+
spec.name = 'bonito'
|
9
|
+
spec.version = Bonito::VERSION
|
10
|
+
spec.authors = ['Tom Finill']
|
11
|
+
spec.email = ['tomfinill@gmail.com']
|
12
|
+
|
13
|
+
spec.summary = 'A simple tool to create demo data'
|
14
|
+
spec.description = 'Create realistic demo data by simulating events occurring over some time period'
|
15
|
+
spec.homepage = 'https://github.com/TomFinill/bonito'
|
16
|
+
spec.license = 'MIT'
|
17
|
+
|
18
|
+
# Specify which files should be added to the gem when it is released.
|
19
|
+
# The `git ls-files -z` loads the files in the RubyGem that have been
|
20
|
+
# added into git.
|
21
|
+
spec.files = Dir.chdir(File.expand_path(__dir__)) do
|
22
|
+
`git ls-files -z`.split("\x0").reject do |f|
|
23
|
+
f.match(%r{^(test|spec|features)/})
|
24
|
+
end
|
25
|
+
end
|
26
|
+
spec.bindir = 'exe'
|
27
|
+
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
28
|
+
spec.require_paths = ['lib']
|
29
|
+
|
30
|
+
spec.add_dependency 'algorithms', '~> 0.5'
|
31
|
+
spec.add_dependency 'ruby-progressbar'
|
32
|
+
spec.add_dependency 'timecop'
|
33
|
+
|
34
|
+
spec.add_development_dependency 'activesupport'
|
35
|
+
spec.add_development_dependency 'bundler', '~> 2.0'
|
36
|
+
spec.add_development_dependency 'factory_bot'
|
37
|
+
spec.add_development_dependency 'faker', '~> 1.9.1'
|
38
|
+
spec.add_development_dependency 'rake', '~> 12.3'
|
39
|
+
spec.add_development_dependency 'rdoc'
|
40
|
+
spec.add_development_dependency 'reek'
|
41
|
+
spec.add_development_dependency 'rspec', '~> 3.0'
|
42
|
+
spec.add_development_dependency 'rubocop'
|
43
|
+
spec.add_development_dependency 'simplecov'
|
44
|
+
|
45
|
+
end
|