mirror of
https://github.com/TalAloni/SMBLibrary.git
synced 2025-07-26 10:48:15 +02:00
216 lines
8.1 KiB
C#
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;
|
|
}
|
|
}
|
|
}
|
|
}
|