about summary refs log tree commit diff stats
path: root/config
diff options
context:
space:
mode:
authorStar Rauchenberger <fefferburbia@gmail.com>2023-11-30 13:29:08 -0500
committerStar Rauchenberger <fefferburbia@gmail.com>2023-11-30 13:29:08 -0500
commit0929719a845897cc8567cf972e07a69a71f0fa6f (patch)
tree2b6f69c1d906abb6e0abf8a0f1d51725bc78087d /config
parent01c1947537e4e23ded0c16812a7cd9d49ad88356 (diff)
downloadwittle-0929719a845897cc8567cf972e07a69a71f0fa6f.tar.gz
wittle-0929719a845897cc8567cf972e07a69a71f0fa6f.tar.bz2
wittle-0929719a845897cc8567cf972e07a69a71f0fa6f.zip
Migrate to a full rails app
Diffstat (limited to 'config')
-rw-r--r--config/application.rb27
-rw-r--r--config/boot.rb4
-rw-r--r--config/cable.yml10
-rw-r--r--config/credentials.yml.enc1
-rw-r--r--config/database.yml25
-rw-r--r--config/environment.rb5
-rw-r--r--config/environments/development.rb76
-rw-r--r--config/environments/production.rb98
-rw-r--r--config/environments/test.rb64
-rw-r--r--config/initializers/assets.rb12
-rw-r--r--config/initializers/content_security_policy.rb25
-rw-r--r--config/initializers/filter_parameter_logging.rb8
-rw-r--r--config/initializers/inflections.rb16
-rw-r--r--config/initializers/permissions_policy.rb13
-rw-r--r--config/locales/en.yml31
-rw-r--r--config/puma.rb35
-rw-r--r--config/routes.rb2
-rw-r--r--config/storage.yml34
18 files changed, 485 insertions, 1 deletions
diff --git a/config/application.rb b/config/application.rb new file mode 100644 index 0000000..095566e --- /dev/null +++ b/config/application.rb
@@ -0,0 +1,27 @@
1require_relative "boot"
2
3require "rails/all"
4
5# Require the gems listed in Gemfile, including any gems
6# you've limited to :test, :development, or :production.
7Bundler.require(*Rails.groups)
8
9module Wittle
10 class Application < Rails::Application
11 # Initialize configuration defaults for originally generated Rails version.
12 config.load_defaults 7.1
13
14 # Please, add to the `ignore` list any other `lib` subdirectories that do
15 # not contain `.rb` files, or that should not be reloaded or eager loaded.
16 # Common ones are `templates`, `generators`, or `middleware`, for example.
17 config.autoload_lib(ignore: %w(assets tasks))
18
19 # Configuration for the application, engines, and railties goes here.
20 #
21 # These settings can be overridden in specific environments using the files
22 # in config/environments, which are processed later.
23 #
24 # config.time_zone = "Central Time (US & Canada)"
25 # config.eager_load_paths << Rails.root.join("extras")
26 end
27end
diff --git a/config/boot.rb b/config/boot.rb new file mode 100644 index 0000000..988a5dd --- /dev/null +++ b/config/boot.rb
@@ -0,0 +1,4 @@
1ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../Gemfile", __dir__)
2
3require "bundler/setup" # Set up gems listed in the Gemfile.
4require "bootsnap/setup" # Speed up boot time by caching expensive operations.
diff --git a/config/cable.yml b/config/cable.yml new file mode 100644 index 0000000..46d6557 --- /dev/null +++ b/config/cable.yml
@@ -0,0 +1,10 @@
1development:
2 adapter: async
3
4test:
5 adapter: test
6
7production:
8 adapter: redis
9 url: <%= ENV.fetch("REDIS_URL") { "redis://localhost:6379/1" } %>
10 channel_prefix: wittle_production
diff --git a/config/credentials.yml.enc b/config/credentials.yml.enc new file mode 100644 index 0000000..494d15d --- /dev/null +++ b/config/credentials.yml.enc
@@ -0,0 +1 @@
80HLuAOBpfETFq58hHZxQQI9DcvkqZzwx5TDwXZ5JPvjIzrj4lOuhFCnd/QfqHsKOyearjd1Ot0+VzmN8pi3dr3sgRaVrqE37F6fBlUW5XxDE02GWmKQk8MTIkVJyLjI+AfiGAEn+W9LoKK1drqsJFnQDYVRIXpKjUTyYYiA8i3M+CqSNJCb41tOsXgVbfORUAEJGncJVGCkHey6alRGWoJioByUe5jPIVnSFNUAz7kOajbB2l7laqxvTVRFXzOOryh44pXeNiJRVmLAlJGfdfTZcB1cD+OjEd0QOfyIrchyGHcGexmV4qrirN+5gVqeJ75D1M9SWVPms/w2RFSlq+miEjnUIYq6LyrLEwQxYSG8ontWMuVpdfdgJrdb6aaPYyVzFxk968BUpjWU9asXsE1sVqK7--q8zHjMUiCIKcGKVK--s/ilfRd7l7AUvTXruie3LA== \ No newline at end of file
diff --git a/config/database.yml b/config/database.yml new file mode 100644 index 0000000..796466b --- /dev/null +++ b/config/database.yml
@@ -0,0 +1,25 @@
1# SQLite. Versions 3.8.0 and up are supported.
2# gem install sqlite3
3#
4# Ensure the SQLite 3 gem is defined in your Gemfile
5# gem "sqlite3"
6#
7default: &default
8 adapter: sqlite3
9 pool: <%= ENV.fetch("RAILS_MAX_THREADS") { 5 } %>
10 timeout: 5000
11
12development:
13 <<: *default
14 database: storage/development.sqlite3
15
16# Warning: The database defined as "test" will be erased and
17# re-generated from your development database when you run "rake".
18# Do not set this db to the same as development or production.
19test:
20 <<: *default
21 database: storage/test.sqlite3
22
23production:
24 <<: *default
25 database: storage/production.sqlite3
diff --git a/config/environment.rb b/config/environment.rb new file mode 100644 index 0000000..cac5315 --- /dev/null +++ b/config/environment.rb
@@ -0,0 +1,5 @@
1# Load the Rails application.
2require_relative "application"
3
4# Initialize the Rails application.
5Rails.application.initialize!
diff --git a/config/environments/development.rb b/config/environments/development.rb new file mode 100644 index 0000000..2e7fb48 --- /dev/null +++ b/config/environments/development.rb
@@ -0,0 +1,76 @@
1require "active_support/core_ext/integer/time"
2
3Rails.application.configure do
4 # Settings specified here will take precedence over those in config/application.rb.
5
6 # In the development environment your application's code is reloaded any time
7 # it changes. This slows down response time but is perfect for development
8 # since you don't have to restart the web server when you make code changes.
9 config.enable_reloading = true
10
11 # Do not eager load code on boot.
12 config.eager_load = false
13
14 # Show full error reports.
15 config.consider_all_requests_local = true
16
17 # Enable server timing
18 config.server_timing = true
19
20 # Enable/disable caching. By default caching is disabled.
21 # Run rails dev:cache to toggle caching.
22 if Rails.root.join("tmp/caching-dev.txt").exist?
23 config.action_controller.perform_caching = true
24 config.action_controller.enable_fragment_cache_logging = true
25
26 config.cache_store = :memory_store
27 config.public_file_server.headers = {
28 "Cache-Control" => "public, max-age=#{2.days.to_i}"
29 }
30 else
31 config.action_controller.perform_caching = false
32
33 config.cache_store = :null_store
34 end
35
36 # Store uploaded files on the local file system (see config/storage.yml for options).
37 config.active_storage.service = :local
38
39 # Don't care if the mailer can't send.
40 config.action_mailer.raise_delivery_errors = false
41
42 config.action_mailer.perform_caching = false
43
44 # Print deprecation notices to the Rails logger.
45 config.active_support.deprecation = :log
46
47 # Raise exceptions for disallowed deprecations.
48 config.active_support.disallowed_deprecation = :raise
49
50 # Tell Active Support which deprecation messages to disallow.
51 config.active_support.disallowed_deprecation_warnings = []
52
53 # Raise an error on page load if there are pending migrations.
54 config.active_record.migration_error = :page_load
55
56 # Highlight code that triggered database queries in logs.
57 config.active_record.verbose_query_logs = true
58
59 # Highlight code that enqueued background job in logs.
60 config.active_job.verbose_enqueue_logs = true
61
62 # Suppress logger output for asset requests.
63 config.assets.quiet = true
64
65 # Raises error for missing translations.
66 # config.i18n.raise_on_missing_translations = true
67
68 # Annotate rendered view with file names.
69 # config.action_view.annotate_rendered_view_with_filenames = true
70
71 # Uncomment if you wish to allow Action Cable access from any origin.
72 # config.action_cable.disable_request_forgery_protection = true
73
74 # Raise error when a before_action's only/except options reference missing actions
75 config.action_controller.raise_on_missing_callback_actions = true
76end
diff --git a/config/environments/production.rb b/config/environments/production.rb new file mode 100644 index 0000000..77a44b1 --- /dev/null +++ b/config/environments/production.rb
@@ -0,0 +1,98 @@
1require "active_support/core_ext/integer/time"
2
3Rails.application.configure do
4 # Settings specified here will take precedence over those in config/application.rb.
5
6 # Code is not reloaded between requests.
7 config.enable_reloading = false
8
9 # Eager load code on boot. This eager loads most of Rails and
10 # your application in memory, allowing both threaded web servers
11 # and those relying on copy on write to perform better.
12 # Rake tasks automatically ignore this option for performance.
13 config.eager_load = true
14
15 # Full error reports are disabled and caching is turned on.
16 config.consider_all_requests_local = false
17 config.action_controller.perform_caching = true
18
19 # Ensures that a master key has been made available in ENV["RAILS_MASTER_KEY"], config/master.key, or an environment
20 # key such as config/credentials/production.key. This key is used to decrypt credentials (and other encrypted files).
21 # config.require_master_key = true
22
23 # Disable serving static files from `public/`, relying on NGINX/Apache to do so instead.
24 # config.public_file_server.enabled = false
25
26 # Compress CSS using a preprocessor.
27 config.assets.js_compressor = :terser
28 # config.assets.css_compressor = :sass
29
30 # Do not fallback to assets pipeline if a precompiled asset is missed.
31 config.assets.compile = false
32
33 # Enable serving of images, stylesheets, and JavaScripts from an asset server.
34 # config.asset_host = "http://assets.example.com"
35
36 # Specifies the header that your server uses for sending files.
37 # config.action_dispatch.x_sendfile_header = "X-Sendfile" # for Apache
38 # config.action_dispatch.x_sendfile_header = "X-Accel-Redirect" # for NGINX
39
40 # Store uploaded files on the local file system (see config/storage.yml for options).
41 config.active_storage.service = :local
42
43 # Mount Action Cable outside main process or domain.
44 # config.action_cable.mount_path = nil
45 # config.action_cable.url = "wss://example.com/cable"
46 # config.action_cable.allowed_request_origins = [ "http://example.com", /http:\/\/example.*/ ]
47
48 # Assume all access to the app is happening through a SSL-terminating reverse proxy.
49 # Can be used together with config.force_ssl for Strict-Transport-Security and secure cookies.
50 # config.assume_ssl = true
51
52 # Force all access to the app over SSL, use Strict-Transport-Security, and use secure cookies.
53 config.force_ssl = true
54
55 # Log to STDOUT by default
56 config.logger = ActiveSupport::Logger.new(STDOUT)
57 .tap { |logger| logger.formatter = ::Logger::Formatter.new }
58 .then { |logger| ActiveSupport::TaggedLogging.new(logger) }
59
60 # Prepend all log lines with the following tags.
61 config.log_tags = [ :request_id ]
62
63 # Info include generic and useful information about system operation, but avoids logging too much
64 # information to avoid inadvertent exposure of personally identifiable information (PII). If you
65 # want to log everything, set the level to "debug".
66 config.log_level = ENV.fetch("RAILS_LOG_LEVEL", "info")
67
68 # Use a different cache store in production.
69 # config.cache_store = :mem_cache_store
70
71 # Use a real queuing backend for Active Job (and separate queues per environment).
72 # config.active_job.queue_adapter = :resque
73 # config.active_job.queue_name_prefix = "wittle_production"
74
75 config.action_mailer.perform_caching = false
76
77 # Ignore bad email addresses and do not raise email delivery errors.
78 # Set this to true and configure the email server for immediate delivery to raise delivery errors.
79 # config.action_mailer.raise_delivery_errors = false
80
81 # Enable locale fallbacks for I18n (makes lookups for any locale fall back to
82 # the I18n.default_locale when a translation cannot be found).
83 config.i18n.fallbacks = true
84
85 # Don't log any deprecations.
86 config.active_support.report_deprecations = false
87
88 # Do not dump schema after migrations.
89 config.active_record.dump_schema_after_migration = false
90
91 # Enable DNS rebinding protection and other `Host` header attacks.
92 # config.hosts = [
93 # "example.com", # Allow requests from example.com
94 # /.*\.example\.com/ # Allow requests from subdomains like `www.example.com`
95 # ]
96 # Skip DNS rebinding protection for the default health check endpoint.
97 # config.host_authorization = { exclude: ->(request) { request.path == "/up" } }
98end
diff --git a/config/environments/test.rb b/config/environments/test.rb new file mode 100644 index 0000000..adbb4a6 --- /dev/null +++ b/config/environments/test.rb
@@ -0,0 +1,64 @@
1require "active_support/core_ext/integer/time"
2
3# The test environment is used exclusively to run your application's
4# test suite. You never need to work with it otherwise. Remember that
5# your test database is "scratch space" for the test suite and is wiped
6# and recreated between test runs. Don't rely on the data there!
7
8Rails.application.configure do
9 # Settings specified here will take precedence over those in config/application.rb.
10
11 # While tests run files are not watched, reloading is not necessary.
12 config.enable_reloading = false
13
14 # Eager loading loads your entire application. When running a single test locally,
15 # this is usually not necessary, and can slow down your test suite. However, it's
16 # recommended that you enable it in continuous integration systems to ensure eager
17 # loading is working properly before deploying your code.
18 config.eager_load = ENV["CI"].present?
19
20 # Configure public file server for tests with Cache-Control for performance.
21 config.public_file_server.enabled = true
22 config.public_file_server.headers = {
23 "Cache-Control" => "public, max-age=#{1.hour.to_i}"
24 }
25
26 # Show full error reports and disable caching.
27 config.consider_all_requests_local = true
28 config.action_controller.perform_caching = false
29 config.cache_store = :null_store
30
31 # Render exception templates for rescuable exceptions and raise for other exceptions.
32 config.action_dispatch.show_exceptions = :rescuable
33
34 # Disable request forgery protection in test environment.
35 config.action_controller.allow_forgery_protection = false
36
37 # Store uploaded files on the local file system in a temporary directory.
38 config.active_storage.service = :test
39
40 config.action_mailer.perform_caching = false
41
42 # Tell Action Mailer not to deliver emails to the real world.
43 # The :test delivery method accumulates sent emails in the
44 # ActionMailer::Base.deliveries array.
45 config.action_mailer.delivery_method = :test
46
47 # Print deprecation notices to the stderr.
48 config.active_support.deprecation = :stderr
49
50 # Raise exceptions for disallowed deprecations.
51 config.active_support.disallowed_deprecation = :raise
52
53 # Tell Active Support which deprecation messages to disallow.
54 config.active_support.disallowed_deprecation_warnings = []
55
56 # Raises error for missing translations.
57 # config.i18n.raise_on_missing_translations = true
58
59 # Annotate rendered view with file names.
60 # config.action_view.annotate_rendered_view_with_filenames = true
61
62 # Raise error when a before_action's only/except options reference missing actions
63 config.action_controller.raise_on_missing_callback_actions = true
64end
diff --git a/config/initializers/assets.rb b/config/initializers/assets.rb new file mode 100644 index 0000000..2eeef96 --- /dev/null +++ b/config/initializers/assets.rb
@@ -0,0 +1,12 @@
1# Be sure to restart your server when you modify this file.
2
3# Version of your assets, change this if you want to expire all your assets.
4Rails.application.config.assets.version = "1.0"
5
6# Add additional assets to the asset load path.
7# Rails.application.config.assets.paths << Emoji.images_path
8
9# Precompile additional assets.
10# application.js, application.css, and all non-JS/CSS in the app/assets
11# folder are already added.
12# Rails.application.config.assets.precompile += %w( admin.js admin.css )
diff --git a/config/initializers/content_security_policy.rb b/config/initializers/content_security_policy.rb new file mode 100644 index 0000000..b3076b3 --- /dev/null +++ b/config/initializers/content_security_policy.rb
@@ -0,0 +1,25 @@
1# Be sure to restart your server when you modify this file.
2
3# Define an application-wide content security policy.
4# See the Securing Rails Applications Guide for more information:
5# https://guides.rubyonrails.org/security.html#content-security-policy-header
6
7# Rails.application.configure do
8# config.content_security_policy do |policy|
9# policy.default_src :self, :https
10# policy.font_src :self, :https, :data
11# policy.img_src :self, :https, :data
12# policy.object_src :none
13# policy.script_src :self, :https
14# policy.style_src :self, :https
15# # Specify URI for violation reports
16# # policy.report_uri "/csp-violation-report-endpoint"
17# end
18#
19# # Generate session nonces for permitted importmap, inline scripts, and inline styles.
20# config.content_security_policy_nonce_generator = ->(request) { request.session.id.to_s }
21# config.content_security_policy_nonce_directives = %w(script-src style-src)
22#
23# # Report violations without enforcing the policy.
24# # config.content_security_policy_report_only = true
25# end
diff --git a/config/initializers/filter_parameter_logging.rb b/config/initializers/filter_parameter_logging.rb new file mode 100644 index 0000000..c2d89e2 --- /dev/null +++ b/config/initializers/filter_parameter_logging.rb
@@ -0,0 +1,8 @@
1# Be sure to restart your server when you modify this file.
2
3# Configure parameters to be partially matched (e.g. passw matches password) and filtered from the log file.
4# Use this to limit dissemination of sensitive information.
5# See the ActiveSupport::ParameterFilter documentation for supported notations and behaviors.
6Rails.application.config.filter_parameters += [
7 :passw, :secret, :token, :_key, :crypt, :salt, :certificate, :otp, :ssn
8]
diff --git a/config/initializers/inflections.rb b/config/initializers/inflections.rb new file mode 100644 index 0000000..3860f65 --- /dev/null +++ b/config/initializers/inflections.rb
@@ -0,0 +1,16 @@
1# Be sure to restart your server when you modify this file.
2
3# Add new inflection rules using the following format. Inflections
4# are locale specific, and you may define rules for as many different
5# locales as you wish. All of these examples are active by default:
6# ActiveSupport::Inflector.inflections(:en) do |inflect|
7# inflect.plural /^(ox)$/i, "\\1en"
8# inflect.singular /^(ox)en/i, "\\1"
9# inflect.irregular "person", "people"
10# inflect.uncountable %w( fish sheep )
11# end
12
13# These inflection rules are supported but not enabled by default:
14# ActiveSupport::Inflector.inflections(:en) do |inflect|
15# inflect.acronym "RESTful"
16# end
diff --git a/config/initializers/permissions_policy.rb b/config/initializers/permissions_policy.rb new file mode 100644 index 0000000..7db3b95 --- /dev/null +++ b/config/initializers/permissions_policy.rb
@@ -0,0 +1,13 @@
1# Be sure to restart your server when you modify this file.
2
3# Define an application-wide HTTP permissions policy. For further
4# information see: https://developers.google.com/web/updates/2018/06/feature-policy
5
6# Rails.application.config.permissions_policy do |policy|
7# policy.camera :none
8# policy.gyroscope :none
9# policy.microphone :none
10# policy.usb :none
11# policy.fullscreen :self
12# policy.payment :self, "https://secure.example.com"
13# end
diff --git a/config/locales/en.yml b/config/locales/en.yml new file mode 100644 index 0000000..6c349ae --- /dev/null +++ b/config/locales/en.yml
@@ -0,0 +1,31 @@
1# Files in the config/locales directory are used for internationalization and
2# are automatically loaded by Rails. If you want to use locales other than
3# English, add the necessary files in this directory.
4#
5# To use the locales, use `I18n.t`:
6#
7# I18n.t "hello"
8#
9# In views, this is aliased to just `t`:
10#
11# <%= t("hello") %>
12#
13# To use a different locale, set it with `I18n.locale`:
14#
15# I18n.locale = :es
16#
17# This would use the information in config/locales/es.yml.
18#
19# To learn more about the API, please read the Rails Internationalization guide
20# at https://guides.rubyonrails.org/i18n.html.
21#
22# Be aware that YAML interprets the following case-insensitive strings as
23# booleans: `true`, `false`, `on`, `off`, `yes`, `no`. Therefore, these strings
24# must be quoted to be interpreted as strings. For example:
25#
26# en:
27# "yes": yup
28# enabled: "ON"
29
30en:
31 hello: "Hello world"
diff --git a/config/puma.rb b/config/puma.rb new file mode 100644 index 0000000..afa809b --- /dev/null +++ b/config/puma.rb
@@ -0,0 +1,35 @@
1# This configuration file will be evaluated by Puma. The top-level methods that
2# are invoked here are part of Puma's configuration DSL. For more information
3# about methods provided by the DSL, see https://puma.io/puma/Puma/DSL.html.
4
5# Puma can serve each request in a thread from an internal thread pool.
6# The `threads` method setting takes two numbers: a minimum and maximum.
7# Any libraries that use thread pools should be configured to match
8# the maximum value specified for Puma. Default is set to 5 threads for minimum
9# and maximum; this matches the default thread size of Active Record.
10max_threads_count = ENV.fetch("RAILS_MAX_THREADS") { 5 }
11min_threads_count = ENV.fetch("RAILS_MIN_THREADS") { max_threads_count }
12threads min_threads_count, max_threads_count
13
14# Specifies that the worker count should equal the number of processors in production.
15if ENV["RAILS_ENV"] == "production"
16 require "concurrent-ruby"
17 worker_count = Integer(ENV.fetch("WEB_CONCURRENCY") { Concurrent.physical_processor_count })
18 workers worker_count if worker_count > 1
19end
20
21# Specifies the `worker_timeout` threshold that Puma will use to wait before
22# terminating a worker in development environments.
23worker_timeout 3600 if ENV.fetch("RAILS_ENV", "development") == "development"
24
25# Specifies the `port` that Puma will listen on to receive requests; default is 3000.
26port ENV.fetch("PORT") { 3000 }
27
28# Specifies the `environment` that Puma will run in.
29environment ENV.fetch("RAILS_ENV") { "development" }
30
31# Specifies the `pidfile` that Puma will use.
32pidfile ENV.fetch("PIDFILE") { "tmp/pids/server.pid" }
33
34# Allow puma to be restarted by `bin/rails restart` command.
35plugin :tmp_restart
diff --git a/config/routes.rb b/config/routes.rb index 346315e..3a93d2a 100644 --- a/config/routes.rb +++ b/config/routes.rb
@@ -1,4 +1,4 @@
1Wittle::Engine.routes.draw do 1Rails.application.routes.draw do
2 root to: 'puzzles#about' 2 root to: 'puzzles#about'
3 get 'archive' => 'puzzles#index' 3 get 'archive' => 'puzzles#index'
4 get ':id' => 'puzzles#show', as: 'puzzle' 4 get ':id' => 'puzzles#show', as: 'puzzle'
diff --git a/config/storage.yml b/config/storage.yml new file mode 100644 index 0000000..4942ab6 --- /dev/null +++ b/config/storage.yml
@@ -0,0 +1,34 @@
1test:
2 service: Disk
3 root: <%= Rails.root.join("tmp/storage") %>
4
5local:
6 service: Disk
7 root: <%= Rails.root.join("storage") %>
8
9# Use bin/rails credentials:edit to set the AWS secrets (as aws:access_key_id|secret_access_key)
10# amazon:
11# service: S3
12# access_key_id: <%= Rails.application.credentials.dig(:aws, :access_key_id) %>
13# secret_access_key: <%= Rails.application.credentials.dig(:aws, :secret_access_key) %>
14# region: us-east-1
15# bucket: your_own_bucket-<%= Rails.env %>
16
17# Remember not to checkin your GCS keyfile to a repository
18# google:
19# service: GCS
20# project: your_project
21# credentials: <%= Rails.root.join("path/to/gcs.keyfile") %>
22# bucket: your_own_bucket-<%= Rails.env %>
23
24# Use bin/rails credentials:edit to set the Azure Storage secret (as azure_storage:storage_access_key)
25# microsoft:
26# service: AzureStorage
27# storage_account_name: your_account_name
28# storage_access_key: <%= Rails.application.credentials.dig(:azure_storage, :storage_access_key) %>
29# container: your_container_name-<%= Rails.env %>
30
31# mirror:
32# service: Mirror
33# primary: local
34# mirrors: [ amazon, google, microsoft ]