Я думаю, что этот эффект происходит от того, как написано 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 и вижу ошибку, в этот момент все файлы, включая файл с ошибкой, не выгружаются
- Я делаю кодовое исправление и нажимаю тот же вызов API, чтобы проверить, сработало ли оно
- вызов направляется так же, как и раньше, к старым классам / объектам / модулям. Это выдает ту же ошибку и снова оставляет загруженные константы в памяти
Этого не происходит, когда традиционные контроллеры генерируют ошибки, потому что они обрабатываются ActionController::Rescue
. Такие исключения не бьют ActionController::Reloader
.
Самое простое решение - поместить аварийное спасательное предложение в промежуточное ПО маршрутизации API, некоторые варианты этого:
def call(env)
# route API call
resuce Exception
Dispatcher.cleanup_application
raise
end
Обратите внимание, что это мой ответ на трехлетний вопрос, и я следовал за стеком вызовов 2.3.3. Более новые версии рельсов могут обрабатывать вещи по-разному.