123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368 |
- using System;
- using System.IO;
- using Wayne.Lib.IO.UnitTest;
- namespace Wayne.Lib.IO
- {
-
-
-
-
-
-
- public class SafeFileWritingStream : Stream
- {
- #region Fields
- private readonly IFileSupport fileSupport;
- private readonly SafeFileWritingInterruptPoint interruptAtStage = SafeFileWritingInterruptPoint.DontInterrupt;
- private readonly SafeFileInfo safeFileName;
- private readonly Stream targetFile;
- private readonly Stream tempOutputFile;
- private bool closed;
- #endregion
- #region Construction
-
-
-
-
-
- [Obsolete("Use constructor taking IFilesupport parameter.")]
- public SafeFileWritingStream(string fileName)
- : this(FileSupport.fileSupport, fileName, SafeFileWritingInterruptPoint.DontInterrupt)
- {
- }
-
-
-
-
-
-
- public SafeFileWritingStream(IFileSupport fileSupport, string fileName)
- : this(fileSupport, fileName, SafeFileWritingInterruptPoint.DontInterrupt)
- {
- }
-
-
-
-
-
-
- private SafeFileWritingStream(IFileSupport fileSupport, string fileName, SafeFileWritingInterruptPoint interruptAtStage)
- {
- fileName = Paths.Parse(fileName);
- this.fileSupport = fileSupport;
- this.interruptAtStage = interruptAtStage;
- SetWritingStage(SafeFileWritingInterruptPoint.WritingNotBegun);
- safeFileName = SafeFileInfo.FromOriginalFileName(fileName);
- try
- {
- if (fileSupport.FileExists(fileName))
- targetFile = fileSupport.Open(fileName, FileMode.Open, FileAccess.Write, FileShare.Read);
- }
- catch (IOException ioException)
- {
- throw new SafeFileWritingIOException("Error opening target file (" + fileName + ")", ioException);
- }
-
- if (fileSupport.FileExists(safeFileName.OldFileName))
- {
- try
- {
- fileSupport.Delete(safeFileName.OldFileName);
- }
- catch (IOException ioException)
- {
- throw new SafeFileWritingIOException("Error deleting a -old file (" + safeFileName.OldFileName + ")", ioException);
- }
- }
- try
- {
- tempOutputFile = fileSupport.Open(safeFileName.TempFileName, FileMode.Create, FileAccess.Write, FileShare.None);
- }
- catch (IOException ioException)
- {
- throw new SafeFileWritingIOException("Error opening the temp file (" + safeFileName.TempFileName + ")", ioException);
- }
- SetWritingStage(SafeFileWritingInterruptPoint.WritingOngoing);
- }
- #endregion
- #region Properties
-
-
-
- public override bool CanRead
- {
- get { return false; }
- }
-
-
-
- public override bool CanSeek
- {
- get { return false; }
- }
-
-
-
- public override bool CanWrite
- {
- get { return true; }
- }
-
-
-
-
-
- public override void Flush()
- {
- tempOutputFile.Flush();
- }
-
-
-
-
-
- public override long Length
- {
- get { return tempOutputFile.Length; }
- }
-
-
-
-
- public override long Position
- {
- get
- {
- return tempOutputFile.Position;
- }
- set
- {
- throw new NotSupportedException();
- }
- }
- #endregion
- #region Methods
-
-
-
-
-
-
-
-
- public override int Read(byte[] buffer, int offset, int count)
- {
- throw new NotSupportedException();
- }
-
-
-
-
-
-
-
- public override long Seek(long offset, SeekOrigin origin)
- {
- throw new NotSupportedException();
- }
-
-
-
-
- public override void SetLength(long value)
- {
- throw new NotSupportedException();
- }
-
-
-
-
-
-
-
-
-
-
-
- public override void Write(byte[] buffer, int offset, int count)
- {
- tempOutputFile.Write(buffer, offset, count);
- }
-
-
-
- public override void Close()
- {
- base.Close();
- if (!closed)
- {
- closed = true;
-
- tempOutputFile.Close();
-
- if (targetFile != null)
- {
- targetFile.Close();
-
- fileSupport.Move(safeFileName.OriginalFileName, safeFileName.OldFileName, 100, 100);
- SetWritingStage(SafeFileWritingInterruptPoint.WritingCompleteOriginalFileRenamedToOld);
- }
-
- fileSupport.Move(safeFileName.TempFileName, safeFileName.OriginalFileName, 100, 100);
- SetWritingStage(SafeFileWritingInterruptPoint.WritingCompleteTempFileRenamedToTargetFile);
- if (targetFile != null)
- {
-
- fileSupport.Delete(safeFileName.OldFileName, 100, 100);
- SetWritingStage(SafeFileWritingInterruptPoint.WritingComplete);
- }
- }
- }
-
-
-
-
-
- private void SetWritingStage(SafeFileWritingInterruptPoint writingStage)
- {
- if (writingStage == interruptAtStage)
- {
- targetFile.Close();
- tempOutputFile.Close();
- throw new SafeFileWritingInterruptedException();
- }
- }
- #endregion
- #region Static Methods
-
-
-
-
-
-
-
-
-
-
-
- [Obsolete("Use Cleanup method supplying file support.")]
- public static void Cleanup(string folderPath, string pattern, EventHandler<SafeFileWritingCleanupEventArgs> temporaryFileFoundCallback, object userToken)
- {
- Cleanup(FileSupport.fileSupport, folderPath, pattern, temporaryFileFoundCallback, userToken);
- }
-
-
-
-
-
-
-
-
-
-
-
-
- public static void Cleanup(IFileSupport fileSupport, string folderPath, string pattern, EventHandler<SafeFileWritingCleanupEventArgs> temporaryFileFoundCallback, object userToken)
- {
- if (fileSupport.DirectoryExists(folderPath))
- {
- foreach (SafeFileInfo safeFile in SafeFileInfo.GetFiles(fileSupport, folderPath, pattern))
- {
-
- if (safeFile.OriginalFileExists && !safeFile.TempFileExists && !safeFile.OldFileExist)
- {
-
- }
-
- else if ((safeFile.OriginalFileExists && safeFile.TempFileExists) ||
- (!safeFile.OriginalFileExists && !safeFile.OldFileExist && safeFile.TempFileExists))
- {
-
- fileSupport.Delete(safeFile.TempFileName);
- SafeFileWritingCleanupEventArgs evArgs = new SafeFileWritingCleanupEventArgs(safeFile.TempFileName, userToken);
- if (temporaryFileFoundCallback != null)
- temporaryFileFoundCallback(null, evArgs);
- if (evArgs.Action == SafeFileWritingCleanupAction.StoreInRestoredTempFileDir)
- {
- string restoredTempFilesDir = Paths.Combine("RestoredTemp");
- fileSupport.EnsureDirectoryExists(restoredTempFilesDir);
- string newFileName = string.Concat(Guid.NewGuid(), Path.GetFileName(safeFile.TempFileName));
- fileSupport.Move(safeFile.TempFileName, Path.Combine(restoredTempFilesDir, newFileName), 100, 100);
- }
- else
- {
- fileSupport.Delete(safeFile.TempFileName);
- }
- }
-
- else if (safeFile.TempFileExists && safeFile.OldFileExist)
- {
-
- fileSupport.Delete(safeFile.OldFileName);
-
- fileSupport.Move(safeFile.TempFileName, safeFile.OriginalFileName, 100, 100);
- }
-
- else if (safeFile.OriginalFileExists && safeFile.OldFileExist)
- {
- fileSupport.Delete(safeFile.OldFileName);
- }
- }
- }
- else
- {
- throw new ArgumentException("Invalid folder " + folderPath);
- }
- }
- #endregion
- }
- }
|