Skip to content

Commit

Permalink
feat: use atomic writes to avoid data corruption in case of crashes (m…
Browse files Browse the repository at this point in the history
…siemens#468)

(cherry picked from commit 2a2abe5)
  • Loading branch information
ostafen authored and richardsheridan committed Jul 27, 2022
1 parent 6307a9f commit 10deb8e
Showing 1 changed file with 15 additions and 8 deletions.
23 changes: 15 additions & 8 deletions tinydb/storages.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
import io
import json
import os
import tempfile
from abc import ABC, abstractmethod
from typing import Dict, Any, Optional

Expand Down Expand Up @@ -125,25 +126,31 @@ def read(self) -> Optional[Dict[str, Dict[str, Any]]]:
return json.load(self._handle)

def write(self, data: Dict[str, Dict[str, Any]]):
# Move the cursor to the beginning of the file just in case
self._handle.seek(0)
file_name = self._handle.name

# Create a temporary file in the same folder
temp_file = tempfile.NamedTemporaryFile(mode=self._mode, prefix=file_name, delete=False)

# Serialize the database state using the user-provided arguments
serialized = json.dumps(data, **self.kwargs)

# Write the serialized data to the file
try:
self._handle.write(serialized)
temp_file.write(serialized)
except io.UnsupportedOperation:
raise IOError('Cannot write to the database. Access mode is "{0}"'.format(self._mode))

# Ensure the file has been written
self._handle.flush()
os.fsync(self._handle.fileno())
temp_file.flush()
os.fsync(temp_file.fileno())

# Replace the current file with the temporary file
temp_file.close()
os.rename(temp_file.name, file_name)

# Remove data that is behind the new cursor in case the file has
# gotten shorter
self._handle.truncate()
# Reopen the file
self._handle.close()
self._handle = open(file_name, mode=self._mode, encoding=self._handle.encoding)


class MemoryStorage(Storage):
Expand Down

0 comments on commit 10deb8e

Please sign in to comment.