Ответ 1
Я думаю, что этот эффект исходит из того, как написано ActionController::Reloader
. Здесь ActionController::Reloader#call
из 2.3.3, обратите внимание на комментарий:
def call(env)
Dispatcher.reload_application
status, headers, body = @app.call(env)
# We do not want to call 'cleanup_application' in an ensure block
# because the returned Rack response body may lazily generate its data. This
# is for example the case if one calls
#
# render :text => lambda { ... code here which refers to application models ... }
#
# in an ActionController.
#
# Instead, we will want to cleanup the application code after the request is
# completely finished. So we wrap the body in a BodyWrapper class so that
# when the Rack handler calls #close during the end of the request, we get to
# run our cleanup code.
[status, headers, BodyWrapper.new(body)]
end
Dispatcher.reload_application
не удаляет автоматически загруженные константы, Dispatcher.cleanup_application
делает. BodyWrapper#close
записывается с учетом возможных исключений:
def close
@body.close if @body.respond_to?(:close)
ensure
Dispatcher.cleanup_application
end
Однако это не помогает, потому что если @app.call
в ActionController::Reloader#call
выдает исключение, BodyWrapper
не получает экземпляр, а Dispatcher.cleanup_application
не вызывается.
Представьте себе следующий сценарий:
- Я вношу изменения в один из моих файлов, который влияет на вызов API
- Я нахожу вызов API и вижу ошибку, в этот момент все файлы, в том числе и с ошибкой, не выгружаются
- Я делаю кодfix и нажимаю тот же API-код, чтобы проверить, работает ли он
- вызов маршрутизируется так же, как и раньше, в старые классы/объекты/модули. Это вызывает такую же ошибку и снова оставляет загруженные константы в памяти
Этого не происходит, когда традиционные контроллеры вызывают ошибки, потому что они обрабатываются ActionController::Rescue
. Такие исключения не попадают ActionController::Reloader
.
Простейшим решением было бы поставить резервное предложение спасения в промежуточное ПО маршрутизации API, некоторые изменения этого:
def call(env)
# route API call
resuce Exception
Dispatcher.cleanup_application
raise
end
Обратите внимание, что это мой ответ на 3-летний вопрос, и я выполнил стек вызовов 2.3.3. Более новые версии рельсов могут обрабатывать разные вещи.