mirror of
https://github.com/Growstuff/growstuff.git
synced 2026-05-25 09:19:15 -04:00
Compare commits
261 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
34470a8457 | ||
|
|
90f4db4853 | ||
|
|
28503cdca0 | ||
|
|
31744491ea | ||
|
|
d541f795f6 | ||
|
|
03837e193e | ||
|
|
612bfcffcb | ||
|
|
d0b785bcd3 | ||
|
|
857c9e9d7a | ||
|
|
913ac45127 | ||
|
|
27c4dbedf2 | ||
|
|
4867c68984 | ||
|
|
a326f92210 | ||
|
|
835e277f49 | ||
|
|
7197843ac3 | ||
|
|
ea1fb90279 | ||
|
|
53b08422ce | ||
|
|
a051523354 | ||
|
|
5563b183dc | ||
|
|
69f6c8d5de | ||
|
|
f69f0861d2 | ||
|
|
d629203570 | ||
|
|
338e705db7 | ||
|
|
dfc53747d9 | ||
|
|
2b8aac19dd | ||
|
|
b52763b7c2 | ||
|
|
45fcb76f30 | ||
|
|
b5c55f42b8 | ||
|
|
53ddb4ef94 | ||
|
|
4147a571a5 | ||
|
|
02f4c7f3ac | ||
|
|
329f77db78 | ||
|
|
345fac4358 | ||
|
|
409f24e4e5 | ||
|
|
5dd22d8d1c | ||
|
|
dbbd7fdeb4 | ||
|
|
3c4e1423f9 | ||
|
|
40600215bc | ||
|
|
0bdf42f6c8 | ||
|
|
711fb28f49 | ||
|
|
77f14feae5 | ||
|
|
1b6bbfcfac | ||
|
|
d39af973e0 | ||
|
|
14dd8bbc8b | ||
|
|
72753c2d43 | ||
|
|
b1854bbd4a | ||
|
|
709cd640d2 | ||
|
|
9e7c6584a6 | ||
|
|
1211be2fef | ||
|
|
d712b440b4 | ||
|
|
517cfd85e5 | ||
|
|
7030c868b6 | ||
|
|
d8b39c69ec | ||
|
|
e7e997290e | ||
|
|
5d88b65284 | ||
|
|
31d97f4c00 | ||
|
|
ec9f57e6b8 | ||
|
|
dd0e9f74e6 | ||
|
|
9d389726f5 | ||
|
|
4fd0096c85 | ||
|
|
28945b6f20 | ||
|
|
b926f07e2a | ||
|
|
b10719ecab | ||
|
|
1527fec441 | ||
|
|
1096fd0e89 | ||
|
|
0cde6ad684 | ||
|
|
8b4951c71a | ||
|
|
8fe3b1da15 | ||
|
|
d2c81d6b29 | ||
|
|
5ad44c10af | ||
|
|
008537a324 | ||
|
|
116a9a8bab | ||
|
|
dc7cbadfc0 | ||
|
|
1c7a3e27d9 | ||
|
|
e66924151a | ||
|
|
6f1eed06cc | ||
|
|
605e2e1392 | ||
|
|
77d2dc6144 | ||
|
|
678dabf710 | ||
|
|
3a7be86ef9 | ||
|
|
a5fa69c7c7 | ||
|
|
1d8c2db674 | ||
|
|
945edd1cbc | ||
|
|
f67d39ed63 | ||
|
|
103d7375d3 | ||
|
|
3b2c4a7893 | ||
|
|
ecf2e82e9e | ||
|
|
82b83ca7a4 | ||
|
|
e784790e19 | ||
|
|
f9bdae77d3 | ||
|
|
dff6827761 | ||
|
|
2f65939d5c | ||
|
|
d7e71c8fc3 | ||
|
|
47ddc915ef | ||
|
|
0d303ee6df | ||
|
|
5dc8bc5ec5 | ||
|
|
78d9e3c970 | ||
|
|
a772bdaa8d | ||
|
|
0402fee04d | ||
|
|
7a1499d854 | ||
|
|
152cdace75 | ||
|
|
ed3ae8bbd6 | ||
|
|
81adae77f8 | ||
|
|
681b5bbd8c | ||
|
|
832cfde28d | ||
|
|
aa9841dc7d | ||
|
|
caa59df3bd | ||
|
|
102eaf3d6b | ||
|
|
61ebd5fe99 | ||
|
|
7e0e936509 | ||
|
|
d3faa75d1d | ||
|
|
1575ee1ddd | ||
|
|
5b5eacb738 | ||
|
|
85eea89792 | ||
|
|
fc46685ab9 | ||
|
|
db0c60be0d | ||
|
|
91e851c802 | ||
|
|
9112d4e2bb | ||
|
|
eed0b257fa | ||
|
|
25bba9c126 | ||
|
|
028c2a28a5 | ||
|
|
fa28fe25b6 | ||
|
|
77978a76f2 | ||
|
|
7be10957ec | ||
|
|
c94d08c816 | ||
|
|
9fdc9e76df | ||
|
|
eba03d0662 | ||
|
|
53d7cddea1 | ||
|
|
89cfd49616 | ||
|
|
ef74cc293d | ||
|
|
0b191049b2 | ||
|
|
e767ea6b38 | ||
|
|
6cbf7d0bba | ||
|
|
cb5d490b1c | ||
|
|
585e45487a | ||
|
|
6310c51879 | ||
|
|
4c33e5ea2d | ||
|
|
caa5bcec4b | ||
|
|
55bfd50123 | ||
|
|
86781f3491 | ||
|
|
48bf7bb2f4 | ||
|
|
6677f8641f | ||
|
|
e5cf8b6091 | ||
|
|
a7acf72f56 | ||
|
|
78f021c2f2 | ||
|
|
34145f7b64 | ||
|
|
ef9bb01fc9 | ||
|
|
e103e6be35 | ||
|
|
4ad4dd0c7f | ||
|
|
4df3354020 | ||
|
|
6077d4da34 | ||
|
|
34b6b54616 | ||
|
|
87d43f1eb2 | ||
|
|
5cc647ee9b | ||
|
|
948c32d1d5 | ||
|
|
2e4378f025 | ||
|
|
548fb3d3fd | ||
|
|
9966eddd37 | ||
|
|
aae7688a0e | ||
|
|
9b16abd373 | ||
|
|
87b04f60f7 | ||
|
|
a8509acd49 | ||
|
|
ca8dae4b78 | ||
|
|
1b3e00fba0 | ||
|
|
935aebcb0d | ||
|
|
4fe8cd566b | ||
|
|
9188bd384d | ||
|
|
8ce0717c0f | ||
|
|
4c7529eb8e | ||
|
|
f8b76e9c84 | ||
|
|
16341da7bd | ||
|
|
0d7542a43d | ||
|
|
8dead992d3 | ||
|
|
cdde3994b2 | ||
|
|
44196f7d65 | ||
|
|
dc62ab05ad | ||
|
|
15e64226c8 | ||
|
|
c860f7c14b | ||
|
|
944af28e3c | ||
|
|
a5b84403bb | ||
|
|
afc4441ce1 | ||
|
|
b7d4781aef | ||
|
|
c2ff3790db | ||
|
|
a2f2e24aa1 | ||
|
|
f1a40e4305 | ||
|
|
4120069e4c | ||
|
|
18a7306fd0 | ||
|
|
b3962bc9bc | ||
|
|
7a112aeecf | ||
|
|
296daa7a56 | ||
|
|
5bf75fa36f | ||
|
|
279c922b34 | ||
|
|
3ed90b8d04 | ||
|
|
634dc936f9 | ||
|
|
2a8c4d4829 | ||
|
|
3ee2537438 | ||
|
|
934bd9adb8 | ||
|
|
d043613bde | ||
|
|
d1603a6e7f | ||
|
|
79e0bc5206 | ||
|
|
61183509b0 | ||
|
|
6c1bbe9f1c | ||
|
|
bab41143fa | ||
|
|
3c7688d2ed | ||
|
|
ef20d1a333 | ||
|
|
9c0ddb61e7 | ||
|
|
19acb238fd | ||
|
|
d38167fd14 | ||
|
|
6b68e9ee4d | ||
|
|
3e2b4f4183 | ||
|
|
65bb523d16 | ||
|
|
efa8c513ea | ||
|
|
43813d1318 | ||
|
|
7e2aec9ec3 | ||
|
|
952224b2e1 | ||
|
|
26b1a3a6ca | ||
|
|
03c9151ecd | ||
|
|
1dd3d658eb | ||
|
|
c3b694ea7d | ||
|
|
a3e2b892d1 | ||
|
|
d0f6d63d36 | ||
|
|
879ce26de7 | ||
|
|
98e27b5af4 | ||
|
|
a7cfecf82c | ||
|
|
d0f28d4bef | ||
|
|
2c090757cc | ||
|
|
106d187e67 | ||
|
|
8521047803 | ||
|
|
e173d0c130 | ||
|
|
53197f27eb | ||
|
|
2460f73673 | ||
|
|
324dcfc821 | ||
|
|
e3bb43477b | ||
|
|
2d309b7645 | ||
|
|
a59f7dd137 | ||
|
|
1fb3c4e0f9 | ||
|
|
b64a294245 | ||
|
|
31d2aa052a | ||
|
|
b7f78d5dc9 | ||
|
|
f1afe1ca79 | ||
|
|
e029ef17ab | ||
|
|
d5c8b9ae27 | ||
|
|
f907ba9e4b | ||
|
|
25070f73d7 | ||
|
|
2fe6012342 | ||
|
|
61468b3587 | ||
|
|
58c4d82087 | ||
|
|
69018f2c5c | ||
|
|
00ea96a73e | ||
|
|
a1e33ae36e | ||
|
|
7f24927dc9 | ||
|
|
a178d3d42a | ||
|
|
6089198f5c | ||
|
|
715ff4d41d | ||
|
|
5992c974c7 | ||
|
|
511e89fa58 | ||
|
|
f8e83cdb9c | ||
|
|
ece7113c89 | ||
|
|
8941c577e4 | ||
|
|
1bf1076076 | ||
|
|
8906fd38ae |
@@ -51,9 +51,6 @@ PreCommit:
|
||||
command: ['bundle', 'exec', 'haml-lint']
|
||||
JsonSyntax:
|
||||
enabled: true
|
||||
BundleOutdated:
|
||||
enabled: true
|
||||
on_warn: warn
|
||||
BundleAudit:
|
||||
enabled: true
|
||||
on_warn: warn
|
||||
|
||||
43
.travis.yml
43
.travis.yml
@@ -7,47 +7,41 @@ cache:
|
||||
- tmp/cache/assets/test/sprockets
|
||||
env:
|
||||
matrix:
|
||||
- GROWSTUFF_ELASTICSEARCH='true' RSPEC_TAG=elasticsearch STATIC_CHECKS=false
|
||||
- GROWSTUFF_ELASTICSEARCH='false' RSPEC_TAG=~elasticsearch STATIC_CHECKS=false
|
||||
- GROWSTUFF_ELASTICSEARCH=true RSPEC_TAG=elasticsearch COVERAGE=true
|
||||
- GROWSTUFF_ELASTICSEARCH=false RSPEC_TAG=~elasticsearch COVERAGE=false
|
||||
- STATIC_CHECKS=true
|
||||
global:
|
||||
- secure: "Z5TpM2jEX4UCvNePnk/LwltQX48U2u9BRc+Iypr1x9QW2o228QJhPIOH39a8RMUrepGnkQIq9q3ZRUn98RfrJz1yThtlNFL3NmzdQ57gKgjGwfpa0e4Dwj/ZJqV2D84tDGjvdVYLP7zzaYZxQcwk/cgNpzKf/jq97HLNP7CYuf4="
|
||||
- GROWSTUFF_SITE_NAME="Growstuff (travis)"
|
||||
- RAILS_SECRET_TOKEN='xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'
|
||||
- secure: "Z5TpM2jEX4UCvNePnk/LwltQX48U2u9BRc+Iypr1x9QW2o228QJhPIOH39a8RMUrepGnkQIq9q3ZRUn98RfrJz1yThtlNFL3NmzdQ57gKgjGwfpa0e4Dwj/ZJqV2D84tDGjvdVYLP7zzaYZxQcwk/cgNpzKf/jq97HLNP7CYuf4="
|
||||
before_install:
|
||||
- ./script/install_phantomjs;
|
||||
- ./script/install_phantomjs.sh
|
||||
- export PATH=$PWD/travis_phantomjs/phantomjs-2.1.1-linux-x86_64/bin:$PATH
|
||||
# Force Travis to use Elastic Search 2.4.0
|
||||
- >
|
||||
if [ "${GROWSTUFF_ELASTICSEARCH}" = "true" ]; then
|
||||
sudo dpkg -r elasticsearch;
|
||||
curl -O https://download.elastic.co/elasticsearch/release/org/elasticsearch/distribution/deb/elasticsearch/2.4.0/elasticsearch-2.4.0.deb;
|
||||
sudo dpkg -i --force-confnew elasticsearch-2.4.0.deb;
|
||||
sudo service elasticsearch start;
|
||||
sleep 10;
|
||||
curl localhost:9200;
|
||||
fi
|
||||
- ./script/install_codeclimate.sh
|
||||
- ./script/install_linters.sh
|
||||
- VERSION="6.2.3" ./script/install_elasticsearch.sh
|
||||
before_script:
|
||||
- set -e
|
||||
- >
|
||||
if [ "${STATIC_CHECKS}" = "true" ]; then
|
||||
./script/install_linters;
|
||||
else
|
||||
RAILS_ENV=test bundle exec rake db:create db:migrate;
|
||||
bundle exec rake assets:precompile;
|
||||
if [ "${COVERAGE}" = "true" ]; then
|
||||
./cc-test-reporter before-build
|
||||
fi
|
||||
- set +e
|
||||
script:
|
||||
- set -e
|
||||
- >
|
||||
if [ "${STATIC_CHECKS}" = "true" ]; then
|
||||
./script/check_static.rb
|
||||
else
|
||||
bundle exec rake db:migrate --trace;
|
||||
set +e;
|
||||
RAILS_ENV=test bundle exec rake db:create db:migrate search:create;
|
||||
bundle exec rake assets:precompile;
|
||||
bundle exec rspec --tag $RSPEC_TAG spec/;
|
||||
bundle exec rake jasmine:ci;
|
||||
fi;
|
||||
- set +e
|
||||
after_script:
|
||||
- >
|
||||
if [ "${COVERAGE}" = "true" ]; then
|
||||
./cc-test-reporter after-build --exit-code $TRAVIS_TEST_RESULT;
|
||||
fi
|
||||
before_deploy:
|
||||
- bundle exec script/heroku_maintenance.rb on
|
||||
deploy:
|
||||
@@ -70,4 +64,5 @@ after_deploy:
|
||||
- bundle exec script/heroku_maintenance.rb off
|
||||
addons:
|
||||
code_climate:
|
||||
repo_token: 462e015bbdaabfb20910fc07f2fea253410ecb131444e00f97dbf32dc6789ca6
|
||||
repo_token:
|
||||
secure: "PfhLGBKRgNqhKuYCJsK+VPhdAzcgWFGeeOyxC/eS8gtlvIISVdgyZE+r30uIei0DFI6zEiN62eW4d+xtT4j7/e2ZcAcx7U52mza/SnQNuu3nCGQDJB8VOvV5NbnwXfi8vfr4e889Mt7k3ocd2c4gqB4UtRqrzhygj7HN+B/GfEk="
|
||||
|
||||
@@ -83,7 +83,12 @@ submit the change with your pull request.
|
||||
- Jeff Kingswood / [ancyentmariner](https://github.com/ancyentmariner)
|
||||
- Logan Gingerich / [logangingerich](https://github.com/logangingerich)
|
||||
- Mark Taffman / [mftaff](https://github.com/mftaff)
|
||||
- Jennifer Kruse / [jenkr55](https://github.com/jenkr55)
|
||||
- Christopher Bazin / [RobotScissors](https://github.com/robotscissors)
|
||||
|
||||
## Bots
|
||||
|
||||
- Security and Dependency Updates / [deppbot](https://github.com/deppbot)
|
||||
### Security and Dependency Updates
|
||||
- DeppBot / [deppbot](https://github.com/deppbot)
|
||||
- dependabot[bot] / [dependabot-bot](https://github.com/dependabot-bot)
|
||||
- dependabot / [dependabot](https://github.com/dependabot)
|
||||
|
||||
7
Gemfile
7
Gemfile
@@ -86,9 +86,9 @@ gem "chartkick"
|
||||
# Project does not use semver, so we want to be in sync with the version of
|
||||
# elasticsearch we use
|
||||
# See https://github.com/elastic/elasticsearch-ruby#compatibility
|
||||
gem "elasticsearch-api", "~> 2.0.0"
|
||||
gem "elasticsearch-model"
|
||||
gem "elasticsearch-rails"
|
||||
gem "elasticsearch-api", ">= 6.0.0"
|
||||
gem "elasticsearch-model", ">= 6.0.0"
|
||||
gem "elasticsearch-rails", ">= 6.0.0"
|
||||
gem "hashie", ">= 3.5.3"
|
||||
|
||||
gem 'rake', '>= 10.0.0'
|
||||
@@ -137,7 +137,6 @@ group :development, :test do
|
||||
gem 'haml-rails' # HTML templating language
|
||||
gem 'haml_lint' # Checks haml files for goodness
|
||||
gem 'i18n-tasks' # adds tests for finding missing and unused translations
|
||||
gem 'jasmine' # javascript unit testing
|
||||
gem 'poltergeist' # for headless JS testing
|
||||
gem 'rainbow', '< 2.2.0' # See https://github.com/sickill/rainbow/issues/44
|
||||
gem 'rspec-activemodel-mocks'
|
||||
|
||||
204
Gemfile.lock
204
Gemfile.lock
@@ -26,8 +26,8 @@ GEM
|
||||
addressable
|
||||
active_median (0.1.4)
|
||||
activerecord
|
||||
active_utils (3.3.9)
|
||||
activesupport (>= 3.2, < 5.2.0)
|
||||
active_utils (3.3.12)
|
||||
activesupport (>= 3.2, < 6.0)
|
||||
i18n
|
||||
activejob (4.2.10)
|
||||
activesupport (= 4.2.10)
|
||||
@@ -51,7 +51,7 @@ GEM
|
||||
public_suffix (>= 2.0.2, < 4.0)
|
||||
arel (6.0.4)
|
||||
ast (2.4.0)
|
||||
autoprefixer-rails (7.2.5)
|
||||
autoprefixer-rails (8.6.4)
|
||||
execjs
|
||||
bcrypt (3.1.11)
|
||||
better_errors (2.2.0)
|
||||
@@ -61,10 +61,8 @@ GEM
|
||||
binding_of_caller (0.8.0)
|
||||
debug_inspector (>= 0.0.1)
|
||||
bluecloth (2.2.0)
|
||||
bonsai-elasticsearch-rails (0.2.0)
|
||||
elasticsearch-model (~> 0)
|
||||
elasticsearch-rails (~> 0)
|
||||
bootstrap-datepicker-rails (1.7.1.1)
|
||||
bonsai-elasticsearch-rails (0.0.4)
|
||||
bootstrap-datepicker-rails (1.8.0.1)
|
||||
railties (>= 3.0)
|
||||
bootstrap-kaminari-views (0.0.5)
|
||||
kaminari (>= 0.13)
|
||||
@@ -74,31 +72,29 @@ GEM
|
||||
sass (>= 3.3.4)
|
||||
bootstrap_form (2.7.0)
|
||||
builder (3.2.3)
|
||||
bullet (5.7.2)
|
||||
bullet (5.7.5)
|
||||
activesupport (>= 3.0.0)
|
||||
uniform_notifier (~> 1.11.0)
|
||||
byebug (10.0.0)
|
||||
cancancan (2.1.3)
|
||||
capybara (2.17.0)
|
||||
byebug (10.0.2)
|
||||
cancancan (2.2.0)
|
||||
capybara (2.18.0)
|
||||
addressable
|
||||
mini_mime (>= 0.1.3)
|
||||
nokogiri (>= 1.3.3)
|
||||
rack (>= 1.0.0)
|
||||
rack-test (>= 0.5.4)
|
||||
xpath (>= 2.0, < 4.0)
|
||||
capybara-email (2.5.0)
|
||||
capybara (~> 2.4)
|
||||
capybara-email (3.0.1)
|
||||
capybara (>= 2.4, < 4.0)
|
||||
mail
|
||||
capybara-screenshot (1.0.18)
|
||||
capybara (>= 1.0, < 3)
|
||||
capybara-screenshot (1.0.21)
|
||||
capybara (>= 1.0, < 4)
|
||||
launchy
|
||||
chartkick (2.2.5)
|
||||
childprocess (0.8.0)
|
||||
chartkick (2.3.5)
|
||||
childprocess (0.9.0)
|
||||
ffi (~> 1.0, >= 1.0.11)
|
||||
climate_control (0.2.0)
|
||||
cliver (0.3.2)
|
||||
cocaine (0.5.8)
|
||||
climate_control (>= 0.0.3, < 1.0)
|
||||
codeclimate-test-reporter (1.0.8)
|
||||
simplecov (<= 0.13)
|
||||
codemirror-rails (5.16.0)
|
||||
@@ -111,7 +107,7 @@ GEM
|
||||
coffee-script-source
|
||||
execjs
|
||||
coffee-script-source (1.12.2)
|
||||
comfortable_mexican_sofa (1.12.10)
|
||||
comfortable_mexican_sofa (1.12.11)
|
||||
active_link_to (>= 1.0.0)
|
||||
bootstrap-sass (>= 3.2.0)
|
||||
bootstrap_form (>= 2.2.0)
|
||||
@@ -134,16 +130,16 @@ GEM
|
||||
term-ansicolor (~> 1.3)
|
||||
thor (~> 0.19.1)
|
||||
tins (~> 1.6)
|
||||
crass (1.0.3)
|
||||
crass (1.0.4)
|
||||
csv_shaper (1.3.0)
|
||||
activesupport (>= 3.0.0)
|
||||
dalli (2.7.6)
|
||||
database_cleaner (1.6.2)
|
||||
dalli (2.7.8)
|
||||
database_cleaner (1.7.0)
|
||||
debug_inspector (0.0.3)
|
||||
devise (4.4.1)
|
||||
devise (4.4.3)
|
||||
bcrypt (~> 3.0)
|
||||
orm_adapter (~> 0.1)
|
||||
railties (>= 4.1.0, < 5.2)
|
||||
railties (>= 4.1.0, < 6.0)
|
||||
responders
|
||||
warden (~> 1.2.3)
|
||||
diff-lcs (1.3)
|
||||
@@ -151,41 +147,41 @@ GEM
|
||||
easy_translate (0.5.1)
|
||||
thread
|
||||
thread_safe
|
||||
elasticsearch (2.0.2)
|
||||
elasticsearch-api (= 2.0.2)
|
||||
elasticsearch-transport (= 2.0.2)
|
||||
elasticsearch-api (2.0.2)
|
||||
elasticsearch (6.1.0)
|
||||
elasticsearch-api (= 6.1.0)
|
||||
elasticsearch-transport (= 6.1.0)
|
||||
elasticsearch-api (6.1.0)
|
||||
multi_json
|
||||
elasticsearch-model (0.1.9)
|
||||
elasticsearch-model (6.0.0)
|
||||
activesupport (> 3)
|
||||
elasticsearch (> 0.4)
|
||||
elasticsearch (> 1)
|
||||
hashie
|
||||
elasticsearch-rails (0.1.9)
|
||||
elasticsearch-transport (2.0.2)
|
||||
elasticsearch-rails (6.0.0)
|
||||
elasticsearch-transport (6.1.0)
|
||||
faraday
|
||||
multi_json
|
||||
erubis (2.7.0)
|
||||
excon (0.60.0)
|
||||
excon (0.62.0)
|
||||
execjs (2.7.0)
|
||||
factory_bot (4.8.2)
|
||||
factory_bot (4.10.0)
|
||||
activesupport (>= 3.0.0)
|
||||
factory_bot_rails (4.8.2)
|
||||
factory_bot (~> 4.8.2)
|
||||
factory_bot_rails (4.10.0)
|
||||
factory_bot (~> 4.10.0)
|
||||
railties (>= 3.0.0)
|
||||
faker (1.8.7)
|
||||
faker (1.9.1)
|
||||
i18n (>= 0.7)
|
||||
faraday (0.12.2)
|
||||
multipart-post (>= 1.2, < 3)
|
||||
ffi (1.9.21)
|
||||
ffi (1.9.25)
|
||||
figaro (1.1.1)
|
||||
thor (~> 0.14)
|
||||
flickraw (0.9.9)
|
||||
font-awesome-sass (4.7.0)
|
||||
sass (>= 3.2)
|
||||
font-awesome-sass (5.0.13)
|
||||
sassc (>= 1.11)
|
||||
formatador (0.2.5)
|
||||
friendly_id (5.2.3)
|
||||
friendly_id (5.2.4)
|
||||
activerecord (>= 4.0.0)
|
||||
geocoder (1.4.5)
|
||||
geocoder (1.4.9)
|
||||
gibbon (1.2.1)
|
||||
httparty
|
||||
multi_json (>= 1.9.0)
|
||||
@@ -241,9 +237,9 @@ GEM
|
||||
haml (>= 4.0, < 6)
|
||||
nokogiri (>= 1.6.0)
|
||||
ruby_parser (~> 3.5)
|
||||
httparty (0.15.6)
|
||||
httparty (0.16.2)
|
||||
multi_xml (>= 0.5.2)
|
||||
i18n (0.9.3)
|
||||
i18n (0.9.5)
|
||||
concurrent-ruby (~> 1.0)
|
||||
i18n-tasks (0.9.12)
|
||||
activesupport (>= 4.0.2)
|
||||
@@ -255,13 +251,7 @@ GEM
|
||||
parser (>= 2.2.3.0)
|
||||
term-ansicolor (>= 1.3.2)
|
||||
terminal-table (>= 1.5.1)
|
||||
jasmine (2.9.0)
|
||||
jasmine-core (>= 2.9.0, < 3.0.0)
|
||||
phantomjs
|
||||
rack (>= 1.2.1)
|
||||
rake
|
||||
jasmine-core (2.99.0)
|
||||
jquery-rails (4.3.1)
|
||||
jquery-rails (4.3.3)
|
||||
rails-dom-testing (>= 1, < 3)
|
||||
railties (>= 4.2.0)
|
||||
thor (>= 0.14, < 2.0)
|
||||
@@ -289,7 +279,7 @@ GEM
|
||||
kaminari-core (= 1.1.1)
|
||||
kaminari-core (1.1.1)
|
||||
kgio (2.11.2)
|
||||
kramdown (1.16.2)
|
||||
kramdown (1.17.0)
|
||||
launchy (2.4.3)
|
||||
addressable (~> 2.3)
|
||||
leaflet-rails (1.3.1)
|
||||
@@ -300,10 +290,10 @@ GEM
|
||||
rb-fsevent (~> 0.9, >= 0.9.4)
|
||||
rb-inotify (~> 0.9, >= 0.9.7)
|
||||
ruby_dep (~> 1.2)
|
||||
loofah (2.2.1)
|
||||
loofah (2.2.2)
|
||||
crass (~> 1.0.2)
|
||||
nokogiri (>= 1.5.9)
|
||||
lumberjack (1.0.12)
|
||||
lumberjack (1.0.13)
|
||||
mail (2.7.0)
|
||||
mini_mime (>= 0.1.1)
|
||||
memcachier (0.0.2)
|
||||
@@ -320,8 +310,8 @@ GEM
|
||||
multi_xml (0.6.0)
|
||||
multipart-post (2.0.0)
|
||||
nenv (0.3.0)
|
||||
newrelic_rpm (4.8.0.341)
|
||||
nokogiri (1.8.2)
|
||||
newrelic_rpm (5.2.0.345)
|
||||
nokogiri (1.8.4)
|
||||
mini_portile2 (~> 2.3.0)
|
||||
notiffany (0.1.1)
|
||||
nenv (~> 0.1)
|
||||
@@ -336,7 +326,7 @@ GEM
|
||||
omniauth (1.8.1)
|
||||
hashie (>= 3.4.6, < 3.6.0)
|
||||
rack (>= 1.6.2, < 3)
|
||||
omniauth-facebook (4.0.0)
|
||||
omniauth-facebook (5.0.0)
|
||||
omniauth-oauth2 (~> 1.2)
|
||||
omniauth-flickr (0.0.19)
|
||||
multi_json (~> 1.11.0)
|
||||
@@ -351,17 +341,16 @@ GEM
|
||||
omniauth-oauth (~> 1.1)
|
||||
rack
|
||||
orm_adapter (0.5.0)
|
||||
paperclip (5.2.1)
|
||||
paperclip (6.1.0)
|
||||
activemodel (>= 4.2.0)
|
||||
activesupport (>= 4.2.0)
|
||||
cocaine (~> 0.5.5)
|
||||
mime-types
|
||||
mimemagic (~> 0.3.0)
|
||||
terrapin (~> 0.6.0)
|
||||
parallel (1.12.1)
|
||||
parser (2.4.0.2)
|
||||
ast (~> 2.3)
|
||||
parser (2.5.1.0)
|
||||
ast (~> 2.4.0)
|
||||
pg (0.21.0)
|
||||
phantomjs (2.1.1.0)
|
||||
platform-api (2.1.0)
|
||||
heroics (~> 0.0.23)
|
||||
moneta (~> 0.8.1)
|
||||
@@ -375,11 +364,11 @@ GEM
|
||||
pry (0.11.3)
|
||||
coderay (~> 1.1.0)
|
||||
method_source (~> 0.9.0)
|
||||
public_suffix (3.0.1)
|
||||
public_suffix (3.0.2)
|
||||
quiet_assets (1.1.0)
|
||||
railties (>= 3.1, < 5.0)
|
||||
rack (1.6.9)
|
||||
rack-protection (2.0.1)
|
||||
rack (1.6.10)
|
||||
rack-protection (2.0.3)
|
||||
rack
|
||||
rack-test (0.6.3)
|
||||
rack (>= 1.0)
|
||||
@@ -403,8 +392,8 @@ GEM
|
||||
activesupport (>= 4.2.0, < 5.0)
|
||||
nokogiri (~> 1.6)
|
||||
rails-deprecated_sanitizer (>= 1.0.1)
|
||||
rails-html-sanitizer (1.0.3)
|
||||
loofah (~> 2.0)
|
||||
rails-html-sanitizer (1.0.4)
|
||||
loofah (~> 2.2, >= 2.2.2)
|
||||
rails-i18n (4.0.9)
|
||||
i18n (~> 0.7)
|
||||
railties (~> 4.0)
|
||||
@@ -420,39 +409,39 @@ GEM
|
||||
thor (>= 0.18.1, < 2.0)
|
||||
rainbow (2.1.0)
|
||||
raindrops (0.19.0)
|
||||
rake (12.3.0)
|
||||
rb-fsevent (0.10.2)
|
||||
rake (12.3.1)
|
||||
rb-fsevent (0.10.3)
|
||||
rb-inotify (0.9.10)
|
||||
ffi (>= 0.5.0, < 2)
|
||||
redis (4.0.1)
|
||||
responders (2.4.0)
|
||||
actionpack (>= 4.2.0, < 5.3)
|
||||
railties (>= 4.2.0, < 5.3)
|
||||
rspec (3.7.0)
|
||||
rspec-core (~> 3.7.0)
|
||||
rspec-expectations (~> 3.7.0)
|
||||
rspec-mocks (~> 3.7.0)
|
||||
rspec (3.8.0)
|
||||
rspec-core (~> 3.8.0)
|
||||
rspec-expectations (~> 3.8.0)
|
||||
rspec-mocks (~> 3.8.0)
|
||||
rspec-activemodel-mocks (1.0.3)
|
||||
activemodel (>= 3.0)
|
||||
activesupport (>= 3.0)
|
||||
rspec-mocks (>= 2.99, < 4.0)
|
||||
rspec-core (3.7.1)
|
||||
rspec-support (~> 3.7.0)
|
||||
rspec-expectations (3.7.0)
|
||||
rspec-core (3.8.0)
|
||||
rspec-support (~> 3.8.0)
|
||||
rspec-expectations (3.8.1)
|
||||
diff-lcs (>= 1.2.0, < 2.0)
|
||||
rspec-support (~> 3.7.0)
|
||||
rspec-mocks (3.7.0)
|
||||
rspec-support (~> 3.8.0)
|
||||
rspec-mocks (3.8.0)
|
||||
diff-lcs (>= 1.2.0, < 2.0)
|
||||
rspec-support (~> 3.7.0)
|
||||
rspec-rails (3.7.2)
|
||||
rspec-support (~> 3.8.0)
|
||||
rspec-rails (3.8.0)
|
||||
actionpack (>= 3.0)
|
||||
activesupport (>= 3.0)
|
||||
railties (>= 3.0)
|
||||
rspec-core (~> 3.7.0)
|
||||
rspec-expectations (~> 3.7.0)
|
||||
rspec-mocks (~> 3.7.0)
|
||||
rspec-support (~> 3.7.0)
|
||||
rspec-support (3.7.1)
|
||||
rspec-core (~> 3.8.0)
|
||||
rspec-expectations (~> 3.8.0)
|
||||
rspec-mocks (~> 3.8.0)
|
||||
rspec-support (~> 3.8.0)
|
||||
rspec-support (3.8.0)
|
||||
rubocop (0.49.1)
|
||||
parallel (~> 1.10)
|
||||
parser (>= 2.3.3.1, < 3.0)
|
||||
@@ -461,12 +450,12 @@ GEM
|
||||
ruby-progressbar (~> 1.7)
|
||||
unicode-display_width (~> 1.0, >= 1.0.1)
|
||||
ruby-progressbar (1.9.0)
|
||||
ruby-units (2.2.1)
|
||||
ruby-units (2.3.0)
|
||||
ruby_dep (1.5.0)
|
||||
ruby_parser (3.10.1)
|
||||
ruby_parser (3.11.0)
|
||||
sexp_processor (~> 4.9)
|
||||
rubyzip (1.2.1)
|
||||
sass (3.5.5)
|
||||
sass (3.5.6)
|
||||
sass-listen (~> 4.0.0)
|
||||
sass-listen (4.0.0)
|
||||
rb-fsevent (~> 0.9, >= 0.9.4)
|
||||
@@ -477,24 +466,27 @@ GEM
|
||||
sprockets (>= 2.8, < 4.0)
|
||||
sprockets-rails (>= 2.0, < 4.0)
|
||||
tilt (>= 1.1, < 3)
|
||||
selenium-webdriver (3.9.0)
|
||||
sassc (1.12.0)
|
||||
ffi (~> 1.9.6)
|
||||
sass (>= 3.3.0)
|
||||
selenium-webdriver (3.14.0)
|
||||
childprocess (~> 0.5)
|
||||
rubyzip (~> 1.2)
|
||||
sexp_processor (4.10.0)
|
||||
sexp_processor (4.11.0)
|
||||
shellany (0.0.1)
|
||||
sidekiq (5.1.0)
|
||||
sidekiq (5.1.3)
|
||||
concurrent-ruby (~> 1.0)
|
||||
connection_pool (~> 2.2, >= 2.2.0)
|
||||
rack-protection (>= 1.5.0)
|
||||
redis (>= 3.3.4, < 5)
|
||||
redis (>= 3.3.5, < 5)
|
||||
simplecov (0.12.0)
|
||||
docile (~> 1.1.0)
|
||||
json (>= 1.8, < 3)
|
||||
simplecov-html (~> 0.10.0)
|
||||
simplecov-html (0.10.2)
|
||||
sparkpost_rails (1.5.0)
|
||||
rails (>= 4.0, < 5.2)
|
||||
sprockets (3.7.1)
|
||||
sparkpost_rails (1.5.1)
|
||||
rails (>= 4.0, < 5.3)
|
||||
sprockets (3.7.2)
|
||||
concurrent-ruby (~> 1.0)
|
||||
rack (> 1, < 3)
|
||||
sprockets-rails (3.2.1)
|
||||
@@ -507,6 +499,8 @@ GEM
|
||||
tins (~> 1.0)
|
||||
terminal-table (1.8.0)
|
||||
unicode-display_width (~> 1.1, >= 1.1.1)
|
||||
terrapin (0.6.0)
|
||||
climate_control (>= 0.0.3, < 1.0)
|
||||
thor (0.19.4)
|
||||
thread (0.2.2)
|
||||
thread_safe (0.3.6)
|
||||
@@ -516,10 +510,10 @@ GEM
|
||||
trollop (1.16.2)
|
||||
tzinfo (1.2.5)
|
||||
thread_safe (~> 0.1)
|
||||
uglifier (4.1.5)
|
||||
uglifier (4.1.15)
|
||||
execjs (>= 0.3.0, < 3)
|
||||
unicode-display_width (1.3.0)
|
||||
unicorn (5.4.0)
|
||||
unicode-display_width (1.3.2)
|
||||
unicorn (5.4.1)
|
||||
kgio (~> 2.6)
|
||||
raindrops (~> 0.7)
|
||||
uniform_notifier (1.11.0)
|
||||
@@ -534,7 +528,7 @@ GEM
|
||||
websocket-extensions (0.1.3)
|
||||
will_paginate (3.1.6)
|
||||
xmlrpc (0.3.0)
|
||||
xpath (3.0.0)
|
||||
xpath (3.1.0)
|
||||
nokogiri (~> 1.8)
|
||||
|
||||
PLATFORMS
|
||||
@@ -567,9 +561,9 @@ DEPENDENCIES
|
||||
dalli
|
||||
database_cleaner
|
||||
devise
|
||||
elasticsearch-api (~> 2.0.0)
|
||||
elasticsearch-model
|
||||
elasticsearch-rails
|
||||
elasticsearch-api (>= 6.0.0)
|
||||
elasticsearch-model (>= 6.0.0)
|
||||
elasticsearch-rails (>= 6.0.0)
|
||||
factory_bot_rails
|
||||
faker
|
||||
figaro
|
||||
@@ -587,7 +581,6 @@ DEPENDENCIES
|
||||
haml_lint
|
||||
hashie (>= 3.5.3)
|
||||
i18n-tasks
|
||||
jasmine
|
||||
jquery-rails
|
||||
jquery-ui-rails (~> 5.0.2)
|
||||
js-routes
|
||||
@@ -631,5 +624,6 @@ DEPENDENCIES
|
||||
|
||||
RUBY VERSION
|
||||
ruby 2.4.1p111
|
||||
|
||||
BUNDLED WITH
|
||||
1.16.1
|
||||
1.16.4
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
# Growstuff
|
||||
|
||||
[](https://travis-ci.org/Growstuff/growstuff)
|
||||
[](https://coveralls.io/r/Growstuff/growstuff)
|
||||
[](https://coveralls.io/github/Growstuff/growstuff?branch=dev)
|
||||
[](https://codeclimate.com/github/Growstuff/growstuff)
|
||||
|
||||
Welcome to the Growstuff project.
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
// Use this file to override Twitter Bootstrap variables or define own variables.
|
||||
|
||||
// Import original variables so they can be used in overrides
|
||||
//@import 'bootstrap/variables.scss'
|
||||
@import 'bootstrap/variables.scss'
|
||||
|
||||
// Base colours
|
||||
|
||||
@@ -10,7 +10,7 @@ $brown: #413f3b
|
||||
|
||||
$green: #5f8e43
|
||||
$blue: #2f4365
|
||||
$red: #8e4d43
|
||||
$red: #ff4d43
|
||||
$orange: #ffa500
|
||||
$yellow: #b2935c
|
||||
$white: #ffffff
|
||||
|
||||
@@ -12,6 +12,10 @@ body
|
||||
.list-inline > li.first
|
||||
padding-left: 0px
|
||||
|
||||
.activity-list
|
||||
list-style-type: none
|
||||
padding: 0
|
||||
|
||||
h2
|
||||
font-size: 150%
|
||||
|
||||
@@ -34,6 +38,21 @@ h3
|
||||
max-width: 100%
|
||||
height: auto
|
||||
|
||||
.profile-sidebar
|
||||
margin-top: -5rem
|
||||
.avatar
|
||||
border-radius: 50%
|
||||
border-radius: 50%
|
||||
z-index: 2
|
||||
position: relative
|
||||
|
||||
.profile-activity
|
||||
background: white
|
||||
padding: 2em
|
||||
margin-top: 2em
|
||||
.container
|
||||
width: 100%
|
||||
|
||||
.sidebar
|
||||
border-left: 1px solid darken($beige, 10%)
|
||||
margin-left: -1px
|
||||
@@ -73,10 +92,16 @@ p.stats
|
||||
display: flex
|
||||
flex: none
|
||||
flex-wrap: wrap
|
||||
justify-content: space-between
|
||||
|
||||
.card-row
|
||||
display: grid
|
||||
grid-template-columns: 50% 50%
|
||||
grid-gap: 25px
|
||||
grid-row-gap: 5px
|
||||
|
||||
.member-thumbnail
|
||||
padding: .25em
|
||||
margin: 1em
|
||||
|
||||
div
|
||||
width: 5em
|
||||
@@ -87,19 +112,47 @@ p.stats
|
||||
padding-left: 1em
|
||||
width: 15em
|
||||
|
||||
.progress
|
||||
border-radius: 0
|
||||
|
||||
.badge-super-late
|
||||
background-color: $red
|
||||
.badge-harvest
|
||||
background-color: $blue
|
||||
|
||||
.planting-super-late
|
||||
.planting-late
|
||||
background-color: $beige
|
||||
|
||||
.planting
|
||||
.planting-badges
|
||||
position: absolute
|
||||
|
||||
.planting-thumbnail
|
||||
padding: 0
|
||||
border: 1px solid darken($beige, 10%)
|
||||
border-radius: 4px
|
||||
.planting-actions
|
||||
top: -8em
|
||||
.planting-name
|
||||
position: relative
|
||||
top: -1em
|
||||
|
||||
dl.planting-attributes
|
||||
dt
|
||||
text-align: left
|
||||
dd
|
||||
margin-left: auto
|
||||
|
||||
.layout-actions
|
||||
width: 100%
|
||||
|
||||
#placesmap, #cropmap
|
||||
height: 500px
|
||||
|
||||
#membermap
|
||||
height: 250px
|
||||
z-index: 0
|
||||
|
||||
.location-not-set
|
||||
height: 250px
|
||||
@@ -146,6 +199,8 @@ p.stats
|
||||
border: none
|
||||
text-align: center
|
||||
margin-bottom: 1.5em
|
||||
max-width: 160px
|
||||
max-height: 200px
|
||||
|
||||
.member-thumbnail
|
||||
text-align: left
|
||||
@@ -192,6 +247,9 @@ li.crop-hierarchy
|
||||
.navbar-bottom
|
||||
margin: 40px 0px 0px 0px !important
|
||||
|
||||
.post-actions
|
||||
margin-bottom: 1rem
|
||||
|
||||
// footer
|
||||
footer
|
||||
#footer1, #footer2, #footer3
|
||||
@@ -221,6 +279,7 @@ footer
|
||||
|
||||
#maincontainer
|
||||
min-height: 80%
|
||||
padding: 50px
|
||||
|
||||
html, body
|
||||
height: 100%
|
||||
@@ -334,3 +393,6 @@ ul.thumbnail-buttons
|
||||
height: 180px
|
||||
.seed-thumbnail
|
||||
height: 220px
|
||||
|
||||
#maincontainer
|
||||
padding: 10px
|
||||
|
||||
@@ -3,7 +3,7 @@ class HarvestsController < ApplicationController
|
||||
after_action :update_crop_medians, only: %i(create update destroy)
|
||||
load_and_authorize_resource
|
||||
respond_to :html, :json
|
||||
respond_to :csv, only: :index
|
||||
respond_to :csv, :rss, only: :index
|
||||
responders :flash
|
||||
|
||||
def index
|
||||
@@ -92,7 +92,7 @@ class HarvestsController < ApplicationController
|
||||
# if this harvest is not linked to a planting, then do nothing
|
||||
return if @harvest.planting.nil?
|
||||
|
||||
@harvest.planting.update_harvest_days
|
||||
@harvest.planting.update_harvest_days!
|
||||
@harvest.crop.update_harvest_medians
|
||||
end
|
||||
end
|
||||
|
||||
@@ -20,6 +20,7 @@ class MembersController < ApplicationController
|
||||
@facebook_auth = @member.auth('facebook')
|
||||
@posts = @member.posts
|
||||
@gardens = @member.gardens.active.order(:name)
|
||||
@harvests = @member.harvests
|
||||
|
||||
# The garden form partial is called from the "New Garden" tab;
|
||||
# it requires a garden to be passed in @garden.
|
||||
|
||||
@@ -77,7 +77,7 @@ class PlantingsController < ApplicationController
|
||||
end
|
||||
|
||||
def update_planting_medians
|
||||
@planting.update_harvest_days
|
||||
@planting.update_harvest_days!
|
||||
end
|
||||
|
||||
def planting_params
|
||||
|
||||
@@ -1,26 +1,9 @@
|
||||
module ApplicationHelper
|
||||
def price_in_dollars(price)
|
||||
format('%.2f', price / 100.0)
|
||||
end
|
||||
|
||||
# 999 cents becomes 9.99 AUD -- for products/orders/etc
|
||||
def price_with_currency(price)
|
||||
format('%.2f %s', price / 100.0, Growstuff::Application.config.currency)
|
||||
end
|
||||
|
||||
def parse_date(str)
|
||||
str ||= '' # Date.parse barfs on nil
|
||||
str == '' ? nil : Date.parse(str)
|
||||
end
|
||||
|
||||
def forex_link(price)
|
||||
pid = price_in_dollars(price)
|
||||
currency = Growstuff::Application.config.currency
|
||||
link = "http://www.wolframalpha.com/input/?i=#{pid}+#{currency}"
|
||||
|
||||
link_to "(convert)", link, target: "_blank", rel: "noopener noreferrer"
|
||||
end
|
||||
|
||||
def build_alert_classes(alert_type = :info)
|
||||
classes = 'alert alert-dismissable '
|
||||
case alert_type.to_sym
|
||||
@@ -86,11 +69,15 @@ module ApplicationHelper
|
||||
def show_inactive_tickbox_path(type, owner, show_all)
|
||||
all = show_all ? '' : 1
|
||||
if owner
|
||||
plantings_by_owner_path(owner: owner.slug, all: all) if type == 'plantings'
|
||||
gardens_by_owner_path(owner: owner.slug, all: all) if type == 'gardens'
|
||||
else
|
||||
plantings_path(all: all) if type == 'plantings'
|
||||
gardens_path(all: all) if type == 'gardens'
|
||||
if type == 'plantings'
|
||||
plantings_by_owner_path(owner: owner.slug, all: all)
|
||||
elsif type == 'gardens'
|
||||
gardens_by_owner_path(owner: owner.slug, all: all)
|
||||
end
|
||||
elsif type == 'plantings'
|
||||
plantings_path(all: all)
|
||||
elsif type == 'gardens'
|
||||
gardens_path(all: all)
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -3,37 +3,49 @@ module PhotosHelper
|
||||
if crop.default_photo.present?
|
||||
crop.default_photo.thumbnail_url
|
||||
else
|
||||
default_image
|
||||
placeholder_image
|
||||
end
|
||||
end
|
||||
|
||||
def garden_image_path(garden)
|
||||
if garden.default_photo.present?
|
||||
garden.default_photo.thumbnail_url
|
||||
else
|
||||
placeholder_image
|
||||
end
|
||||
end
|
||||
|
||||
def planting_image_path(planting)
|
||||
if planting.photos.present?
|
||||
planting.photos.first.thumbnail_url
|
||||
planting.photos.order(date_taken: :desc).first.thumbnail_url
|
||||
else
|
||||
default_image
|
||||
placeholder_image
|
||||
end
|
||||
end
|
||||
|
||||
def harvest_image_path(harvest)
|
||||
if harvest.photos.present?
|
||||
harvest.photos.first.thumbnail_url
|
||||
harvest.photos.order(date_taken: :desc).first.thumbnail_url
|
||||
elsif harvest.planting.present?
|
||||
planting_image_path(harvest.planting)
|
||||
else
|
||||
default_image
|
||||
placeholder_image
|
||||
end
|
||||
end
|
||||
|
||||
def seed_image_path(seed)
|
||||
if seed.default_photo
|
||||
if seed.default_photo.present?
|
||||
seed.default_photo.thumbnail_url
|
||||
elsif seed.crop.default_photo.present?
|
||||
seed.crop.default_photo.thumbnail_url
|
||||
else
|
||||
default_image
|
||||
placeholder_image
|
||||
end
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def default_image
|
||||
def placeholder_image
|
||||
'placeholder_150.png'
|
||||
end
|
||||
end
|
||||
|
||||
@@ -32,4 +32,24 @@ module PlantingsHelper
|
||||
def plantings_active_tickbox_path(owner, show_all)
|
||||
show_inactive_tickbox_path('plantings', owner, show_all)
|
||||
end
|
||||
|
||||
def days_from_now_to_finished(planting)
|
||||
return unless planting.finish_is_predicatable?
|
||||
(planting.finish_predicted_at - Time.zone.today).to_i
|
||||
end
|
||||
|
||||
def days_from_now_to_first_harvest(planting)
|
||||
return unless planting.planted_at.present? && planting.first_harvest_predicted_at.present?
|
||||
(planting.first_harvest_predicted_at - Time.zone.today).to_i
|
||||
end
|
||||
|
||||
def planting_classes(planting)
|
||||
classes = []
|
||||
classes << 'planting-growing' if planting.growing?
|
||||
classes << 'planting-finished' if planting.finished?
|
||||
classes << 'planting-harvest-time' if planting.harvest_time?
|
||||
classes << 'planting-late' if planting.late?
|
||||
classes << 'planting-super-late' if planting.super_late?
|
||||
classes.join(' ')
|
||||
end
|
||||
end
|
||||
|
||||
@@ -26,8 +26,8 @@ class Notifier < ActionMailer::Base
|
||||
def planting_reminder(member)
|
||||
@member = member
|
||||
|
||||
@plantings = @member.plantings.first(5)
|
||||
@harvests = @member.harvests.first(5)
|
||||
@plantings = @member.plantings.order(planted_at: :desc).first(5)
|
||||
@harvests = @member.harvests.order(harvested_at: :desc).first(5)
|
||||
|
||||
# Encrypting
|
||||
message = { member_id: @member.id, type: :send_planting_reminder }
|
||||
|
||||
7
app/models/concerns/ownable.rb
Normal file
7
app/models/concerns/ownable.rb
Normal file
@@ -0,0 +1,7 @@
|
||||
module Ownable
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
included do
|
||||
belongs_to :owner, class_name: 'Member', foreign_key: 'owner_id', counter_cache: true
|
||||
end
|
||||
end
|
||||
61
app/models/concerns/predict_harvest.rb
Normal file
61
app/models/concerns/predict_harvest.rb
Normal file
@@ -0,0 +1,61 @@
|
||||
module PredictHarvest
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
included do # rubocop:disable Metrics/BlockLength
|
||||
# dates
|
||||
def first_harvest_date
|
||||
harvests_with_dates.minimum(:harvested_at)
|
||||
end
|
||||
|
||||
def last_harvest_date
|
||||
harvests_with_dates.maximum(:harvested_at)
|
||||
end
|
||||
|
||||
def first_harvest_predicted_at
|
||||
return unless crop.median_days_to_first_harvest.present? && planted_at.present?
|
||||
planted_at + crop.median_days_to_first_harvest.days
|
||||
end
|
||||
|
||||
def last_harvest_predicted_at
|
||||
return unless crop.median_days_to_last_harvest.present? && planted_at.present?
|
||||
planted_at + crop.median_days_to_last_harvest.days
|
||||
end
|
||||
|
||||
# actions
|
||||
def update_harvest_days!
|
||||
days_to_first_harvest = nil
|
||||
days_to_last_harvest = nil
|
||||
if planted_at.present? && harvests_with_dates.size.positive?
|
||||
days_to_first_harvest = (first_harvest_date - planted_at).to_i
|
||||
days_to_last_harvest = (last_harvest_date - planted_at).to_i if finished?
|
||||
end
|
||||
update(days_to_first_harvest: days_to_first_harvest, days_to_last_harvest: days_to_last_harvest)
|
||||
end
|
||||
|
||||
# status
|
||||
def harvest_time?
|
||||
return false if crop.perennial || finished
|
||||
|
||||
# We have harvests but haven't finished
|
||||
harvests.size.positive? ||
|
||||
|
||||
# or, we don't have harvests, but we predict we should by now
|
||||
(first_harvest_predicted_at.present? &&
|
||||
harvests.empty? &&
|
||||
first_harvest_predicted_at < Time.zone.today)
|
||||
end
|
||||
|
||||
def before_harvest_time?
|
||||
first_harvest_predicted_at.present? &&
|
||||
harvests.empty? &&
|
||||
first_harvest_predicted_at.present? &&
|
||||
first_harvest_predicted_at > Time.zone.today
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def harvests_with_dates
|
||||
harvests.where.not(harvested_at: nil)
|
||||
end
|
||||
end
|
||||
end
|
||||
80
app/models/concerns/predict_planting.rb
Normal file
80
app/models/concerns/predict_planting.rb
Normal file
@@ -0,0 +1,80 @@
|
||||
module PredictPlanting
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
included do # rubocop:disable Metrics/BlockLength
|
||||
## Triggers
|
||||
before_save :calculate_lifespan
|
||||
|
||||
def calculate_lifespan
|
||||
self.lifespan = (planted_at.present? && finished_at.present? ? finished_at - planted_at : nil)
|
||||
end
|
||||
|
||||
# dates
|
||||
def finish_predicted_at
|
||||
if planted_at.blank?
|
||||
nil
|
||||
elsif crop.median_lifespan.present?
|
||||
planted_at + crop.median_lifespan.days
|
||||
elsif crop.parent.present? && crop.parent.median_lifespan.present?
|
||||
planted_at + crop.parent.median_lifespan.days
|
||||
end
|
||||
end
|
||||
|
||||
# days
|
||||
def expected_lifespan
|
||||
if actual_lifespan.present?
|
||||
actual_lifespan
|
||||
elsif crop.median_lifespan.present?
|
||||
crop.median_lifespan
|
||||
elsif crop.parent.present? && crop.parent.median_lifespan.present?
|
||||
crop.parent.median_lifespan
|
||||
end
|
||||
end
|
||||
|
||||
def actual_lifespan
|
||||
return unless planted_at.present? && finished_at.present?
|
||||
(finished_at - planted_at).to_i
|
||||
end
|
||||
|
||||
def days_since_planted
|
||||
(Time.zone.today - planted_at).to_i if planted_at.present?
|
||||
end
|
||||
|
||||
# progress
|
||||
def percentage_grown
|
||||
if finished?
|
||||
100
|
||||
elsif !finish_is_predicatable?
|
||||
nil
|
||||
elsif growing?
|
||||
calculate_percentage_grown
|
||||
elsif planted?
|
||||
0
|
||||
end
|
||||
end
|
||||
|
||||
# states
|
||||
def finish_is_predicatable?
|
||||
crop.annual? && planted_at.present? && finish_predicted_at.present?
|
||||
end
|
||||
|
||||
# Planting has live more then 90 days past predicted finish
|
||||
def super_late?
|
||||
late? && (finish_predicted_at + 90.days) < Time.zone.today
|
||||
end
|
||||
|
||||
def late?
|
||||
crop.annual? && !finished &&
|
||||
planted_at.present? &&
|
||||
finish_predicted_at.present? &&
|
||||
finish_predicted_at <= Time.zone.today
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def calculate_percentage_grown
|
||||
percent = (days_since_planted / expected_lifespan.to_f) * 100
|
||||
(percent > 100 ? 100 : percent)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -11,28 +11,26 @@ class Crop < ActiveRecord::Base
|
||||
has_many :scientific_names, after_add: :update_index, after_remove: :update_index, dependent: :destroy
|
||||
accepts_nested_attributes_for :scientific_names, allow_destroy: true, reject_if: :all_blank
|
||||
has_many :alternate_names, after_add: :update_index, after_remove: :update_index, dependent: :destroy
|
||||
has_many :plantings
|
||||
has_many :plantings, dependent: :destroy
|
||||
has_many :seeds, dependent: :destroy
|
||||
has_many :harvests, dependent: :destroy
|
||||
has_many :photos, through: :plantings
|
||||
has_many :seeds
|
||||
has_many :harvests
|
||||
has_many :plant_parts, -> { uniq.reorder("plant_parts.name") }, through: :harvests
|
||||
belongs_to :creator, class_name: 'Member'
|
||||
belongs_to :requester, class_name: 'Member'
|
||||
belongs_to :parent, class_name: 'Crop'
|
||||
has_many :varieties, class_name: 'Crop', foreign_key: 'parent_id'
|
||||
has_many :varieties, class_name: 'Crop', foreign_key: 'parent_id', dependent: :nullify
|
||||
has_and_belongs_to_many :posts # rubocop:disable Rails/HasAndBelongsToMany
|
||||
|
||||
##
|
||||
## Scopes
|
||||
scope :recent, -> { approved.order(created_at: :desc) }
|
||||
scope :toplevel, -> { approved.where(parent_id: nil) }
|
||||
scope :popular, -> { approved.reorder("plantings_count desc, lower(name) asc") }
|
||||
# ok on sqlite and psql, but not on mysql
|
||||
scope :randomized, -> { approved.reorder('random()') }
|
||||
scope :popular, -> { approved.order("plantings_count desc, lower(name) asc") }
|
||||
scope :pending_approval, -> { where(approval_status: "pending") }
|
||||
scope :approved, -> { where(approval_status: "approved") }
|
||||
scope :rejected, -> { where(approval_status: "rejected") }
|
||||
scope :interesting, -> { approved.has_photos.randomized }
|
||||
scope :interesting, -> { approved.has_photos }
|
||||
scope :has_photos, -> { includes(:photos).where.not(photos: { id: nil }) }
|
||||
|
||||
##
|
||||
@@ -80,18 +78,18 @@ class Crop < ActiveRecord::Base
|
||||
} do
|
||||
mappings dynamic: 'false' do
|
||||
indexes :id, type: 'long'
|
||||
indexes :name, type: 'string', analyzer: 'gs_edgeNGram_analyzer'
|
||||
indexes :approval_status, type: 'string'
|
||||
indexes :name, type: 'text', analyzer: 'gs_edgeNGram_analyzer'
|
||||
indexes :approval_status, type: 'text'
|
||||
indexes :scientific_names do
|
||||
indexes :name,
|
||||
type: 'string',
|
||||
type: 'text',
|
||||
analyzer: 'gs_edgeNGram_analyzer',
|
||||
# Disabling field-length norm (norm). If the norm option is turned on(by default),
|
||||
# higher weigh would be given for shorter fields, which in our case is irrelevant.
|
||||
norms: { enabled: false }
|
||||
end
|
||||
indexes :alternate_names do
|
||||
indexes :name, type: 'string', analyzer: 'gs_edgeNGram_analyzer'
|
||||
indexes :name, type: 'text', analyzer: 'gs_edgeNGram_analyzer'
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -120,12 +118,7 @@ class Crop < ActiveRecord::Base
|
||||
# later we can choose a default photo based on different criteria,
|
||||
# eg. popularity
|
||||
def default_photo
|
||||
# most recent photo
|
||||
return photos.order(created_at: :desc).first if photos.any?
|
||||
|
||||
# Crop has no photos? Look for the most recent harvest with a photo.
|
||||
harvest_with_photo = Harvest.where(crop_id: id).joins(:photos).order('harvests.id DESC').limit(1).first
|
||||
harvest_with_photo.photos.first if harvest_with_photo
|
||||
first_photo(:plantings) || first_photo(:harvests) || first_photo(:seeds)
|
||||
end
|
||||
|
||||
# returns hash indicating whether this crop is grown in
|
||||
@@ -192,7 +185,7 @@ class Crop < ActiveRecord::Base
|
||||
end
|
||||
|
||||
def update_medians
|
||||
plantings.each(&:update_harvest_days)
|
||||
plantings.each(&:update_harvest_days!)
|
||||
update_lifespan_medians
|
||||
update_harvest_medians
|
||||
end
|
||||
@@ -242,4 +235,8 @@ class Crop < ActiveRecord::Base
|
||||
return unless reason_for_rejection == "other" && rejection_notes.blank?
|
||||
errors.add(:rejection_notes, "must be added if the reason for rejection is \"other\"")
|
||||
end
|
||||
|
||||
def first_photo(type)
|
||||
Photo.joins(type).where("#{type}": { crop_id: id }).order("photos.created_at DESC").first
|
||||
end
|
||||
end
|
||||
|
||||
@@ -1,10 +1,10 @@
|
||||
class Forum < ActiveRecord::Base
|
||||
extend FriendlyId
|
||||
include Ownable
|
||||
validates :name, presence: true
|
||||
friendly_id :name, use: %i(slugged finders)
|
||||
|
||||
has_many :posts
|
||||
belongs_to :owner, class_name: "Member"
|
||||
|
||||
def to_s
|
||||
name
|
||||
|
||||
@@ -2,9 +2,9 @@ class Garden < ActiveRecord::Base
|
||||
extend FriendlyId
|
||||
include Geocodable
|
||||
include PhotoCapable
|
||||
include Ownable
|
||||
friendly_id :garden_slug, use: %i(slugged finders)
|
||||
|
||||
belongs_to :owner, class_name: 'Member', foreign_key: 'owner_id', counter_cache: true
|
||||
has_many :plantings, dependent: :destroy
|
||||
has_many :crops, through: :plantings
|
||||
|
||||
|
||||
@@ -2,6 +2,7 @@ class Harvest < ActiveRecord::Base
|
||||
include ActionView::Helpers::NumberHelper
|
||||
extend FriendlyId
|
||||
include PhotoCapable
|
||||
include Ownable
|
||||
|
||||
friendly_id :harvest_slug, use: %i(slugged finders)
|
||||
|
||||
@@ -33,7 +34,6 @@ class Harvest < ActiveRecord::Base
|
||||
##
|
||||
## Relationships
|
||||
belongs_to :crop
|
||||
belongs_to :owner, class_name: 'Member', counter_cache: true
|
||||
belongs_to :plant_part
|
||||
belongs_to :planting
|
||||
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
class Photo < ActiveRecord::Base
|
||||
belongs_to :owner, class_name: 'Member'
|
||||
include Ownable
|
||||
|
||||
PHOTO_CAPABLE = %w(Garden Planting Harvest Seed).freeze
|
||||
|
||||
|
||||
@@ -5,6 +5,8 @@ class PlantPart < ActiveRecord::Base
|
||||
has_many :harvests
|
||||
has_many :crops, -> { uniq }, through: :harvests
|
||||
|
||||
validates :name, presence: true, uniqueness: true
|
||||
|
||||
def to_s
|
||||
name
|
||||
end
|
||||
|
||||
@@ -2,6 +2,9 @@ class Planting < ActiveRecord::Base
|
||||
extend FriendlyId
|
||||
include PhotoCapable
|
||||
include Finishable
|
||||
include Ownable
|
||||
include PredictPlanting
|
||||
include PredictHarvest
|
||||
friendly_id :planting_slug, use: %i(slugged finders)
|
||||
|
||||
# Constants
|
||||
@@ -12,12 +15,7 @@ class Planting < ActiveRecord::Base
|
||||
'graft', 'layering'
|
||||
].freeze
|
||||
|
||||
##
|
||||
## Triggers
|
||||
before_save :calculate_lifespan
|
||||
|
||||
belongs_to :garden
|
||||
belongs_to :owner, class_name: 'Member', counter_cache: true
|
||||
belongs_to :crop, counter_cache: true
|
||||
has_many :harvests, dependent: :destroy
|
||||
|
||||
@@ -59,6 +57,10 @@ class Planting < ActiveRecord::Base
|
||||
in: PLANTED_FROM_VALUES, message: "%<value>s is not a valid planting method"
|
||||
}
|
||||
|
||||
def age_in_days
|
||||
(Time.zone.today - planted_at).to_i if planted_at.present?
|
||||
end
|
||||
|
||||
def planting_slug
|
||||
[
|
||||
owner.login_name,
|
||||
@@ -81,61 +83,20 @@ class Planting < ActiveRecord::Base
|
||||
photos.order(created_at: :desc).first
|
||||
end
|
||||
|
||||
def finished?
|
||||
finished || (finished_at.present? && finished_at <= Time.zone.today)
|
||||
end
|
||||
|
||||
def planted?
|
||||
planted_at.present? && planted_at <= Date.current
|
||||
planted_at.present? && planted_at <= Time.zone.today
|
||||
end
|
||||
|
||||
def finish_predicted_at
|
||||
planted_at + crop.median_lifespan.days if crop.median_lifespan.present? && planted_at.present?
|
||||
end
|
||||
|
||||
def calculate_lifespan
|
||||
self.lifespan = (planted_at.present? && finished_at.present? ? finished_at - planted_at : nil)
|
||||
end
|
||||
|
||||
def expected_lifespan
|
||||
if planted_at.present? && finished_at.present?
|
||||
return (finished_at - planted_at).to_i
|
||||
end
|
||||
crop.median_lifespan
|
||||
end
|
||||
|
||||
def days_since_planted
|
||||
(Time.zone.today - planted_at).to_i if planted_at.present?
|
||||
end
|
||||
|
||||
def percentage_grown
|
||||
return 100 if finished
|
||||
return if planted_at.blank? || expected_lifespan.blank?
|
||||
p = (days_since_planted / expected_lifespan.to_f) * 100
|
||||
return p if p <= 100
|
||||
100
|
||||
end
|
||||
|
||||
def update_harvest_days
|
||||
days_to_first_harvest = nil
|
||||
days_to_last_harvest = nil
|
||||
if planted_at.present? && harvests_with_dates.size.positive?
|
||||
days_to_first_harvest = (first_harvest_date - planted_at).to_i
|
||||
days_to_last_harvest = (last_harvest_date - planted_at).to_i if finished?
|
||||
end
|
||||
update(days_to_first_harvest: days_to_first_harvest, days_to_last_harvest: days_to_last_harvest)
|
||||
end
|
||||
|
||||
def first_harvest_date
|
||||
harvests_with_dates.minimum(:harvested_at)
|
||||
end
|
||||
|
||||
def last_harvest_date
|
||||
harvests_with_dates.maximum(:harvested_at)
|
||||
def growing?
|
||||
planted? && !finished?
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def harvests_with_dates
|
||||
harvests.where.not(harvested_at: nil)
|
||||
end
|
||||
|
||||
# check that any finished_at date occurs after planted_at
|
||||
def finished_must_be_after_planted
|
||||
return unless planted_at && finished_at # only check if we have both
|
||||
|
||||
@@ -2,6 +2,7 @@ class Seed < ActiveRecord::Base
|
||||
extend FriendlyId
|
||||
include PhotoCapable
|
||||
include Finishable
|
||||
include Ownable
|
||||
friendly_id :seed_slug, use: %i(slugged finders)
|
||||
|
||||
TRADABLE_TO_VALUES = %w(nowhere locally nationally internationally).freeze
|
||||
@@ -12,8 +13,6 @@ class Seed < ActiveRecord::Base
|
||||
#
|
||||
# Relationships
|
||||
belongs_to :crop
|
||||
belongs_to :owner, class_name: 'Member', foreign_key: 'owner_id', counter_cache: true
|
||||
|
||||
belongs_to :parent_planting, class_name: 'Planting', foreign_key: 'parent_planting_id' # parent
|
||||
has_many :child_plantings, class_name: 'Planting',
|
||||
foreign_key: 'parent_seed_id', dependent: :nullify # children
|
||||
|
||||
@@ -3,21 +3,19 @@ class CropSearchService
|
||||
def self.search(query)
|
||||
if ENV['GROWSTUFF_ELASTICSEARCH'] == "true"
|
||||
search_str = query.nil? ? "" : query.downcase
|
||||
response = Crop.__elasticsearch__.search( # Finds documents which match any field, but uses the _score from
|
||||
# the best field insead of adding up _score from each field.
|
||||
response = Crop.__elasticsearch__.search(
|
||||
query: {
|
||||
multi_match: {
|
||||
query: search_str.to_s,
|
||||
analyzer: "standard",
|
||||
fields: ["name",
|
||||
"scientific_names.scientific_name",
|
||||
"alternate_names.name"]
|
||||
bool: {
|
||||
filter: {
|
||||
term: { "approval_status" => "approved" }
|
||||
},
|
||||
must: {
|
||||
query_string: {
|
||||
query: "*#{search_str}*"
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
filter: {
|
||||
term: { approval_status: "approved" }
|
||||
},
|
||||
size: 50
|
||||
}
|
||||
)
|
||||
response.records.to_a
|
||||
else
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
.well
|
||||
.row
|
||||
.col-md-4
|
||||
= link_to image_tag((crop.default_photo ? crop.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
= link_to image_tag(crop_image_path(crop),
|
||||
alt: '',
|
||||
class: 'img crop-image'),
|
||||
crop
|
||||
|
||||
@@ -4,9 +4,9 @@
|
||||
Nobody has planted this crop yet.
|
||||
- else
|
||||
%ul
|
||||
- crop.plantings.take(3).each do |planting|
|
||||
- crop.plantings.order(planted_at: :desc).limit(3).each do |planting|
|
||||
%li
|
||||
= link_to display_planting(planting), planting_path(planting)
|
||||
= link_to planting, planting_path(planting)
|
||||
= render partial: 'members/location', locals: { member: planting.owner }
|
||||
%small
|
||||
= distance_of_time_in_words(planting.created_at, Time.zone.now)
|
||||
@@ -18,4 +18,3 @@
|
||||
%p= link_to "Plant #{crop.name}", new_planting_path(crop_id: crop.id)
|
||||
- else
|
||||
= render partial: 'shared/signin_signup', locals: { to: "track your #{crop.name} plantings" }
|
||||
|
||||
|
||||
@@ -2,7 +2,7 @@
|
||||
.thumbnail
|
||||
.crop-thumbnail
|
||||
- if crop
|
||||
= link_to image_tag((crop.default_photo ? crop.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
= link_to image_tag(crop_image_path(crop),
|
||||
alt: crop.name, class: 'img'),
|
||||
crop
|
||||
.cropinfo
|
||||
|
||||
@@ -13,12 +13,9 @@
|
||||
.col-md-10
|
||||
.row
|
||||
- if garden.plantings.current.size.positive?
|
||||
- garden.plantings.current.includes(:crop).each do |planting|
|
||||
.col-md-2.col-sm-6.col-xs-6
|
||||
.hover-wrapper
|
||||
.text
|
||||
= render 'plantings/actions', planting: planting
|
||||
= render partial: "plantings/thumbnail", locals: { planting: planting }
|
||||
- garden.plantings.current.order(created_at: :desc).includes(:crop, :photos).each do |planting|
|
||||
.col-md-2.col-sm-4.col-xs-6
|
||||
= render "plantings/thumbnail", planting: planting
|
||||
- else
|
||||
.col-md-2.col-sm-6.col-xs-6 no plantings
|
||||
- if can?(:edit, garden)
|
||||
|
||||
@@ -1,3 +1,3 @@
|
||||
= link_to image_tag((garden.default_photo ? garden.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
= link_to image_tag(garden_image_path(garden),
|
||||
alt: garden.name, class: 'img-responsive'),
|
||||
garden_path(garden)
|
||||
|
||||
@@ -8,7 +8,7 @@
|
||||
.panel-body{ id: "gardens_panel_body" }
|
||||
.row
|
||||
.col-md-4
|
||||
= link_to image_tag((garden.default_photo ? garden.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
= link_to image_tag(garden_image_path(garden),
|
||||
alt: garden.name, class: 'img'),
|
||||
garden_path(garden)
|
||||
.col-md-8
|
||||
|
||||
@@ -8,7 +8,7 @@
|
||||
.panel-body
|
||||
.row
|
||||
.col-md-4
|
||||
= link_to image_tag((harvest.default_photo ? harvest.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
= link_to image_tag(harvest_image_path(harvest),
|
||||
alt: harvest.crop.name, class: 'img'),
|
||||
harvest
|
||||
.col-md-8
|
||||
@@ -21,8 +21,8 @@
|
||||
%dd= display_quantity(harvest)
|
||||
%dt Harvest date :
|
||||
%dd= harvest.harvested_at
|
||||
%dd Notes:
|
||||
%dt=display_harvest_description(harvest)
|
||||
%dt Notes:
|
||||
%dd= display_harvest_description(harvest)
|
||||
- if harvest.planting.present?
|
||||
%dt Harvested from
|
||||
%dd= link_to(harvest.planting, planting_path(harvest.planting))
|
||||
|
||||
@@ -1,11 +1,10 @@
|
||||
- harvests.each do |h|
|
||||
- cache h do
|
||||
.row
|
||||
.col-md-3.col-xs-4{ style: 'padding-bottom: 6px' }
|
||||
.col-lg-6.col-md-3.col-xs-4.homepage-listing
|
||||
= render 'harvests/image_with_popover', harvest: h
|
||||
.col-md-9.col-xs-4
|
||||
.col-lg-3.col-md-9.col-xs-4
|
||||
= link_to h.crop, crop_path(h.crop)
|
||||
%br/
|
||||
%small
|
||||
%i
|
||||
= h.owner.location
|
||||
%i= h.owner.location
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
planting_path(@harvest.planting)
|
||||
in
|
||||
= link_to @harvest.planting.garden, garden_path(@harvest.planting.garden)
|
||||
- elsif @matching_plantings && @harvest.owner == current_member
|
||||
- elsif @matching_plantings && @matching_plantings.any? && @harvest.owner == current_member
|
||||
Is this from one of these plantings?
|
||||
= form_for(@harvest) do |f|
|
||||
- @matching_plantings.each do |planting|
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
.thumbnail
|
||||
.harvest-thumbnail
|
||||
- if harvest
|
||||
= link_to image_tag((harvest.default_photo ? harvest.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
= link_to image_tag(harvest_image_path(harvest),
|
||||
alt: harvest.crop.name, class: 'img'),
|
||||
harvest
|
||||
.harvestinfo
|
||||
|
||||
@@ -24,8 +24,10 @@
|
||||
%ul.list-inline
|
||||
%li The data on this page is available in the following formats:
|
||||
- if @owner
|
||||
%li= link_to "RSS", harvests_by_owner_path(@owner, format: 'rss')
|
||||
%li= link_to "CSV", harvests_by_owner_path(@owner, format: 'csv')
|
||||
%li= link_to "JSON", harvests_by_owner_path(@owner, format: 'json')
|
||||
- else
|
||||
%li= link_to "RSS", harvests_path(format: 'rss')
|
||||
%li= link_to "CSV", harvests_path(format: 'csv')
|
||||
%li= link_to "JSON", harvests_path(format: 'json')
|
||||
|
||||
19
app/views/harvests/index.rss.haml
Normal file
19
app/views/harvests/index.rss.haml
Normal file
@@ -0,0 +1,19 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
%rss{ version: 2.0 }
|
||||
%channel
|
||||
%title
|
||||
Recent harvests from #{@owner ? @owner : 'all members'} (#{ENV['GROWSTUFF_SITE_NAME']})
|
||||
%link= harvests_url
|
||||
- @harvests.each do |harvest|
|
||||
%item
|
||||
%title #{harvest.owner.login_name}'s #{harvest.crop.name}
|
||||
%pubdate= harvest.harvested_at.to_s(:rfc822)
|
||||
%description
|
||||
:escaped
|
||||
<p>Crop: #{harvest.crop ? harvest.crop : 'unknown' }</p>
|
||||
<p>Quantity: #{harvest.quantity ? harvest.quantity : 'unknown' }</p>
|
||||
<p>Harvested on: #{harvest.harvested_at ? harvest.harvested_at : 'unknown' }</p>
|
||||
:escaped_markdown
|
||||
#{ strip_tags harvest.description }
|
||||
%link= harvest_url(harvest)
|
||||
%guid= harvest_url(harvest)
|
||||
@@ -1,16 +1,13 @@
|
||||
.col-md-12
|
||||
|
||||
%h1= ENV['GROWSTUFF_SITE_NAME']
|
||||
|
||||
.row
|
||||
.col-md-8.info
|
||||
%p= t('.intro', site_name: ENV['GROWSTUFF_SITE_NAME'])
|
||||
|
||||
= render partial: 'stats'
|
||||
.col-md-4.signup
|
||||
%p= t('.perks')
|
||||
%p= link_to(t('.sign_up'), new_member_registration_path, class: 'btn btn-primary btn-lg')
|
||||
%p
|
||||
%small
|
||||
= t('.already_html', sign_in: link_to(t('.sign_in_linktext'), new_member_session_path))
|
||||
|
||||
.container
|
||||
.row
|
||||
.col-md-12
|
||||
%h1= ENV['GROWSTUFF_SITE_NAME']
|
||||
.col-md-8.info
|
||||
%p= t('.intro', site_name: ENV['GROWSTUFF_SITE_NAME'])
|
||||
= render partial: 'stats'
|
||||
.col-md-4.signup
|
||||
%p= t('.perks')
|
||||
%p= link_to(t('.sign_up'), new_member_registration_path, class: 'btn btn-primary btn-lg')
|
||||
%p
|
||||
%small
|
||||
= t('.already_html', sign_in: link_to(t('.sign_in_linktext'), new_member_session_path))
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
- cache cache_key_for(Crop, 'interesting'), expires_in: 1.day do
|
||||
.row
|
||||
%h2= t('.our_crops')
|
||||
- Crop.interesting.includes(:scientific_names, :photos).limit(8).each do |c|
|
||||
.col-md-4.col-sm-3.col-xs-6= render 'crops/thumbnail', crop: c
|
||||
- Crop.interesting.includes(:scientific_names, :photos).shuffle.first(12).each do |c|
|
||||
.col-lg-2.col-md-4.col-sm-3.col-xs-6
|
||||
= render 'crops/thumbnail', crop: c
|
||||
|
||||
@@ -1,3 +1,3 @@
|
||||
- cache cache_key_for(Harvest) do
|
||||
%h2 Recently Harvested
|
||||
= render 'harvests/list', harvests: Harvest.includes(:crop, :owner, :photos).has_photos.recent.first(5)
|
||||
= render 'harvests/list', harvests: Harvest.includes(:crop, :owner, :photos).has_photos.recent.first(6)
|
||||
|
||||
@@ -1,3 +1,3 @@
|
||||
- cache cache_key_for(Planting, 'home'), expires_in: 1.day do
|
||||
%h2= t('.recently_planted')
|
||||
= render 'plantings/list', plantings: Planting.includes(:crop, garden: :owner).has_photos.recent.limit(5)
|
||||
= render 'plantings/list', plantings: Planting.includes(:crop, garden: :owner).has_photos.recent.limit(6)
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
- cache cache_key_for(Seed, 'interesting'), expires_in: 1.day do
|
||||
%h2= t('.title')
|
||||
.row
|
||||
- Seed.current.tradable.order(created_at: :desc).limit(6).each do |seed|
|
||||
- Seed.current.tradable.includes(:owner, :crop).order(created_at: :desc).limit(6).each do |seed|
|
||||
.col-md-2.col-sm-2.col-xs-6
|
||||
.thumbnail.seed-thumbnail
|
||||
- cache cache_key_for(Crop, seed.id) do
|
||||
|
||||
@@ -19,11 +19,11 @@
|
||||
= render 'blurb'
|
||||
|
||||
.row
|
||||
.col-md-6.col-sm-12
|
||||
.col-lg-8.col-md-6.col-sm-12
|
||||
= render 'crops'
|
||||
.col-md-3.col-sm-6
|
||||
.col-lg-2.col-md-3.col-sm-6
|
||||
= render 'plantings'
|
||||
.col-md-3.col-sm-6
|
||||
.col-lg-2.col-md-3.col-sm-6
|
||||
= render 'harvests'
|
||||
.col-md-12
|
||||
- cache cache_key_for(Crop, 'recent') do
|
||||
|
||||
@@ -5,7 +5,7 @@
|
||||
%body
|
||||
= render partial: "layouts/header"
|
||||
|
||||
.container#maincontainer
|
||||
#maincontainer
|
||||
.row
|
||||
.col-md-12
|
||||
- if content_for?(:title)
|
||||
|
||||
@@ -1,4 +1,3 @@
|
||||
%h2 #{member.login_name}'s gardens
|
||||
.tabbable
|
||||
%ul.nav.nav-tabs
|
||||
- first_garden = true
|
||||
@@ -45,10 +44,10 @@
|
||||
= link_to "Add photo", new_photo_path(type: "garden", id: g.id), class: 'btn btn-primary'
|
||||
|
||||
%h3 What's planted here?
|
||||
.row
|
||||
.card-row
|
||||
- unless g.featured_plantings.empty?
|
||||
- g.featured_plantings.each.with_index do |planting|
|
||||
.col-xs-12.col-lg-6
|
||||
.card
|
||||
= render partial: "plantings/card", locals: { planting: planting }
|
||||
|
||||
%p
|
||||
|
||||
6
app/views/members/_harvests.html.haml
Normal file
6
app/views/members/_harvests.html.haml
Normal file
@@ -0,0 +1,6 @@
|
||||
.card-row-short
|
||||
- harvests.each do |harvest|
|
||||
.card
|
||||
= render partial: "harvests/card", locals: { harvest: harvest }
|
||||
- if !harvests.any?
|
||||
#{member.login_name} hasn't harvested anything yet.
|
||||
@@ -1,6 +1,6 @@
|
||||
- if member.latitude && member.longitude
|
||||
#membermap
|
||||
%p
|
||||
%p.pull-right
|
||||
See other members, plantings, seeds and more near
|
||||
= link_to member.location, place_path(member.location, anchor: "members")
|
||||
- else
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
|
||||
%h3 Activity
|
||||
|
||||
%ul.list-inline
|
||||
%ul.activity-list
|
||||
%li
|
||||
- if !member.plantings.empty?
|
||||
= link_to localize_plural(member.plantings, Planting), plantings_by_owner_path(owner: member)
|
||||
|
||||
@@ -9,31 +9,40 @@
|
||||
= tag("meta", property: "og:site_name", content: ENV['GROWSTUFF_SITE_NAME'])
|
||||
- content_for :buttonbar do
|
||||
- if can? :update, @member
|
||||
= link_to 'Edit profile', edit_member_registration_path, class: 'btn btn-default'
|
||||
= link_to 'Edit profile', edit_member_registration_path, class: 'btn btn-default pull-right'
|
||||
- if can?(:create, Notification) && current_member != @member
|
||||
= link_to 'Send message', new_notification_path(recipient_id: @member.id), class: 'btn btn-default'
|
||||
= link_to 'Send message', new_notification_path(recipient_id: @member.id), class: 'btn btn-default pull-right'
|
||||
|
||||
- if current_member && current_member != @member # must be logged in, can't follow yourself
|
||||
- follow = current_member.get_follow(@member)
|
||||
- if !follow && can?(:create, Follow) # not already following
|
||||
= link_to 'Follow', follows_path(followed_id: @member.id), method: :post, class: 'btn btn-default'
|
||||
= link_to 'Follow', follows_path(followed_id: @member.id), method: :post, class: 'btn btn-default pull-right'
|
||||
- if follow && can?(:destroy, follow) # already following
|
||||
= link_to 'Unfollow', follow_path(follow), method: :delete, class: 'btn btn-default'
|
||||
= link_to 'Unfollow', follow_path(follow), method: :delete, class: 'btn btn-default pull-right'
|
||||
|
||||
- content_for :member_rss_login_name, @member.login_name
|
||||
- content_for :member_rss_slug, @member.slug
|
||||
|
||||
.row
|
||||
|
||||
.col-md-9
|
||||
= render partial: "map", locals: { member: @member }
|
||||
= render partial: "bio", locals: { member: @member }
|
||||
= render partial: "gardens", locals: { member: @member, gardens: @gardens }
|
||||
.col-md-3
|
||||
= render partial: "map", locals: { member: @member }
|
||||
.col-md-2.profile-sidebar
|
||||
= render partial: "avatar", locals: { member: @member }
|
||||
= render partial: "bio", locals: { member: @member }
|
||||
= render partial: "roles", locals: { member: @member }
|
||||
= render partial: "stats", locals: { member: @member }
|
||||
= render partial: "contact", locals: { member: @member,
|
||||
twitter_auth: @twitter_auth,
|
||||
flickr_auth: @flickr_auth,
|
||||
facebook_auth: @facebook_auth }
|
||||
|
||||
.col-md-10
|
||||
%ul.nav.nav-pills.nav-justified
|
||||
%li.active
|
||||
%a{ "data-toggle" => "tab", href: "#gardens" } Gardens
|
||||
%li
|
||||
%a{ "data-toggle" => "tab", href: "#harvests" } Harvests
|
||||
.tab-content.profile-activity
|
||||
.tab-pane.active#gardens
|
||||
= render partial: "gardens", locals: { member: @member, gardens: @gardens }
|
||||
.tab-pane#harvests
|
||||
= render partial: "harvests", locals: { member: @member, harvests: @harvests }
|
||||
|
||||
@@ -8,5 +8,5 @@
|
||||
= render 'shared/buttons/harvest_planting', planting: planting
|
||||
= render 'shared/buttons/save_seeds', planting: planting
|
||||
|
||||
- if can? :destroy, planting
|
||||
= render 'shared/buttons/delete', path: planting
|
||||
- if can? :destroy, planting
|
||||
= render 'shared/buttons/delete', path: planting
|
||||
|
||||
20
app/views/plantings/_badges.html.haml
Normal file
20
app/views/plantings/_badges.html.haml
Normal file
@@ -0,0 +1,20 @@
|
||||
// Finish times
|
||||
- if planting.finish_is_predicatable?
|
||||
- if planting.super_late?
|
||||
%span.badge.badge-super-late= t('.super_late')
|
||||
= render 'shared/buttons/finish_planting', planting: planting
|
||||
- elsif planting.late?
|
||||
%span.badge.badge-late= t('.late_finishing')
|
||||
- else
|
||||
%span.badge
|
||||
= days_from_now_to_finished(planting)
|
||||
= t('.days_until_finished')
|
||||
|
||||
// Harvest times
|
||||
- unless planting.super_late?
|
||||
- if planting.harvest_time?
|
||||
%span.badge.badge-harvest= t('.harvesting_now')
|
||||
- elsif planting.before_harvest_time?
|
||||
%span.badge
|
||||
= days_from_now_to_first_harvest(planting)
|
||||
= t('.days_until_harvest')
|
||||
@@ -8,7 +8,7 @@
|
||||
.panel-body
|
||||
.row
|
||||
.col-xs-12.col-md-5
|
||||
= link_to image_tag((planting.default_photo ? planting.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
= link_to image_tag(planting_image_path(planting),
|
||||
alt: planting.crop_id, class: 'img img-responsive'),
|
||||
planting
|
||||
.col-xs-12.col-md-7
|
||||
|
||||
@@ -5,6 +5,6 @@
|
||||
planting,
|
||||
rel: "popover",
|
||||
'data-trigger': 'hover',
|
||||
'data-title': planting.to_s,
|
||||
'data-title': planting.crop.name,
|
||||
'data-content': render('plantings/popover', planting: planting),
|
||||
'data-html': true
|
||||
|
||||
@@ -1,15 +1,10 @@
|
||||
- plantings.each do |p|
|
||||
- cache p do
|
||||
.row
|
||||
.col-md-3.col-xs-4.homepage-listing
|
||||
.col-lg-6.col-md-3.col-xs-4.homepage-listing
|
||||
= render 'plantings/image_with_popover', planting: p
|
||||
.col-md-9.col-xs-4
|
||||
.col-lg-3.col-md-9.col-xs-4
|
||||
= link_to p.crop, p.crop
|
||||
in
|
||||
= succeed "'s" do
|
||||
= link_to p.garden.owner, p.garden.owner
|
||||
= link_to display_garden_name(p.garden), p.garden
|
||||
%br/
|
||||
%small
|
||||
%i
|
||||
= p.location
|
||||
%i= p.location
|
||||
|
||||
@@ -15,4 +15,3 @@
|
||||
= link_to 'Plant something', new_planting_path, class: 'btn btn-primary'
|
||||
- else
|
||||
= render partial: 'shared/signin_signup', locals: { to: "track what you've planted" }
|
||||
|
||||
|
||||
@@ -1,7 +1,11 @@
|
||||
%p
|
||||
%small
|
||||
Quantity:
|
||||
= planting.quantity ? planting.quantity : 'unknown'
|
||||
%br/
|
||||
Planted on:
|
||||
= planting.planted_at.to_s
|
||||
- if planting.quantity.present?
|
||||
%p
|
||||
%small
|
||||
Quantity:
|
||||
= planting.quantity
|
||||
|
||||
- if planting.planted_at.present?
|
||||
%p
|
||||
%small
|
||||
Planted:
|
||||
= planting.planted_at.to_s
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
- if planting.crop.perennial
|
||||
%p Perennial
|
||||
= render "plantings/progress_bar", status: "perennial", progress: nil
|
||||
- elsif !planting.planted?
|
||||
- if show_explanation
|
||||
%p Progress: 0% - not planted yet
|
||||
%p= t('.progress_0_not_planted_yet')
|
||||
= render "plantings/progress_bar", status: "not planted", progress: 0
|
||||
- elsif planting.finished?
|
||||
= render "plantings/progress_bar", status: 'finished', progress: 100
|
||||
|
||||
@@ -1,10 +1,13 @@
|
||||
.thumbnail
|
||||
.planting-thumbnail
|
||||
- if planting
|
||||
= link_to image_tag((planting.default_photo ? planting.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
alt: planting.crop.name, class: 'img'),
|
||||
planting
|
||||
.plantinginfo
|
||||
.planting
|
||||
.planting-badges
|
||||
= render 'plantings/badges', planting: planting
|
||||
.hover-wrapper
|
||||
.thumbnail
|
||||
.planting-thumbnail{ class: planting_classes(planting) }
|
||||
= link_to image_tag(planting_image_path(planting),
|
||||
alt: planting.crop.name, class: 'img'), planting_path(planting)
|
||||
= render 'plantings/progress', planting: planting, show_explanation: false
|
||||
.planting-name
|
||||
= render 'plantings/progress', planting: planting, show_explanation: false
|
||||
= link_to planting.crop.name, planting
|
||||
.text
|
||||
.planting-actions= render 'plantings/actions', planting: planting
|
||||
|
||||
@@ -3,16 +3,16 @@
|
||||
= render 'nav', owner: @owner, show_all: @show_all
|
||||
|
||||
- if @owner
|
||||
= link_to "View #{@owner}'s profile >>", member_path(@owner)
|
||||
= link_to t('.view_owners_profile', owner: @owner), member_path(@owner)
|
||||
|
||||
.pagination
|
||||
= page_entries_info @plantings
|
||||
= will_paginate @plantings
|
||||
|
||||
.row
|
||||
.card-row
|
||||
- unless @plantings.empty?
|
||||
- @plantings.each.with_index do |planting|
|
||||
.col-xs-12.col-lg-6
|
||||
.card
|
||||
= render partial: "plantings/card", locals: { planting: planting }
|
||||
|
||||
.pagination
|
||||
@@ -20,12 +20,7 @@
|
||||
= will_paginate @plantings
|
||||
|
||||
%ul.list-inline
|
||||
%li The data on this page is available in the following formats:
|
||||
- if @owner
|
||||
%li= link_to "CSV", plantings_by_owner_path(@owner, format: 'csv')
|
||||
%li= link_to "JSON", plantings_by_owner_path(@owner, format: 'json')
|
||||
%li= link_to "RSS", plantings_by_owner_path(@owner, format: 'rss')
|
||||
- else
|
||||
%li= link_to "CSV", plantings_path(format: 'csv')
|
||||
%li= link_to "JSON", plantings_path(format: 'json')
|
||||
%li= link_to "RSS", plantings_path(format: 'rss')
|
||||
%li= t('.the_data_on_this_page_is_available_in_the_following_formats')
|
||||
- ['csv', 'json', 'rss'].each do |format|
|
||||
%li= link_to format.upcase,
|
||||
(@owner ? plantings_by_owner_path(@owner, format: format) : plantings_path(format: format))
|
||||
|
||||
@@ -36,8 +36,12 @@
|
||||
= link_to 'Edit Post', edit_post_path(@post), class: 'btn btn-default btn-xs'
|
||||
- if can? :destroy, @post
|
||||
= link_to 'Delete Post', @post, method: :delete,
|
||||
data: { confirm: 'Are you sure?' },
|
||||
class: 'btn btn-default btn-xs'
|
||||
data: { confirm: 'Are you sure?' },
|
||||
class: 'btn btn-default btn-xs'
|
||||
|
||||
- if @post.comments.count > 10 && can?(:create, Comment)
|
||||
.post-actions
|
||||
= link_to 'Comment', new_comment_path(post_id: @post.id), class: 'btn btn-primary'
|
||||
|
||||
= render partial: "comments", locals: { post: @post }
|
||||
|
||||
|
||||
@@ -8,7 +8,7 @@
|
||||
.panel-body
|
||||
.row
|
||||
.col-md-4
|
||||
= link_to image_tag((seed.crop.default_photo ? seed.crop.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
= link_to image_tag(seed_image_path(seed),
|
||||
alt: seed.crop.name, class: 'img'),
|
||||
seed.crop
|
||||
.col-md-8
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
.thumbnail
|
||||
.seed-thumbnail
|
||||
= link_to image_tag((seed.default_photo ? seed.default_photo.thumbnail_url : 'placeholder_150.png'),
|
||||
= link_to image_tag(seed_image_path(seed),
|
||||
alt: seed.crop.name, class: 'img'),
|
||||
seed_path(seed)
|
||||
.seedinfo
|
||||
|
||||
@@ -24,10 +24,10 @@
|
||||
= page_entries_info @seeds
|
||||
= will_paginate @seeds
|
||||
|
||||
.row
|
||||
.card-row
|
||||
- unless @seeds.empty?
|
||||
- @seeds.each do |seed|
|
||||
.col-md-6
|
||||
.seedcard
|
||||
= render 'seeds/card', seed: seed
|
||||
|
||||
.pagination
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
- unless planting.finished
|
||||
- if can?(:edit, planting) && !planting.finished
|
||||
= link_to planting_path(planting, planting: { finished: 1 }),
|
||||
method: :put, class: 'btn btn-default btn-xs append-date' do
|
||||
%span.glyphicon.glyphicon-ok{ title: "Finished" }
|
||||
|
||||
@@ -52,6 +52,11 @@ en:
|
||||
location_helper: If you have a location set in your profile, it will be used when you create a new garden.
|
||||
location: "%{owner}'s %{garden}"
|
||||
updated: Garden was successfully updated.
|
||||
overview:
|
||||
gardensphoto: gardens/photo
|
||||
plantingsthumbnail: plantings/thumbnail
|
||||
no_plantings: no plantings
|
||||
gardensactions: gardens/actions
|
||||
harvests:
|
||||
created: Harvest was successfully created.
|
||||
index:
|
||||
@@ -188,7 +193,18 @@ en:
|
||||
crop_plantings: Everyone's %{crop} plantings
|
||||
default: Everyone's plantings
|
||||
owner_plantings: "%{owner} plantings"
|
||||
view_owners_profile: View %{owner}'s profile >>
|
||||
the_data_on_this_page_is_available_in_the_following_formats: 'The data on this page is available in the following formats:'
|
||||
string: "%{crop} planting in %{garden} by %{owner}"
|
||||
badges:
|
||||
late_finishing: late finishing
|
||||
super_late: super late
|
||||
sharedbuttonsfinish_planting: shared/buttons/finish_planting
|
||||
days_until_finished: days until finished
|
||||
harvesting_now: harvesting now
|
||||
days_until_harvest: days until harvest
|
||||
progress:
|
||||
progress_0_not_planted_yet: 'Progress: 0% - not planted yet'
|
||||
posts:
|
||||
index:
|
||||
title:
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
class SetPredictionData < ActiveRecord::Migration
|
||||
def up
|
||||
say "Updating all plantings time to first harvest"
|
||||
Planting.all.each(&:update_harvest_days)
|
||||
Planting.all.each(&:update_harvest_days!)
|
||||
say "Updating crop median time to first harvest, and lifespan"
|
||||
Crop.all.each do |crop|
|
||||
crop.update_lifespan_medians
|
||||
|
||||
6
db/migrate/20180401220637_add_member_count_caches.rb
Normal file
6
db/migrate/20180401220637_add_member_count_caches.rb
Normal file
@@ -0,0 +1,6 @@
|
||||
class AddMemberCountCaches < ActiveRecord::Migration
|
||||
def change
|
||||
add_column :members, :photos_count, :integer
|
||||
add_column :members, :forums_count, :integer
|
||||
end
|
||||
end
|
||||
@@ -11,7 +11,7 @@
|
||||
#
|
||||
# It's strongly recommended that you check this file into your version control system.
|
||||
|
||||
ActiveRecord::Schema.define(version: 20180213005731) do
|
||||
ActiveRecord::Schema.define(version: 20180401220637) do
|
||||
|
||||
# These are extensions that must be enabled in order to support this database
|
||||
enable_extension "plpgsql"
|
||||
@@ -318,6 +318,8 @@ ActiveRecord::Schema.define(version: 20180213005731) do
|
||||
t.integer "harvests_count"
|
||||
t.integer "seeds_count"
|
||||
t.datetime "deleted_at"
|
||||
t.integer "photos_count"
|
||||
t.integer "forums_count"
|
||||
end
|
||||
|
||||
add_index "members", ["confirmation_token"], name: "index_members_on_confirmation_token", unique: true, using: :btree
|
||||
|
||||
11
lib/tasks/search.rake
Normal file
11
lib/tasks/search.rake
Normal file
@@ -0,0 +1,11 @@
|
||||
namespace :search do
|
||||
desc "Create elastic search index"
|
||||
task create: :environment do
|
||||
puts Crop.__elasticsearch__.create_index! force: true
|
||||
end
|
||||
|
||||
desc 'Refresh elastic search index'
|
||||
task refresh: :environment do
|
||||
puts Crop.__elasticsearch__.refresh_index!
|
||||
end
|
||||
end
|
||||
7
script/install_codeclimate.sh
Executable file
7
script/install_codeclimate.sh
Executable file
@@ -0,0 +1,7 @@
|
||||
#!/bin/bash
|
||||
|
||||
if [ "${COVERAGE}" = "true" ]; then
|
||||
set -euv
|
||||
curl -L https://codeclimate.com/downloads/test-reporter/test-reporter-latest-linux-amd64 > ./cc-test-reporter;
|
||||
chmod +x ./cc-test-reporter;
|
||||
fi
|
||||
15
script/install_elasticsearch.sh
Executable file
15
script/install_elasticsearch.sh
Executable file
@@ -0,0 +1,15 @@
|
||||
#!/bin/bash
|
||||
|
||||
if [ "${GROWSTUFF_ELASTICSEARCH}" = "true" ]; then
|
||||
[[ -z "$VERSION" ]] && VERSION="6.2.3"
|
||||
set -euv
|
||||
sudo dpkg -r elasticsearch
|
||||
wget https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-${VERSION}.deb
|
||||
wget https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-${VERSION}.deb.sha512
|
||||
shasum -a 512 -c elasticsearch-${VERSION}.deb.sha512
|
||||
sudo dpkg -i --force-confnew elasticsearch-${VERSION}.deb
|
||||
|
||||
sudo service elasticsearch start
|
||||
sleep 10
|
||||
curl -v localhost:9200
|
||||
fi
|
||||
@@ -1,12 +0,0 @@
|
||||
#!/bin/bash
|
||||
set -euv
|
||||
|
||||
gem install --update overcommit rubocop haml-lint bundler-audit
|
||||
npm install
|
||||
pip install yamllint --user
|
||||
|
||||
overcommit --install
|
||||
overcommit --sign
|
||||
overcommit --sign pre-commit
|
||||
|
||||
bundle-audit update
|
||||
14
script/install_linters.sh
Executable file
14
script/install_linters.sh
Executable file
@@ -0,0 +1,14 @@
|
||||
#!/bin/bash
|
||||
|
||||
if [ "${STATIC_CHECKS}" = "true" ]; then
|
||||
set -euv
|
||||
gem install --update overcommit rubocop haml-lint bundler-audit;
|
||||
npm install;
|
||||
pip install yamllint --user;
|
||||
|
||||
overcommit --install;
|
||||
overcommit --sign;
|
||||
overcommit --sign pre-commit;
|
||||
|
||||
bundle-audit update;
|
||||
fi
|
||||
@@ -5,18 +5,23 @@ describe Charts::GardensController do
|
||||
let(:valid_params) { { name: 'My second Garden' } }
|
||||
|
||||
let(:garden) { FactoryBot.create :garden }
|
||||
|
||||
context "when not signed in" do
|
||||
describe 'GET timeline' do
|
||||
before { get :timeline, garden_id: garden.to_param }
|
||||
|
||||
it { expect(response).to be_success }
|
||||
end
|
||||
end
|
||||
|
||||
context "when signed in" do
|
||||
before(:each) { sign_in member }
|
||||
|
||||
let!(:member) { FactoryBot.create(:member) }
|
||||
|
||||
describe 'GET timeline' do
|
||||
before { get :timeline, garden_id: garden.to_param }
|
||||
|
||||
it { expect(response).to be_success }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -2,6 +2,7 @@ require 'rails_helper'
|
||||
|
||||
describe CommentsController do
|
||||
subject { response }
|
||||
|
||||
let(:member) { FactoryBot.create(:member) }
|
||||
|
||||
before(:each) do
|
||||
@@ -20,6 +21,7 @@ describe CommentsController do
|
||||
|
||||
describe "returns an RSS feed" do
|
||||
before { get :index, format: "rss" }
|
||||
|
||||
it { is_expected.to be_success }
|
||||
it { is_expected.to render_template("comments/index") }
|
||||
it { expect(response.content_type).to eq("application/rss+xml") }
|
||||
@@ -81,11 +83,13 @@ describe CommentsController do
|
||||
expect(response).to redirect_to(comment.post)
|
||||
end
|
||||
end
|
||||
|
||||
describe "not my comment" do
|
||||
let(:comment) { FactoryBot.create :comment }
|
||||
|
||||
it { expect(response).not_to be_success }
|
||||
end
|
||||
|
||||
describe "attempting to change post_id" do
|
||||
let(:post) { FactoryBot.create :post, subject: 'our post' }
|
||||
let(:other_post) { FactoryBot.create :post, subject: 'the other post' }
|
||||
|
||||
@@ -16,6 +16,7 @@ describe CropsController do
|
||||
describe "GET crop wrangler homepage" do
|
||||
describe 'fetches the crop wrangler homepage' do
|
||||
before { get :wrangle }
|
||||
|
||||
it { is_expected.to be_success }
|
||||
it { is_expected.to render_template("crops/wrangle") }
|
||||
it { expect(assigns[:crop_wranglers]).to eq(Role.crop_wranglers) }
|
||||
@@ -25,6 +26,7 @@ describe CropsController do
|
||||
describe "GET crop hierarchy " do
|
||||
describe 'fetches the crop hierarchy page' do
|
||||
before { get :hierarchy }
|
||||
|
||||
it { is_expected.to be_success }
|
||||
it { is_expected.to render_template("crops/hierarchy") }
|
||||
end
|
||||
@@ -33,6 +35,7 @@ describe CropsController do
|
||||
describe "GET crop search" do
|
||||
describe 'fetches the crop search page' do
|
||||
before { get :search }
|
||||
|
||||
it { is_expected.to be_success }
|
||||
it { is_expected.to render_template("crops/search") }
|
||||
end
|
||||
@@ -41,6 +44,7 @@ describe CropsController do
|
||||
describe "GET RSS feed" do
|
||||
describe "returns an RSS feed" do
|
||||
before { get :index, format: "rss" }
|
||||
|
||||
it { is_expected.to be_success }
|
||||
it { is_expected.to render_template("crops/index") }
|
||||
it { expect(response.content_type).to eq("application/rss+xml") }
|
||||
|
||||
@@ -5,15 +5,20 @@ RSpec.describe GardensController, type: :controller do
|
||||
let(:valid_params) { { name: 'My second Garden' } }
|
||||
|
||||
let(:garden) { FactoryBot.create :garden }
|
||||
|
||||
context "when not signed in" do
|
||||
describe 'GET new' do
|
||||
before { get :new, id: garden.to_param }
|
||||
|
||||
it { expect(response).to redirect_to(new_member_session_path) }
|
||||
end
|
||||
|
||||
describe 'PUT create' do
|
||||
before { put :create, garden: valid_params }
|
||||
|
||||
it { expect(response).to redirect_to(new_member_session_path) }
|
||||
end
|
||||
|
||||
describe 'changing existing records' do
|
||||
before do
|
||||
allow(Garden).to receive(:find).and_return(:garden)
|
||||
@@ -23,20 +28,27 @@ RSpec.describe GardensController, type: :controller do
|
||||
expect(garden).not_to receive(:update!)
|
||||
expect(garden).not_to receive(:destroy)
|
||||
end
|
||||
|
||||
describe 'GET edit' do
|
||||
before { get :edit, id: garden.to_param }
|
||||
|
||||
it { expect(response).to redirect_to(new_member_session_path) }
|
||||
end
|
||||
|
||||
describe 'POST update' do
|
||||
before { post :update, id: garden.to_param, garden: valid_params }
|
||||
|
||||
it { expect(response).to redirect_to(new_member_session_path) }
|
||||
end
|
||||
|
||||
describe 'DELETE' do
|
||||
before { delete :destroy, id: garden.to_param, params: { garden: valid_params } }
|
||||
|
||||
it { expect(response).to redirect_to(new_member_session_path) }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context "when signed in" do
|
||||
before(:each) { sign_in member }
|
||||
|
||||
@@ -56,14 +68,19 @@ RSpec.describe GardensController, type: :controller do
|
||||
|
||||
describe 'GET edit' do
|
||||
before { get :edit, id: not_my_garden.to_param }
|
||||
|
||||
it { expect(response).to redirect_to(root_path) }
|
||||
end
|
||||
|
||||
describe 'POST update' do
|
||||
before { post :update, id: not_my_garden.to_param, garden: valid_params }
|
||||
|
||||
it { expect(response).to redirect_to(root_path) }
|
||||
end
|
||||
|
||||
describe 'DELETE' do
|
||||
before { delete :destroy, id: not_my_garden.to_param, params: { garden: valid_params } }
|
||||
|
||||
it { expect(response).to redirect_to(root_path) }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -22,23 +22,27 @@ describe HarvestsController do
|
||||
|
||||
describe "assigns all harvests as @harvests" do
|
||||
before { get :index, {} }
|
||||
|
||||
it { assigns(:harvests).should =~ [harvest1, harvest2] }
|
||||
end
|
||||
|
||||
describe "picks up owner from params and shows owner's harvests only" do
|
||||
before { get :index, owner: member1.slug }
|
||||
|
||||
it { expect(assigns(:owner)).to eq member1 }
|
||||
it { expect(assigns(:harvests)).to eq [harvest1] }
|
||||
end
|
||||
|
||||
describe "picks up crop from params and shows the harvests for the crop only" do
|
||||
before { get :index, crop: maize.name }
|
||||
|
||||
it { expect(assigns(:crop)).to eq maize }
|
||||
it { expect(assigns(:harvests)).to eq [harvest2] }
|
||||
end
|
||||
|
||||
describe "generates a csv" do
|
||||
before { get :index, format: "csv" }
|
||||
|
||||
it { expect(response.status).to eq 200 }
|
||||
end
|
||||
end
|
||||
@@ -48,6 +52,7 @@ describe HarvestsController do
|
||||
|
||||
describe "assigns the requested harvest as @harvest" do
|
||||
before { get :show, id: harvest.to_param }
|
||||
|
||||
it { expect(assigns(:harvest)).to eq(harvest) }
|
||||
end
|
||||
end
|
||||
@@ -69,6 +74,7 @@ describe HarvestsController do
|
||||
|
||||
describe "assigns the requested harvest as @harvest" do
|
||||
before { get :edit, id: harvest.to_param }
|
||||
|
||||
it { expect(assigns(:harvest)).to eq(harvest) }
|
||||
end
|
||||
end
|
||||
@@ -96,6 +102,7 @@ describe HarvestsController do
|
||||
let(:planting) { FactoryBot.create(:planting, owner_id: member.id, garden: member.gardens.first) }
|
||||
|
||||
before { post :create, harvest: valid_attributes.merge(planting_id: planting.id) }
|
||||
|
||||
it { expect(Harvest.last.planting.id).to eq(planting.id) }
|
||||
end
|
||||
end
|
||||
@@ -124,6 +131,7 @@ describe HarvestsController do
|
||||
allow(Harvest).to receive(:new).and_return(harvest)
|
||||
post :create, harvest: valid_attributes.merge(planting_id: not_my_planting.id)
|
||||
end
|
||||
|
||||
it { expect(harvest.planting_id).not_to eq(not_my_planting.id) }
|
||||
end
|
||||
end
|
||||
@@ -179,6 +187,7 @@ describe HarvestsController do
|
||||
put :update, id: harvest.to_param,
|
||||
harvest: valid_attributes.merge(planting_id: not_my_planting.id)
|
||||
end
|
||||
|
||||
it { expect(harvest.planting_id).to eq(nil) }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -11,6 +11,7 @@ describe LikesController do
|
||||
describe "POST create" do
|
||||
it { expect(response.content_type).to eq "application/json" }
|
||||
before { post :create, post_id: blogpost.id, format: :json }
|
||||
|
||||
it { expect(Like.last.likeable_id).to eq(blogpost.id) }
|
||||
it { expect(Like.last.likeable_type).to eq('Post') }
|
||||
it { JSON.parse(response.body)["description"] == "1 like" }
|
||||
@@ -26,6 +27,7 @@ describe LikesController do
|
||||
|
||||
describe "DELETE destroy" do
|
||||
before { delete :destroy, id: like.id, format: :json }
|
||||
|
||||
it { expect(response.content_type).to eq "application/json" }
|
||||
|
||||
describe "un-liking something i liked before" do
|
||||
|
||||
@@ -39,6 +39,7 @@ describe PhotosController do
|
||||
|
||||
describe "planting photos" do
|
||||
before(:each) { get :new, type: "planting", id: planting.id }
|
||||
|
||||
it { assigns(:flickr_auth).should be_an_instance_of(Authentication) }
|
||||
it { assigns(:item).should eq planting }
|
||||
it { expect(flash[:alert]).not_to be_present }
|
||||
@@ -47,12 +48,14 @@ describe PhotosController do
|
||||
|
||||
describe "harvest photos" do
|
||||
before { get :new, type: "harvest", id: harvest.id }
|
||||
|
||||
it { assigns(:item).should eq harvest }
|
||||
it { expect(flash[:alert]).not_to be_present }
|
||||
end
|
||||
|
||||
describe "garden photos" do
|
||||
before { get :new, type: "garden", id: garden.id }
|
||||
|
||||
it { assigns(:item).should eq garden }
|
||||
it { expect(flash[:alert]).not_to be_present }
|
||||
end
|
||||
@@ -76,6 +79,7 @@ describe PhotosController do
|
||||
|
||||
describe "with valid params" do
|
||||
before { controller.stub(:current_member) { member } }
|
||||
|
||||
it "attaches the photo to a planting" do
|
||||
post :create, photo: { flickr_photo_id: photo.flickr_photo_id }, type: "planting", id: planting.id
|
||||
expect(flash[:alert]).not_to be_present
|
||||
@@ -87,6 +91,7 @@ describe PhotosController do
|
||||
post :create, photo: { flickr_photo_id: photo.flickr_photo_id }, type: "planting", id: planting.id
|
||||
post :create, photo: { flickr_photo_id: photo.flickr_photo_id }, type: "planting", id: planting.id
|
||||
end
|
||||
|
||||
it { expect(flash[:alert]).not_to be_present }
|
||||
it { expect(Photo.last.plantings.size).to eq 1 }
|
||||
end
|
||||
@@ -127,6 +132,7 @@ describe PhotosController do
|
||||
|
||||
describe "with matching owners" do
|
||||
before { controller.stub(:current_member) { member } }
|
||||
|
||||
it "creates the planting/photo link" do
|
||||
planting = FactoryBot.create(:planting, garden: garden, owner: member)
|
||||
photo = FactoryBot.create(:photo, owner: member)
|
||||
|
||||
@@ -20,17 +20,20 @@ describe PlantingsController do
|
||||
|
||||
describe "assigns all plantings as @plantings" do
|
||||
before { get :index, {} }
|
||||
|
||||
it { expect(assigns(:plantings)).to match [planting1, planting2] }
|
||||
end
|
||||
|
||||
describe "picks up owner from params and shows owner's plantings only" do
|
||||
before { get :index, owner: member1.slug }
|
||||
|
||||
it { expect(assigns(:owner)).to eq member1 }
|
||||
it { expect(assigns(:plantings)).to eq [planting1] }
|
||||
end
|
||||
|
||||
describe "picks up crop from params and shows the plantings for the crop only" do
|
||||
before { get :index, crop: maize.name }
|
||||
|
||||
it { expect(assigns(:crop)).to eq maize }
|
||||
it { expect(assigns(:plantings)).to eq [planting2] }
|
||||
end
|
||||
@@ -41,11 +44,13 @@ describe PlantingsController do
|
||||
let(:crop) { FactoryBot.create(:crop) }
|
||||
|
||||
before { get :new, crop_id: crop.id }
|
||||
|
||||
it { expect(assigns(:crop)).to eq(crop) }
|
||||
end
|
||||
|
||||
describe "doesn't die if no crop specified" do
|
||||
before { get :new, {} }
|
||||
|
||||
it { expect(assigns(:crop)).to be_a_new(Crop) }
|
||||
end
|
||||
|
||||
@@ -53,6 +58,7 @@ describe PlantingsController do
|
||||
let(:garden) { FactoryBot.create(:garden, owner: member) }
|
||||
|
||||
before { get :new, garden_id: garden.id }
|
||||
|
||||
it { expect(assigns(:garden)).to eq(garden) }
|
||||
end
|
||||
|
||||
@@ -61,6 +67,7 @@ describe PlantingsController do
|
||||
let(:garden) { FactoryBot.create(:garden, owner: another_member) }
|
||||
|
||||
before { get :new, garden_id: garden.id }
|
||||
|
||||
it { expect(assigns(:garden)).not_to eq(garden) }
|
||||
end
|
||||
|
||||
@@ -69,6 +76,7 @@ describe PlantingsController do
|
||||
let!(:garden) { FactoryBot.create(:garden, owner: member) }
|
||||
|
||||
before { get :new, crop_id: crop.id }
|
||||
|
||||
it { expect(assigns(:crop)).not_to eq(crop) }
|
||||
end
|
||||
|
||||
@@ -77,22 +85,27 @@ describe PlantingsController do
|
||||
let!(:garden) { FactoryBot.create(:garden, owner: member) }
|
||||
|
||||
before { get :new, crop_id: crop.id }
|
||||
|
||||
it { expect(assigns(:crop)).not_to eq(crop) }
|
||||
end
|
||||
|
||||
describe "doesn't die if no garden specified" do
|
||||
before { get :new, {} }
|
||||
|
||||
it { expect(assigns(:garden)).to be_a_new(Garden) }
|
||||
end
|
||||
|
||||
describe "sets the date of the planting to today" do
|
||||
before { get :new, {} }
|
||||
|
||||
it { expect(assigns(:planting).planted_at).to eq Time.zone.today }
|
||||
end
|
||||
|
||||
context 'with parent seed' do
|
||||
let(:seed) { FactoryBot.create :seed, owner: member }
|
||||
|
||||
before { get :new, seed_id: seed.to_param }
|
||||
|
||||
it { expect(assigns(:seed)).to eq(seed) }
|
||||
end
|
||||
end
|
||||
@@ -100,6 +113,7 @@ describe PlantingsController do
|
||||
describe 'POST :create' do
|
||||
describe "sets the owner automatically" do
|
||||
before { post :create, planting: valid_attributes }
|
||||
|
||||
it { expect(assigns(:planting).owner).to eq subject.current_member }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -5,6 +5,7 @@ describe SeedsController do
|
||||
|
||||
describe "GET index" do
|
||||
before { get :index, owner: owner.slug }
|
||||
|
||||
it "picks up owner from params" do
|
||||
assigns(:owner).should eq(owner)
|
||||
end
|
||||
@@ -21,7 +22,9 @@ describe SeedsController do
|
||||
|
||||
context 'with parent planting' do
|
||||
let(:planting) { FactoryBot.create :planting, owner: owner }
|
||||
|
||||
before { get :new, planting_id: planting.to_param }
|
||||
|
||||
it { expect(assigns(:planting)).to eq(planting) }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -2,13 +2,13 @@
|
||||
|
||||
FactoryBot.define do
|
||||
factory :alternate_name do
|
||||
name "alternate name"
|
||||
name { "alternate name" }
|
||||
crop
|
||||
creator
|
||||
|
||||
factory :alternate_eggplant do
|
||||
association :crop, factory: :eggplant
|
||||
name "aubergine"
|
||||
name { "aubergine" }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -3,14 +3,14 @@
|
||||
FactoryBot.define do
|
||||
factory :authentication do
|
||||
member
|
||||
provider 'twitter'
|
||||
uid 'foo'
|
||||
secret 'bar'
|
||||
name 'baz'
|
||||
provider { 'twitter' }
|
||||
uid { 'foo' }
|
||||
secret { 'bar' }
|
||||
name { 'baz' }
|
||||
|
||||
factory :flickr_authentication do
|
||||
provider 'flickr'
|
||||
uid 'blah@blah'
|
||||
provider { 'flickr' }
|
||||
uid { 'blah@blah' }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -1,83 +1,83 @@
|
||||
FactoryBot.define do
|
||||
factory :crop do
|
||||
name "magic bean"
|
||||
en_wikipedia_url "http://en.wikipedia.org/wiki/Magic_bean"
|
||||
approval_status "approved"
|
||||
name { "magic bean" }
|
||||
en_wikipedia_url { "http://en.wikipedia.org/wiki/Magic_bean" }
|
||||
approval_status { "approved" }
|
||||
creator
|
||||
|
||||
factory :annual_crop, parent: :crop do
|
||||
perennial false
|
||||
perennial { false }
|
||||
end
|
||||
factory :perennial_crop, parent: :crop do
|
||||
perennial true
|
||||
perennial { true }
|
||||
end
|
||||
|
||||
factory :tomato do
|
||||
name "tomato"
|
||||
en_wikipedia_url "http://en.wikipedia.org/wiki/Tomato"
|
||||
name { "tomato" }
|
||||
en_wikipedia_url { "http://en.wikipedia.org/wiki/Tomato" }
|
||||
end
|
||||
|
||||
factory :maize do
|
||||
name "maize"
|
||||
en_wikipedia_url "http://en.wikipedia.org/wiki/Maize"
|
||||
name { "maize" }
|
||||
en_wikipedia_url { "http://en.wikipedia.org/wiki/Maize" }
|
||||
end
|
||||
|
||||
factory :chard do
|
||||
name "chard"
|
||||
name { "chard" }
|
||||
end
|
||||
|
||||
factory :walnut do
|
||||
name "walnut"
|
||||
name { "walnut" }
|
||||
end
|
||||
|
||||
factory :apple do
|
||||
name "apple"
|
||||
name { "apple" }
|
||||
end
|
||||
|
||||
factory :pear do
|
||||
name "pear"
|
||||
name { "pear" }
|
||||
end
|
||||
|
||||
# for testing varieties
|
||||
factory :roma do
|
||||
name "roma tomato"
|
||||
name { "roma tomato" }
|
||||
end
|
||||
|
||||
factory :popcorn do
|
||||
name "popcorn"
|
||||
name { "popcorn" }
|
||||
end
|
||||
|
||||
factory :eggplant do
|
||||
name "eggplant"
|
||||
name { "eggplant" }
|
||||
end
|
||||
|
||||
# This should have a name that is alphabetically earlier than :uppercase
|
||||
# crop to ensure that the ordering tests work.
|
||||
factory :lowercasecrop do
|
||||
name "ffrench bean"
|
||||
name { "ffrench bean" }
|
||||
end
|
||||
|
||||
factory :uppercasecrop do
|
||||
name "Swiss chard"
|
||||
name { "Swiss chard" }
|
||||
end
|
||||
|
||||
factory :autoloaded_crop do
|
||||
creator "cropbot"
|
||||
creator { "cropbot" }
|
||||
end
|
||||
|
||||
# for testing crop request
|
||||
factory :crop_request do
|
||||
name "Ultra berry"
|
||||
en_wikipedia_url ""
|
||||
approval_status "pending"
|
||||
name { "Ultra berry" }
|
||||
en_wikipedia_url { "" }
|
||||
approval_status { "pending" }
|
||||
association :requester, factory: :member
|
||||
request_notes "Please approve this even though it's fake."
|
||||
request_notes { "Please approve this even though it's fake." }
|
||||
end
|
||||
|
||||
factory :rejected_crop do
|
||||
name "Fail bean"
|
||||
approval_status "rejected"
|
||||
reason_for_rejection "Totally fake"
|
||||
name { "Fail bean" }
|
||||
approval_status { "rejected" }
|
||||
reason_for_rejection { "Totally fake" }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -2,8 +2,8 @@
|
||||
|
||||
FactoryBot.define do
|
||||
factory :forum do
|
||||
name "Permaculture"
|
||||
description "*Everything* about permaculture!"
|
||||
name { "Permaculture" }
|
||||
description { "*Everything* about permaculture!" }
|
||||
owner
|
||||
end
|
||||
end
|
||||
|
||||
@@ -1,24 +1,24 @@
|
||||
FactoryBot.define do
|
||||
factory :garden do
|
||||
name 'Springfield Community Garden'
|
||||
description "This is a **totally** cool garden"
|
||||
name { Faker::Vehicle.vin }
|
||||
description { "This is a **totally** cool garden" }
|
||||
owner
|
||||
active true
|
||||
area 23
|
||||
area_unit "acre"
|
||||
location "Greenwich, UK"
|
||||
active { true }
|
||||
area { 23 }
|
||||
area_unit { "acre" }
|
||||
location { "Greenwich, UK" }
|
||||
|
||||
factory :inactive_garden do
|
||||
active false
|
||||
active { false }
|
||||
end
|
||||
|
||||
# the following are used for testing alphabetical ordering
|
||||
factory :garden_a do
|
||||
name 'A garden starting with A'
|
||||
name { 'A garden starting with A' }
|
||||
end
|
||||
|
||||
factory :garden_z do
|
||||
name 'Zzzz this garden makes me sleepy'
|
||||
name { 'Zzzz this garden makes me sleepy' }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -3,15 +3,15 @@
|
||||
FactoryBot.define do
|
||||
factory :harvest do
|
||||
crop { planting.present? ? planting.crop : FactoryBot.create(:crop) }
|
||||
plant_part
|
||||
planting nil
|
||||
plant_part { FactoryBot.create :plant_part }
|
||||
planting { nil }
|
||||
owner { planting.present? ? planting.owner : FactoryBot.create(:member) }
|
||||
harvested_at { Time.zone.local(2015, 9, 17) }
|
||||
quantity "3"
|
||||
unit "individual"
|
||||
weight_quantity 6
|
||||
weight_unit "kg"
|
||||
description "A lovely harvest"
|
||||
quantity { "3" }
|
||||
unit { "individual" }
|
||||
weight_quantity { 6 }
|
||||
weight_unit { "kg" }
|
||||
description { "A lovely harvest" }
|
||||
|
||||
factory :harvest_with_planting do
|
||||
planting
|
||||
@@ -19,10 +19,10 @@ FactoryBot.define do
|
||||
end
|
||||
|
||||
trait :long_description do
|
||||
description "This is a very long description that is so very long that it will need to be cut off"
|
||||
description { "This is a very long description that is so very long that it will need to be cut off" }
|
||||
end
|
||||
|
||||
trait :no_description do
|
||||
description ""
|
||||
description { "" }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -4,59 +4,59 @@ FactoryBot.define do
|
||||
|
||||
factory :member, aliases: %i(author owner sender recipient creator) do
|
||||
login_name { generate(:login_name) }
|
||||
password 'password1'
|
||||
password { 'password1' }
|
||||
email { generate(:email) }
|
||||
tos_agreement true
|
||||
tos_agreement { true }
|
||||
confirmed_at { Time.now }
|
||||
show_email false
|
||||
bio 'I love seeds'
|
||||
show_email { false }
|
||||
bio { 'I love seeds' }
|
||||
|
||||
# cropbot is needed for certain tests, eg. Crop.create_from_csv
|
||||
factory :cropbot do
|
||||
login_name 'cropbot'
|
||||
login_name { 'cropbot' }
|
||||
end
|
||||
|
||||
factory :no_tos_member do
|
||||
tos_agreement false
|
||||
tos_agreement { false }
|
||||
end
|
||||
|
||||
factory :newsletter_recipient_member do
|
||||
newsletter true
|
||||
newsletter { true }
|
||||
end
|
||||
|
||||
factory :no_bio_member do
|
||||
bio nil
|
||||
bio { nil }
|
||||
end
|
||||
|
||||
factory :unconfirmed_member do
|
||||
confirmed_at nil
|
||||
confirmed_at { nil }
|
||||
end
|
||||
|
||||
# this member has very loose privacy settings
|
||||
factory :public_member do
|
||||
login_name 'NothingToHide'
|
||||
show_email true
|
||||
login_name { 'NothingToHide' }
|
||||
show_email { true }
|
||||
end
|
||||
|
||||
factory :london_member do
|
||||
sequence(:login_name) { |n| "JohnH#{n}" } # for the astronomer who figured out longitude
|
||||
location 'Greenwich, UK'
|
||||
location { 'Greenwich, UK' }
|
||||
# including lat/long explicitly because geocoder doesn't work with FG
|
||||
latitude 51.483
|
||||
longitude 0.004
|
||||
latitude { 51.483 }
|
||||
longitude { 0.004 }
|
||||
end
|
||||
|
||||
factory :edinburgh_member do
|
||||
location 'Edinburgh'
|
||||
latitude 55.953252
|
||||
location { 'Edinburgh' }
|
||||
latitude { 55.953252 }
|
||||
longitude { -3.188267 }
|
||||
end
|
||||
|
||||
factory :south_pole_member do
|
||||
sequence(:login_name) { |n| "ScottRF#{n}" }
|
||||
location 'Amundsen-Scott Base, Antarctica'
|
||||
location { 'Amundsen-Scott Base, Antarctica' }
|
||||
latitude { -90 }
|
||||
longitude 0
|
||||
longitude { 0 }
|
||||
end
|
||||
|
||||
factory :admin_member do
|
||||
@@ -69,39 +69,39 @@ FactoryBot.define do
|
||||
end
|
||||
|
||||
factory :invalid_member_shortname do
|
||||
login_name 'a'
|
||||
login_name { 'a' }
|
||||
end
|
||||
|
||||
factory :invalid_member_longname do
|
||||
login_name 'MarmadukeBlundellHollinsheadBlundellTolemachePlantagenetWhistlebinkie3rdDukeofMarmoset'
|
||||
login_name { 'MarmadukeBlundellHollinsheadBlundellTolemachePlantagenetWhistlebinkie3rdDukeofMarmoset' }
|
||||
end
|
||||
|
||||
factory :invalid_member_spaces do
|
||||
login_name "a b"
|
||||
login_name { "a b" }
|
||||
end
|
||||
|
||||
factory :invalid_member_badchars do
|
||||
login_name 'aa%$'
|
||||
login_name { 'aa%$' }
|
||||
end
|
||||
|
||||
factory :invalid_member_badname do
|
||||
login_name 'admin'
|
||||
login_name { 'admin' }
|
||||
end
|
||||
|
||||
factory :valid_member_alphanumeric do
|
||||
login_name 'abc123'
|
||||
login_name { 'abc123' }
|
||||
end
|
||||
|
||||
factory :valid_member_uppercase do
|
||||
login_name 'ABC123'
|
||||
login_name { 'ABC123' }
|
||||
end
|
||||
|
||||
factory :valid_member_underscore do
|
||||
login_name 'abc_123'
|
||||
login_name { 'abc_123' }
|
||||
end
|
||||
|
||||
factory :no_email_notifications_member do
|
||||
send_notification_email false
|
||||
send_notification_email { false }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -4,9 +4,10 @@ FactoryBot.define do
|
||||
factory :notification, aliases: [:message] do
|
||||
sender
|
||||
recipient
|
||||
subject "MyString"
|
||||
body "MyText"
|
||||
read false
|
||||
subject { "MyString" }
|
||||
|
||||
body { "MyText" }
|
||||
read { false }
|
||||
post
|
||||
|
||||
factory :no_email_notification do
|
||||
|
||||
@@ -3,17 +3,17 @@
|
||||
FactoryBot.define do
|
||||
factory :photo do
|
||||
owner
|
||||
flickr_photo_id 1
|
||||
flickr_photo_id { 1 }
|
||||
title { Faker::HarryPotter.quote }
|
||||
license_name "CC-BY"
|
||||
license_url "http://example.com/license.html"
|
||||
license_name { "CC-BY" }
|
||||
license_url { "http://example.com/license.html" }
|
||||
thumbnail_url { "http://example.com/#{Faker::File.file_name}.jpg" }
|
||||
fullsize_url { "http://example.com/#{Faker::File.file_name}.jpg" }
|
||||
link_url { Faker::Internet.url }
|
||||
|
||||
factory :unlicensed_photo do
|
||||
license_name "All rights reserved"
|
||||
license_url ""
|
||||
license_name { "All rights reserved" }
|
||||
license_url { "" }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -2,6 +2,6 @@
|
||||
|
||||
FactoryBot.define do
|
||||
factory :plant_part do
|
||||
name "pollen"
|
||||
name { "#{Faker::Book.title}_#{rand(100..999)}" }
|
||||
end
|
||||
end
|
||||
|
||||
@@ -4,35 +4,37 @@ FactoryBot.define do
|
||||
garden { FactoryBot.create :garden, owner: owner }
|
||||
crop
|
||||
planted_at { Time.zone.local(2014, 7, 30) }
|
||||
quantity 33
|
||||
description "This is a *really* good plant."
|
||||
quantity { 33 }
|
||||
description { "This is a *really* good plant." }
|
||||
finished { false }
|
||||
finished_at { nil }
|
||||
|
||||
factory :seed_planting do
|
||||
planted_from 'seed'
|
||||
planted_from { 'seed' }
|
||||
end
|
||||
|
||||
factory :seedling_planting do
|
||||
planted_from 'seedling'
|
||||
planted_from { 'seedling' }
|
||||
end
|
||||
|
||||
factory :cutting_planting do
|
||||
planted_from 'cutting'
|
||||
planted_from { 'cutting' }
|
||||
end
|
||||
|
||||
factory :sunny_planting do
|
||||
sunniness 'sun'
|
||||
sunniness { 'sun' }
|
||||
end
|
||||
|
||||
factory :semi_shady_planting do
|
||||
sunniness 'semi-shade'
|
||||
sunniness { 'semi-shade' }
|
||||
end
|
||||
|
||||
factory :shady_planting do
|
||||
sunniness 'shade'
|
||||
sunniness { 'shade' }
|
||||
end
|
||||
|
||||
factory :finished_planting do
|
||||
finished true
|
||||
finished { true }
|
||||
planted_at { Time.zone.local(2014, 7, 30) }
|
||||
finished_at { Time.zone.local(2014, 8, 30) }
|
||||
end
|
||||
|
||||
@@ -1,18 +1,19 @@
|
||||
FactoryBot.define do
|
||||
factory :post do
|
||||
subject "A Post"
|
||||
body "This is some text."
|
||||
subject { "A Post" }
|
||||
|
||||
body { "This is some text." }
|
||||
author
|
||||
created_at { Time.now }
|
||||
|
||||
# Markdown is allowed in posts
|
||||
factory :markdown_post do
|
||||
body "This is some **strong** text."
|
||||
body { "This is some **strong** text." }
|
||||
end
|
||||
|
||||
# HTML isn't allowed in posts
|
||||
factory :html_post do
|
||||
body '<a href="http://evil.com">EVIL</a>'
|
||||
body { '<a href="http://evil.com">EVIL</a>' }
|
||||
end
|
||||
|
||||
factory :forum_post do
|
||||
|
||||
@@ -2,17 +2,17 @@
|
||||
|
||||
FactoryBot.define do
|
||||
factory :role do
|
||||
name "Moderator"
|
||||
description "These people moderate the forums"
|
||||
name { "Moderator" }
|
||||
description { "These people moderate the forums" }
|
||||
initialize_with { Role.find_or_create_by(name: name) }
|
||||
|
||||
factory :admin do
|
||||
name "admin"
|
||||
name { "admin" }
|
||||
end
|
||||
|
||||
factory :crop_wrangler do
|
||||
name "Crop Wrangler"
|
||||
description "they wrangle crops"
|
||||
name { "Crop Wrangler" }
|
||||
description { "they wrangle crops" }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -1,17 +1,17 @@
|
||||
FactoryBot.define do
|
||||
factory :scientific_name do
|
||||
association :crop, factory: :crop
|
||||
name "Beanus Magicus"
|
||||
name { "Beanus Magicus" }
|
||||
creator
|
||||
|
||||
factory :zea_mays do
|
||||
association :crop, factory: :maize
|
||||
name "Zea mays"
|
||||
name { "Zea mays" }
|
||||
end
|
||||
|
||||
factory :solanum_lycopersicum do
|
||||
association :crop, factory: :tomato
|
||||
name "Solanum lycopersicum"
|
||||
name { "Solanum lycopersicum" }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -4,30 +4,30 @@ FactoryBot.define do
|
||||
factory :seed do
|
||||
owner
|
||||
crop
|
||||
description "MyText"
|
||||
quantity 1
|
||||
plant_before "2013-07-15"
|
||||
tradable_to 'nowhere'
|
||||
organic 'unknown'
|
||||
gmo 'unknown'
|
||||
heirloom 'unknown'
|
||||
days_until_maturity_min nil
|
||||
days_until_maturity_max nil
|
||||
finished_at nil
|
||||
description { "MyText" }
|
||||
quantity { 1 }
|
||||
plant_before { "2013-07-15" }
|
||||
tradable_to { 'nowhere' }
|
||||
organic { 'unknown' }
|
||||
gmo { 'unknown' }
|
||||
heirloom { 'unknown' }
|
||||
days_until_maturity_min { nil }
|
||||
days_until_maturity_max { nil }
|
||||
finished_at { nil }
|
||||
|
||||
factory :finished_seed do
|
||||
finished true
|
||||
finished { true }
|
||||
finished_at { Date.new }
|
||||
end
|
||||
|
||||
factory :tradable_seed do
|
||||
tradable_to "locally"
|
||||
finished false
|
||||
finished_at nil
|
||||
tradable_to { "locally" }
|
||||
finished { false }
|
||||
finished_at { nil }
|
||||
end
|
||||
|
||||
factory :untradable_seed do
|
||||
tradable_to "nowhere"
|
||||
tradable_to { "nowhere" }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user