-
Notifications
You must be signed in to change notification settings - Fork 92
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Optimization task 4 #149
base: master
Are you sure you want to change the base?
Optimization task 4 #149
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,85 @@ | ||
# Задание 4 | ||
|
||
## Актуальная проблема | ||
В проекте dev.to выявлена проблема производительности главной страницы: | ||
- Медленный рендеринг главной страницы (StoriesController#index) (Особенно затратный рендеринг partial-ов _single_story.html.erb) | ||
- Отсутствие кэширования страниц | ||
|
||
## Формирование метрик | ||
Для оценки эффективности оптимизации определены следующие метрики: | ||
|
||
- Время полной загрузки главной страницы | ||
- Время рендеринга partial _single_story.html.erb | ||
- Количество запросов к БД при рендеринге страницы | ||
- Использование CPU и памяти | ||
- Включил кеширование на локальном окружении | ||
- Использование `benchmark` с помощью `ab` (`ab -n 100 -c 5 http://localhost:3000/`) | ||
- Добавил local_production окружение | ||
|
||
## Feedback-Loop | ||
Построен быстрый цикл обратной связи: | ||
|
||
- NewRelic APM для мониторинга метрик | ||
- rack-mini-profiler для профилирования рендеринга | ||
- Поиск точек роста | ||
|
||
## Использованы инструменты профилирования: | ||
|
||
- NewRelic для анализа узких мест | ||
- rack-mini-profiler для детального профилирования рендеринга | ||
- Логи Rails для анализа SQL-запросов | ||
|
||
## Результаты оптимизации: | ||
|
||
### 1. Отсутствует local_production окружение | ||
#### Произвел замеры с использованием `ab` до добавления local_production: | ||
``` | ||
Concurrency Level: 5 | ||
Time taken for tests: 61.726 seconds | ||
Complete requests: 100 | ||
Failed requests: 99 | ||
(Connect: 0, Receive: 0, Length: 99, Exceptions: 0) | ||
Total transferred: 16131253 bytes | ||
HTML transferred: 16030175 bytes | ||
Requests per second: 1.62 [#/sec] (mean) | ||
Time per request: 3086.288 [ms] (mean) | ||
Time per request: 617.258 [ms] (mean, across all concurrent requests) | ||
Transfer rate: 255.21 [Kbytes/sec] received | ||
``` | ||
|
||
#### После добавления: | ||
``` | ||
Concurrency Level: 5 | ||
Time taken for tests: 14.988 seconds | ||
Complete requests: 100 | ||
Failed requests: 0 | ||
Total transferred: 13161800 bytes | ||
HTML transferred: 13115100 bytes | ||
Requests per second: 6.67 [#/sec] (mean) | ||
Time per request: 749.387 [ms] (mean) | ||
Time per request: 149.877 [ms] (mean, across all concurrent requests) | ||
Transfer rate: 857.59 [Kbytes/sec] received | ||
``` | ||
- Время обработки всех запросов сократилось в 4 раза с 60 до 15 секунд. Также количество Failed сократилось с 99 до 0. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 99 failed - там response size чуть не сходится; видимо есть какие-то отличия от исходного ответа, который берётся за эталон; а при local_production всё стабильно в этом плане |
||
|
||
### 2. Отсутствие кеширования partial-ов _single_story.html.erb | ||
- Readme задания и rack-mini-profiler | ||
- Добавил кеширование partial'а, учел, что в него входят счётчики лайков и комментариев. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 👍 |
||
|
||
#### Замеры после добавления кеширования: | ||
``` | ||
Concurrency Level: 5 | ||
Time taken for tests: 7.179 seconds | ||
Complete requests: 100 | ||
Failed requests: 0 | ||
Total transferred: 13028000 bytes | ||
HTML transferred: 12981300 bytes | ||
Requests per second: 13.93 [#/sec] (mean) | ||
Time per request: 358.953 [ms] (mean) | ||
Time per request: 71.791 [ms] (mean, across all concurrent requests) | ||
Transfer rate: 1772.19 [Kbytes/sec] received | ||
``` | ||
- Время обработки всех запросов сократилось в два раза с 14 до 7 секунд. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. 👍 |
||
- Время выполнения отдельного запроса составило 360ms | ||
|
||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,109 @@ | ||
# rubocop:disable Metrics/BlockLength | ||
# | ||
def yarn_integrity_enabled? | ||
ENV.fetch("YARN_INTEGRITY_ENABLED", "true") == "true" | ||
end | ||
|
||
Rails.application.configure do | ||
# Verifies that versions and hashed value of the package contents in the project's package.json | ||
config.webpacker.check_yarn_integrity = yarn_integrity_enabled? | ||
|
||
# Settings specified here will take precedence over those in config/application.rb. | ||
|
||
# In the development environment your application's code is reloaded on | ||
# every request. This slows down response time but is perfect for development | ||
# since you don't have to restart the web server when you make code changes. | ||
config.cache_classes = true | ||
|
||
# Do not eager load code on boot. | ||
config.eager_load = true | ||
|
||
# Show full error reports and disable caching. | ||
config.consider_all_requests_local = true | ||
|
||
# Enable/disable caching. By default caching is disabled. | ||
if Rails.root.join("tmp/caching-dev.txt").exist? | ||
config.action_controller.perform_caching = true | ||
|
||
config.cache_store = :memory_store | ||
config.public_file_server.headers = { | ||
"Cache-Control" => "public, max-age=172800" | ||
} | ||
else | ||
config.action_controller.perform_caching = false | ||
|
||
config.cache_store = :null_store | ||
end | ||
|
||
config.assets_debug = false | ||
config.assets_compile = false | ||
|
||
config.web_console.development_only = false | ||
|
||
# Don't care if the mailer can't send. | ||
config.action_mailer.raise_delivery_errors = false | ||
|
||
# Print deprecation notices to the Rails logger. | ||
config.active_support.deprecation = :log | ||
|
||
# Raise an error on page load if there are pending migrations. | ||
config.active_record.migration_error = :page_load | ||
|
||
# Debug mode disables concatenation and preprocessing of assets. | ||
# This option may cause significant delays in view rendering with a large | ||
# number of complex assets. | ||
config.assets.debug = false | ||
|
||
# Asset digests allow you to set far-future HTTP expiration dates on all assets, | ||
# yet still be able to expire them through the digest params. | ||
config.assets.digest = false | ||
|
||
# Supress logger output for asset requests. | ||
config.assets.quiet = true | ||
|
||
# Adds additional error checking when serving assets at runtime. | ||
# Checks for improperly declared sprockets dependencies. | ||
# Raises helpful error messages. | ||
config.assets.raise_runtime_errors = true | ||
|
||
config.action_mailer.perform_caching = false | ||
|
||
config.app_domain = "localhost:3000" | ||
|
||
config.action_mailer.default_url_options = { host: "localhost:3000" } | ||
config.action_mailer.delivery_method = :smtp | ||
config.action_mailer.perform_deliveries = true | ||
config.action_mailer.default_url_options = { host: config.app_domain } | ||
config.action_mailer.smtp_settings = { | ||
address: "smtp.gmail.com", | ||
port: "587", | ||
enable_starttls_auto: true, | ||
user_name: '<%= ENV["DEVELOPMENT_EMAIL_USERNAME"] %>', | ||
password: '<%= ENV["DEVELOPMENT_EMAIL_PASSWORD"] %>', | ||
authentication: :plain, | ||
domain: "localhost:3000" | ||
} | ||
|
||
config.action_mailer.preview_path = "#{Rails.root}/spec/mailers/previews" | ||
|
||
# Raises error for missing translations | ||
# config.action_view.raise_on_missing_translations = true | ||
|
||
config.public_file_server.enabled = true | ||
|
||
config.file_watcher = ActiveSupport::EventedFileUpdateChecker | ||
|
||
# Install the Timber.io logger | ||
send_logs_to_timber = ENV["SEND_LOGS_TO_TIMBER"] || "false" # <---- set to false to stop sending dev logs to Timber.io | ||
log_device = send_logs_to_timber == "true" ? Timber::LogDevices::HTTP.new(ENV["TIMBER"]) : STDOUT | ||
logger = Timber::Logger.new(log_device) | ||
logger.level = config.log_level | ||
config.logger = ActiveSupport::TaggedLogging.new(logger) | ||
|
||
config.after_initialize do | ||
Bullet.enable = true | ||
Bullet.console = true | ||
end | ||
end | ||
|
||
# rubocop:enable Metrics/BlockLength |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
👍