"WARNING Conda.gateways.disk:exp_backoff_fn(47): Uncaught Backoff With Errno 41" During "conda Install"
Solution 1:
These warnings that you see are "normal" according to the current conda source tree. To understand the origin of the aforementioned warnings, let's have a look at the source code in questions and a recent commit in the conda repository (https://github.com/conda/conda). The relevant source code that prints the warnings you see is the following:
https://github.com/conda/conda/blob/4.3.4/conda/gateways/disk/init.py
# -*- coding: utf-8 -*-
from __future__ import absolute_import, division, print_function, unicode_literals
import sys
from errno import EACCES, ENOENT, EPERM, EPROTOTYPE
from logging import getLogger
from os.path import basename
from time import sleep
from ...common.compat import on_win
log = getLogger(__name__)
MAX_TRIES = 7
def exp_backoff_fn(fn, *args, **kwargs):
"""Mostly for retrying file operations that fail on Windows due to virus scanners"""
max_tries = kwargs.pop('max_tries', MAX_TRIES)
if not on_win:
return fn(*args, **kwargs)
import random
# with max_tries = 6, max total time ~= 3.2 sec
# with max_tries = 7, max total time ~= 6.5 sec
for n in range(max_tries):
try:
result = fn(*args, **kwargs)
except (OSError, IOError) as e:
log.trace(repr(e))
if e.errno in (EPERM, EACCES):
if n == max_tries-1:
raise
sleep_time = ((2 ** n) + random.random()) * 0.1
caller_frame = sys._getframe(1)
log.trace("retrying %s/%s %s() in %g sec",
basename(caller_frame.f_code.co_filename),
caller_frame.f_lineno,
fn.__name__,
sleep_time)
sleep(sleep_time)
elif e.errno in (ENOENT, EPROTOTYPE):
# errno.ENOENT File not found error / No such file or directory
# errno.EPROTOTYPE OSError(41, 'The directory is not empty')
raise
else:
log.warn("Uncaught backoff with errno %d", e.errno)
raise
else:
return result
From the above source code it appears to be a warning thrown on Windows that might appear when
retrying file operations that fail on Windows due to virus scanners
Going into the specifics using https://msdn.microsoft.com/en-us/library/t3ayayh1.aspx, it appears that errno 41 corresponds to
ENOTEMPTY: Directory not empty
which signals that the directory specified is not empty.
It is an uncaught error because they don't have a branch that deals with this kind of error (ENOTEMPTY), while for example they have in the case it's another error like EPERM or EACCES.
In commit https://github.com/conda/conda/commit/fb2a783d9b9371559b5ea82aaf8ae631c2ce0450#diff-3757ed9862260ae3b54768b3e482e3fe
they explicitly remove reporting EPROTOTYPE
as OSError(41, 'The directory is not empty')
, so now you see that error number reported as a warning in the
log.warn("Uncaught backoff with errno %d", e.errno)
The other part they modify is in https://github.com/conda/conda/blob/fb2a783d9b9371559b5ea82aaf8ae631c2ce0450/conda/gateways/disk/delete.py, in the delete_trash()
function, so now, if you would enable the info log, most probably you would see a line like
"Unable to fully clean trash directory %s\nThere are %d remaining file(s)."
enabled by
log.info("Unable to fully clean trash directory %s\nThere are %d remaining file(s).",
trash_dir, len(files_remaining))
Now, delete_trash()
is called by both your commands that you cite (install, update):
https://github.com/conda/conda/blob/f4b386476307e3979679957292d4f6e4c581df03/conda/cli/main_install.py
https://github.com/conda/conda/blob/a26b1eff17dcaf12f03aea5bbe8dee1e01308de7/conda/cli/main_update.py
As can be seen, delete_trash()
is triggered respectively in 'install' and 'update' files previously mentioned respectively by the following code fragments:
from ..gateways.disk.delete import delete_trash
# some other code ...
def execute(args, parser):
install(args, parser, 'install')
delete_trash()
and
from ..gateways.disk.delete import delete_trash
# some other code ...
def execute(args, parser):
install(args, parser, 'update')
delete_trash()
delete_trash()
will then trigger that code path via backoff_rmdir()
or backoff_unlink()
that ultimately will lead to the warning you see from exp_backoff_fn()
as seen before.
So, to sum up, the main chain of calls would be
update or install --> delete_trash() --> backoff_rmdir() or backoff_unlink() --> exp_backoff_fn() --> your warning message
According to the source code modifications done in the repository, the developers believe than that these are warning that you can safely ignore because these warnings happen in the "cleaning" phase of the update or installation commands, i.e. after update or install operation have been successfully performed.
I couldn't say 100% wether you could safely ignore these warnings. If after few attempts then the command to delete the trash directory succeeds then there's no problem. But if it doesn't succeed then you will hit the problem that this directory will grow bigger and bigger as the result of not deleting it. There were few issue opened in the repo for that and I don't know if the fixes cover the code path that you hit. My impression is that might not. To get further insights you could activate the info log level.
UPDATE: This issue https://github.com/conda/conda/issues/4164 exactly mention the warnings you reported, because people were getting long update and install time due to all the retries. As I mentioned that after all the retries (exponential back-offs) the delete operation could either succeed or fail, that person also mention this aspect in his report.
As you can see in here
https://github.com/conda/conda/issues/3664
there are few hacks that people employ to solve the problem of long waiting times because of the retries and will also make the next run of your command conda install X
or conda update X
without warning. These suggestions are:
- (to speedup time for retries/delete) set
MAX_TRIES = 1
in your copy of conda/gateways/disk/init.py - remove the .trash directory before you run the next
conda install X
orconda update X
. See https://github.com/conda/conda/issues/3664 for a workaround employed by some people to automatize deletion of that directory using simple scripts. This should be usually safe to do it.
So the answers to your questions would be:
1) You can use the workaround mentioned in https://github.com/conda/conda/issues/3664, which is using the following powershell script (and another script):
$cir = conda info --root
$trash_dir = "$($cir)\pkgs\.trash"
if (Test-Path $trash_dir){
Remove-Item -Recurse -Force $trash_dir
}
conda --debug update --all --yes --quiet
to basically clean up that .trash directory;
2) You can safely ignore the warnings in the sense that they won't affect the functionality; the problem is that the more .trash gets populated, the more time and retries will go on to delete the items so you will hit performance problems; as you mentioned it's kind of having a "leak", but it doesn't affect the functionality. That directory is supposed to be emptied and removed, as it contains trash no longer needed. The system will try to delete it but might not be able to do it. So use 1).
UPDATE 2: As mentioned in one of my comments, one of the key changes is in the file conda/gateways/disk/__init__.py
, which was a "fix" (https://github.com/conda/conda/commit/6cb3be39aec1c738678ae27b3a264941d08c859a) that made it to 4.3.6 version of conda (conda 4.3.6 release info) which solves the warning in question.
The key point to not seeing that warning is by having a branch that explicitly catches and deals with the kind of error explained before. Now, when en error of type ENOTEMPTY will happen (which is the error that was triggering the printing of the warning in this case), this will be caught and will not go to the branch that prints the warning examined by the question.
To understand the main differences, in version 4.3.4 it was
elif e.errno in (ENOENT, EPROTOTYPE):
raise
else:
log.warn("Uncaught backoff with errno %d", e.errno)
raise
while in version 4.3.6 it became:
elif e.errno in (ENOENT, ENOTEMPTY):
raise
else:
log.warn("Uncaught backoff with errno %s %d", errorcode[e.errno], e.errno)
raise
and you clearly see that now for that error will not go into the else
branch so you won't see that message in this case.
Solution 2:
These warnings have been present on my computer with conda version 4.3.4 and 4.3.5 but not in the 4.2.x version and not in the latest release (4.3.6) anymore.
I guess the best way to "fix" the problem would be to update conda:
$ conda update conda
or to downgrade to 4.2:
$ conda install conda=4.2
With both versions the warnings disappear.
Post a Comment for ""WARNING Conda.gateways.disk:exp_backoff_fn(47): Uncaught Backoff With Errno 41" During "conda Install""