Merge pull request #54376 from dwoz/tmp_token

Move tokens in place with an atomic operation
This commit is contained in:
Daniel Wozniak 2019-09-03 13:40:49 -07:00 committed by GitHub
commit 11016cedf3
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 57 additions and 1 deletions

View file

@ -34,6 +34,7 @@ def mk_token(opts, tdata):
hash_type = getattr(hashlib, opts.get('hash_type', 'md5'))
tok = six.text_type(hash_type(os.urandom(512)).hexdigest())
t_path = os.path.join(opts['token_dir'], tok)
temp_t_path = '{}.tmp'.format(t_path)
while os.path.isfile(t_path):
tok = six.text_type(hash_type(os.urandom(512)).hexdigest())
t_path = os.path.join(opts['token_dir'], tok)
@ -41,8 +42,9 @@ def mk_token(opts, tdata):
serial = salt.payload.Serial(opts)
try:
with salt.utils.files.set_umask(0o177):
with salt.utils.files.fopen(t_path, 'w+b') as fp_:
with salt.utils.files.fopen(temp_t_path, 'w+b') as fp_:
fp_.write(serial.dumps(tdata))
os.rename(temp_t_path, t_path)
except (IOError, OSError):
log.warning(
'Authentication failure: can not write token file "%s".', t_path)

View file

@ -0,0 +1 @@
# -*- coding: utf-8 -*-

View file

@ -0,0 +1,53 @@
# -*- coding: utf-8 -*-
from __future__ import absolute_import, print_function, unicode_literals
import os
import salt.utils.files
import salt.tokens.localfs
from tests.support.unit import TestCase, skipIf
from tests.support.helpers import with_tempdir
from tests.support.mock import NO_MOCK, NO_MOCK_REASON, patch
class CalledWith(object):
def __init__(self, func, called_with=None):
self.func = func
if called_with is None:
self.called_with = []
else:
self.called_with = called_with
def __call__(self, *args, **kwargs):
self.called_with.append((args, kwargs))
return self.func(*args, **kwargs)
@skipIf(NO_MOCK, NO_MOCK_REASON)
class WriteTokenTest(TestCase):
@with_tempdir()
def test_write_token(self, tmpdir):
'''
Validate tokens put in place with an atomic move
'''
opts = {
'token_dir': tmpdir
}
fopen = CalledWith(salt.utils.files.fopen)
rename = CalledWith(os.rename)
with patch('salt.utils.files.fopen', fopen), patch('os.rename', rename):
tdata = salt.tokens.localfs.mk_token(opts, {})
assert 'token' in tdata
t_path = os.path.join(tmpdir, tdata['token'])
temp_t_path = '{}.tmp'.format(t_path)
assert len(fopen.called_with) == 1, len(fopen.called_with)
assert fopen.called_with == [
((temp_t_path, 'w+b'), {})
], fopen.called_with
assert len(rename.called_with) == 1, len(rename.called_with)
assert rename.called_with == [
((temp_t_path, t_path), {})
], rename.called_with