I think the race condition can be removed by doing something like this
(It's similar to what mkstemp does):

while True:
    try:
        fd = os.open(os.path.join(settings.MEDIA_ROOT, filename),
os.O_RDWR | os.O_CREAT | os.O_EXCL)
        fp = os.fdopen(fd, 'wb')
        fp.write(raw_contents)
        fp.close()
        break;
    except OSError, e:
        if e.errno != errno.EEXIST:
            raise
    try:
        dot_index = filename.rindex('.')
    except ValueError: # filename has no dot
        filename += '_'
    else:
        filename = filename[:dot_index] + '_' + filename[dot_index:]

Does this look reasonable?

Regards
Enrico

On Tue, 2007-04-24 at 22:45 +1000, Malcolm Tredinnick wrote:
> On Tue, 2007-04-24 at 21:06 +1200, Enrico de Klerk wrote:
> > Hi there,
> > 
> > I've noticed a possible race condition in django/db/models/base.py.
> > 
> > When a file field is saved in the _save_FIELD_file function it first
> > checks whether a file with the same name already exists, and then adds
> > underscores to the filename until it generates a unique name and then
> > saves the file.
> > 
> > # If the filename already exists, keep adding an underscore to the name
> > of
> >         # the file until the filename doesn't exist.
> >         while os.path.exists(os.path.join(settings.MEDIA_ROOT,
> > filename)):
> >             try:
> >                 dot_index = filename.rindex('.')
> >             except ValueError: # filename has no dot
> >                 filename += '_'
> >             else:
> >                 filename = filename[:dot_index] + '_' +
> > filename[dot_index:]
> > 
> >         # Write the file to disk.
> >         setattr(self, field.attname, filename)
> > 
> >         full_filename = self._get_FIELD_filename(field)
> >         fp = open(full_filename, 'wb')
> >         fp.write(raw_contents)
> >         fp.close()
> > 
> > It looks like there may be a timing window where if two files with the
> > same name are saved at roughly the same time, the first one will be
> > overwritten. Shouldn't this use something like mkstemp that does this
> > atomically?
> 
> The problem is that mkstemp doesn't give you a filename that looks
> anything like the original. Even if you control the prefix and suffix,
> there's still a large random block in the filename. Your analysis is
> correct, though -- it is vulnerable to a race. Worth thinking about if
> there's something we can do that's even less likely to collide.
> 
> Regards,
> Malcolm
> 
> 
> 
> > 


--~--~---------~--~----~------------~-------~--~----~
You received this message because you are subscribed to the Google Groups 
"Django users" group.
To post to this group, send email to django-users@googlegroups.com
To unsubscribe from this group, send email to [EMAIL PROTECTED]
For more options, visit this group at 
http://groups.google.com/group/django-users?hl=en
-~----------~----~----~----~------~----~------~--~---

Reply via email to