2016-08-03 10:59:49 +00:00
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
// <copyright file="TempFiles.cs" company="Microsoft">
|
|
|
|
//
|
2017-08-21 15:34:15 +00:00
|
|
|
// <OWNER>Microsoft</OWNER>
|
2016-08-03 10:59:49 +00:00
|
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
|
|
// </copyright>
|
|
|
|
//------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
namespace System.CodeDom.Compiler {
|
|
|
|
using System;
|
|
|
|
using System.Collections;
|
|
|
|
using System.Diagnostics;
|
|
|
|
using System.IO;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using System.Text;
|
|
|
|
using Microsoft.Win32;
|
|
|
|
using System.Security;
|
|
|
|
using System.Security.Permissions;
|
|
|
|
using System.Security.Principal;
|
|
|
|
using System.ComponentModel;
|
|
|
|
using System.Security.Cryptography;
|
|
|
|
using System.Globalization;
|
|
|
|
using System.Runtime.Versioning;
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>Represents a collection of temporary file names that are all based on a
|
|
|
|
/// single base filename located in a temporary directory.</para>
|
|
|
|
/// </devdoc>
|
|
|
|
[PermissionSet(SecurityAction.LinkDemand, Name="FullTrust")]
|
|
|
|
[PermissionSet(SecurityAction.InheritanceDemand, Name="FullTrust")]
|
|
|
|
[Serializable]
|
|
|
|
public class TempFileCollection : ICollection, IDisposable {
|
|
|
|
string basePath;
|
|
|
|
string tempDir;
|
|
|
|
bool keepFiles;
|
|
|
|
Hashtable files;
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public TempFileCollection() : this(null, false) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public TempFileCollection(string tempDir) : this(tempDir, false) {
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public TempFileCollection(string tempDir, bool keepFiles) {
|
|
|
|
this.keepFiles = keepFiles;
|
|
|
|
this.tempDir = tempDir;
|
|
|
|
#if !FEATURE_CASE_SENSITIVE_FILESYSTEM
|
|
|
|
files = new Hashtable(StringComparer.OrdinalIgnoreCase);
|
|
|
|
#else
|
|
|
|
files = new Hashtable();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <internalonly/>
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para> To allow it's stuff to be cleaned up</para>
|
|
|
|
/// </devdoc>
|
|
|
|
void IDisposable.Dispose() {
|
|
|
|
Dispose(true);
|
|
|
|
GC.SuppressFinalize(this);
|
|
|
|
}
|
|
|
|
protected virtual void Dispose(bool disposing) {
|
|
|
|
// It is safe to call Delete from here even if Dispose is called from Finalizer
|
|
|
|
// because the graph of objects is guaranteed to be there and
|
|
|
|
// neither Hashtable nor String have a finalizer of their own that could
|
|
|
|
// be called before TempFileCollection Finalizer
|
|
|
|
Delete();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
~TempFileCollection() {
|
|
|
|
Dispose(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public string AddExtension(string fileExtension) {
|
|
|
|
return AddExtension(fileExtension, keepFiles);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public string AddExtension(string fileExtension, bool keepFile) {
|
|
|
|
if (fileExtension == null || fileExtension.Length == 0)
|
|
|
|
throw new ArgumentException(SR.GetString(SR.InvalidNullEmptyArgument, "fileExtension"), "fileExtension"); // fileExtension not specified
|
|
|
|
string fileName = BasePath + "." + fileExtension;
|
|
|
|
AddFile(fileName, keepFile);
|
|
|
|
return fileName;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public void AddFile(string fileName, bool keepFile) {
|
|
|
|
if (fileName == null || fileName.Length == 0)
|
|
|
|
throw new ArgumentException(SR.GetString(SR.InvalidNullEmptyArgument, "fileName"), "fileName"); // fileName not specified
|
|
|
|
|
|
|
|
if (files[fileName] != null)
|
|
|
|
throw new ArgumentException(SR.GetString(SR.DuplicateFileName, fileName), "fileName"); // duplicate fileName
|
|
|
|
files.Add(fileName, (object)keepFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public IEnumerator GetEnumerator() {
|
|
|
|
return files.Keys.GetEnumerator();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <internalonly/>
|
|
|
|
IEnumerator IEnumerable.GetEnumerator() {
|
|
|
|
return files.Keys.GetEnumerator();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <internalonly/>
|
|
|
|
void ICollection.CopyTo(Array array, int start) {
|
|
|
|
files.Keys.CopyTo(array, start);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public void CopyTo(string[] fileNames, int start) {
|
|
|
|
files.Keys.CopyTo(fileNames, start);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public int Count {
|
|
|
|
get {
|
|
|
|
return files.Count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <internalonly/>
|
|
|
|
int ICollection.Count {
|
|
|
|
get { return files.Count; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <internalonly/>
|
|
|
|
object ICollection.SyncRoot {
|
|
|
|
get { return null; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <internalonly/>
|
|
|
|
bool ICollection.IsSynchronized {
|
|
|
|
get { return false; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public string TempDir {
|
|
|
|
get { return tempDir == null ? string.Empty : tempDir; }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public string BasePath {
|
|
|
|
get {
|
|
|
|
EnsureTempNameCreated();
|
|
|
|
return basePath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
[ResourceExposure(ResourceScope.None)]
|
|
|
|
[ResourceConsumption(ResourceScope.Machine, ResourceScope.Machine)]
|
|
|
|
void EnsureTempNameCreated() {
|
|
|
|
if (basePath == null) {
|
|
|
|
|
|
|
|
string tempFileName = null;
|
|
|
|
FileStream tempFileStream;
|
|
|
|
bool uniqueFile = false;
|
|
|
|
int retryCount = 5000;
|
|
|
|
do {
|
|
|
|
try {
|
|
|
|
basePath = GetTempFileName(TempDir);
|
|
|
|
|
|
|
|
string full = Path.GetFullPath(basePath);
|
|
|
|
|
|
|
|
new FileIOPermission(FileIOPermissionAccess.AllAccess, full).Demand();
|
|
|
|
|
|
|
|
// make sure the filename is unique.
|
|
|
|
tempFileName = basePath + ".tmp";
|
|
|
|
using (tempFileStream = new FileStream(tempFileName, FileMode.CreateNew, FileAccess.Write)) { }
|
|
|
|
uniqueFile = true;
|
|
|
|
}
|
|
|
|
catch (IOException e) {
|
|
|
|
retryCount--;
|
|
|
|
|
|
|
|
uint HR_ERROR_FILE_EXISTS = unchecked(((uint)0x80070000) | NativeMethods.ERROR_FILE_EXISTS);
|
|
|
|
if (retryCount == 0 || Marshal.GetHRForException(e) != HR_ERROR_FILE_EXISTS)
|
|
|
|
throw;
|
|
|
|
|
|
|
|
uniqueFile = false;
|
|
|
|
}
|
|
|
|
}while (!uniqueFile);
|
|
|
|
files.Add(tempFileName, keepFiles);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
public bool KeepFiles {
|
|
|
|
get { return keepFiles; }
|
|
|
|
set { keepFiles = value; }
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KeepFile(string fileName) {
|
|
|
|
object keep = files[fileName];
|
|
|
|
if (keep == null) return false;
|
|
|
|
return (bool)keep;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// <devdoc>
|
|
|
|
/// <para>[To be supplied.]</para>
|
|
|
|
/// </devdoc>
|
|
|
|
[ResourceExposure(ResourceScope.None)]
|
|
|
|
[ResourceConsumption(ResourceScope.Machine, ResourceScope.Machine)]
|
|
|
|
public void Delete() {
|
|
|
|
if (files != null && files.Count > 0) {
|
|
|
|
string[] fileNames = new string[files.Count];
|
|
|
|
files.Keys.CopyTo(fileNames, 0);
|
|
|
|
foreach (string fileName in fileNames) {
|
|
|
|
if (!KeepFile(fileName)) {
|
|
|
|
Delete(fileName);
|
|
|
|
files.Remove(fileName);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// This function deletes files after reverting impersonation.
|
|
|
|
[ResourceExposure(ResourceScope.None)]
|
|
|
|
[ResourceConsumption(ResourceScope.Process, ResourceScope.Process)]
|
|
|
|
internal void SafeDelete() {
|
|
|
|
#if !FEATURE_PAL
|
|
|
|
WindowsImpersonationContext impersonation = Executor.RevertImpersonation();
|
|
|
|
#endif
|
|
|
|
try{
|
|
|
|
Delete();
|
|
|
|
}
|
|
|
|
finally {
|
|
|
|
#if !FEATURE_PAL
|
|
|
|
Executor.ReImpersonate(impersonation);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
[ResourceExposure(ResourceScope.Machine)]
|
|
|
|
[ResourceConsumption(ResourceScope.Machine)]
|
|
|
|
void Delete(string fileName) {
|
|
|
|
try {
|
|
|
|
File.Delete(fileName);
|
|
|
|
}
|
|
|
|
catch {
|
|
|
|
// Ignore all exceptions
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
[ResourceExposure(ResourceScope.Machine)]
|
|
|
|
[ResourceConsumption(ResourceScope.Machine)]
|
|
|
|
static string GetTempFileName(string tempDir) {
|
|
|
|
string fileName;
|
|
|
|
if (String.IsNullOrEmpty(tempDir)) tempDir = Path.GetTempPath();
|
|
|
|
|
|
|
|
string randomFileName = Path.GetFileNameWithoutExtension(Path.GetRandomFileName());
|
|
|
|
|
|
|
|
if (tempDir.EndsWith("\\", StringComparison.Ordinal))
|
|
|
|
fileName = tempDir + randomFileName;
|
|
|
|
else
|
|
|
|
fileName = tempDir + "\\" + randomFileName;
|
|
|
|
|
|
|
|
return fileName;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|