SMBLibrary/SMBLibrary/Client/SMB2FileStore.cs

216 lines
8.1 KiB
C#

/* Copyright (C) 2017 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
*
* You can redistribute this program and/or modify it under the terms of
* the GNU Lesser Public License as published by the Free Software Foundation,
* either version 3 of the License, or (at your option) any later version.
*/
using System;
using System.Collections.Generic;
using SMBLibrary.SMB2;
using Utilities;
namespace SMBLibrary.Client
{
public class SMB2FileStore : INTFileStore
{
private SMB2Client m_client;
private uint m_treeID;
public SMB2FileStore(SMB2Client client, uint treeID)
{
m_client = client;
m_treeID = treeID;
}
public NTStatus CreateFile(out object handle, out FileStatus fileStatus, string path, AccessMask desiredAccess, FileAttributes fileAttributes, ShareAccess shareAccess, CreateDisposition createDisposition, CreateOptions createOptions, SecurityContext securityContext)
{
handle = null;
fileStatus = FileStatus.FILE_DOES_NOT_EXIST;
CreateRequest request = new CreateRequest();
request.Name = path;
request.DesiredAccess = desiredAccess;
request.FileAttributes = fileAttributes;
request.ShareAccess = shareAccess;
request.CreateDisposition = createDisposition;
request.CreateOptions = createOptions;
request.ImpersonationLevel = ImpersonationLevel.Impersonation;
TrySendCommand(request);
SMB2Command response = m_client.WaitForCommand(SMB2CommandName.Create);
if (response != null)
{
if (response.Header.Status == NTStatus.STATUS_SUCCESS && response is CreateResponse)
{
CreateResponse createResponse = ((CreateResponse)response);
handle = createResponse.FileId;
fileStatus = ToFileStatus(createResponse.CreateAction);
}
return response.Header.Status;
}
return NTStatus.STATUS_INVALID_SMB;
}
public NTStatus CloseFile(object handle)
{
CloseRequest request = new CloseRequest();
request.FileId = (FileID)handle;
TrySendCommand(request);
SMB2Command response = m_client.WaitForCommand(SMB2CommandName.Close);
if (response != null)
{
return response.Header.Status;
}
return NTStatus.STATUS_INVALID_SMB;
}
public NTStatus ReadFile(out byte[] data, object handle, long offset, int maxCount)
{
data = null;
ReadRequest request = new ReadRequest();
request.FileId = (FileID)handle;
request.Offset = (ulong)offset;
request.ReadLength = (uint)maxCount;
TrySendCommand(request);
SMB2Command response = m_client.WaitForCommand(SMB2CommandName.Read);
if (response != null)
{
if (response.Header.Status == NTStatus.STATUS_SUCCESS && response is ReadResponse)
{
data = ((ReadResponse)response).Data;
}
return response.Header.Status;
}
return NTStatus.STATUS_INVALID_SMB;
}
public NTStatus WriteFile(out int numberOfBytesWritten, object handle, long offset, byte[] data)
{
numberOfBytesWritten = 0;
WriteRequest request = new WriteRequest();
request.FileId = (FileID)handle;
request.Offset = (ulong)offset;
request.Data = data;
TrySendCommand(request);
SMB2Command response = m_client.WaitForCommand(SMB2CommandName.Write);
if (response != null)
{
if (response.Header.Status == NTStatus.STATUS_SUCCESS && response is WriteResponse)
{
numberOfBytesWritten = (int)((WriteResponse)response).Count;
}
return response.Header.Status;
}
return NTStatus.STATUS_INVALID_SMB;
}
public NTStatus FlushFileBuffers(object handle)
{
throw new NotImplementedException();
}
public NTStatus LockFile(object handle, long byteOffset, long length, bool exclusiveLock)
{
throw new NotImplementedException();
}
public NTStatus UnlockFile(object handle, long byteOffset, long length)
{
throw new NotImplementedException();
}
public NTStatus QueryDirectory(out List<QueryDirectoryFileInformation> result, object handle, string fileName, FileInformationClass informationClass)
{
throw new NotImplementedException();
}
public NTStatus GetFileInformation(out FileInformation result, object handle, FileInformationClass informationClass)
{
throw new NotImplementedException();
}
public NTStatus SetFileInformation(object handle, FileInformation information)
{
throw new NotImplementedException();
}
public NTStatus GetFileSystemInformation(out FileSystemInformation result, FileSystemInformationClass informationClass)
{
throw new NotImplementedException();
}
public NTStatus NotifyChange(out object ioRequest, object handle, NotifyChangeFilter completionFilter, bool watchTree, int outputBufferSize, OnNotifyChangeCompleted onNotifyChangeCompleted, object context)
{
throw new NotImplementedException();
}
public NTStatus Cancel(object ioRequest)
{
throw new NotImplementedException();
}
public NTStatus DeviceIOControl(object handle, uint ctlCode, byte[] input, out byte[] output, int maxOutputLength)
{
output = null;
IOCtlRequest request = new IOCtlRequest();
request.CtlCode = ctlCode;
request.IsFSCtl = true;
request.FileId = (FileID)handle;
request.Input = input;
request.MaxOutputResponse = (uint)maxOutputLength;
TrySendCommand(request);
SMB2Command response = m_client.WaitForCommand(SMB2CommandName.IOCtl);
if (response != null)
{
if ((response.Header.Status == NTStatus.STATUS_SUCCESS || response.Header.Status == NTStatus.STATUS_BUFFER_OVERFLOW) && response is IOCtlResponse)
{
output = ((IOCtlResponse)response).Output;
}
return response.Header.Status;
}
return NTStatus.STATUS_INVALID_SMB;
}
public NTStatus Disconnect()
{
TreeDisconnectRequest request = new TreeDisconnectRequest();
TrySendCommand(request);
SMB2Command response = m_client.WaitForCommand(SMB2CommandName.TreeDisconnect);
if (response != null)
{
return response.Header.Status;
}
return NTStatus.STATUS_INVALID_SMB;
}
private void TrySendCommand(SMB2Command request)
{
request.Header.TreeID = m_treeID;
m_client.TrySendCommand(request);
}
private static FileStatus ToFileStatus(CreateAction createAction)
{
switch (createAction)
{
case CreateAction.FILE_SUPERSEDED:
return FileStatus.FILE_SUPERSEDED;
case CreateAction.FILE_OPENED:
return FileStatus.FILE_OPENED;
case CreateAction.FILE_CREATED:
return FileStatus.FILE_CREATED;
case CreateAction.FILE_OVERWRITTEN:
return FileStatus.FILE_OVERWRITTEN;
default:
return FileStatus.FILE_OPENED;
}
}
}
}