SMB1Session.cs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. /* Copyright (C) 2014-2017 Tal Aloni <tal.aloni.il@gmail.com>. All rights reserved.
  2. *
  3. * You can redistribute this program and/or modify it under the terms of
  4. * the GNU Lesser Public License as published by the Free Software Foundation,
  5. * either version 3 of the License, or (at your option) any later version.
  6. */
  7. using System;
  8. using System.Collections.Generic;
  9. using System.IO;
  10. using Utilities;
  11. namespace SMBLibrary.Server
  12. {
  13. internal class SMB1Session
  14. {
  15. private const int MaxSearches = 2048; // Windows servers initialize Server.MaxSearches to 2048.
  16. private SMB1ConnectionState m_connection;
  17. private ushort m_userID;
  18. private byte[] m_sessionKey;
  19. private SecurityContext m_securityContext;
  20. private DateTime m_creationDT;
  21. // Key is TID
  22. private Dictionary<ushort, ISMBShare> m_connectedTrees = new Dictionary<ushort, ISMBShare>();
  23. // Key is FID
  24. private Dictionary<ushort, OpenFileObject> m_openFiles = new Dictionary<ushort, OpenFileObject>();
  25. // Key is search handle a.k.a. Search ID
  26. private Dictionary<ushort, OpenSearch> m_openSearches = new Dictionary<ushort, OpenSearch>();
  27. private ushort m_nextSearchHandle = 1;
  28. public SMB1Session(SMB1ConnectionState connection, ushort userID, string userName, string machineName, byte[] sessionKey, object accessToken)
  29. {
  30. m_connection = connection;
  31. m_userID = userID;
  32. m_sessionKey = sessionKey;
  33. m_securityContext = new SecurityContext(userName, machineName, connection.ClientEndPoint, connection.AuthenticationContext, accessToken);
  34. m_creationDT = DateTime.Now;
  35. }
  36. public ushort? AddConnectedTree(ISMBShare share)
  37. {
  38. ushort? treeID = m_connection.AllocateTreeID();
  39. if (treeID.HasValue)
  40. {
  41. m_connectedTrees.Add(treeID.Value, share);
  42. }
  43. return treeID;
  44. }
  45. public ISMBShare GetConnectedTree(ushort treeID)
  46. {
  47. ISMBShare share;
  48. m_connectedTrees.TryGetValue(treeID, out share);
  49. return share;
  50. }
  51. public void DisconnectTree(ushort treeID)
  52. {
  53. ISMBShare share;
  54. m_connectedTrees.TryGetValue(treeID, out share);
  55. if (share != null)
  56. {
  57. List<ushort> fileIDList = new List<ushort>(m_openFiles.Keys);
  58. foreach (ushort fileID in fileIDList)
  59. {
  60. OpenFileObject openFile = m_openFiles[fileID];
  61. if (openFile.TreeID == treeID)
  62. {
  63. share.FileStore.CloseFile(openFile.Handle);
  64. m_openFiles.Remove(fileID);
  65. }
  66. }
  67. m_connectedTrees.Remove(treeID);
  68. }
  69. }
  70. public bool IsTreeConnected(ushort treeID)
  71. {
  72. return m_connectedTrees.ContainsKey(treeID);
  73. }
  74. /// <param name="relativePath">Should include the path relative to the share</param>
  75. /// <returns>FileID</returns>
  76. public ushort? AddOpenFile(ushort treeID, string relativePath)
  77. {
  78. return AddOpenFile(treeID, relativePath, null);
  79. }
  80. public ushort? AddOpenFile(ushort treeID, string relativePath, object handle)
  81. {
  82. ushort? fileID = m_connection.AllocateFileID();
  83. if (fileID.HasValue)
  84. {
  85. m_openFiles.Add(fileID.Value, new OpenFileObject(treeID, relativePath, handle));
  86. }
  87. return fileID;
  88. }
  89. public OpenFileObject GetOpenFileObject(ushort fileID)
  90. {
  91. OpenFileObject openFile;
  92. m_openFiles.TryGetValue(fileID, out openFile);
  93. return openFile;
  94. }
  95. public void RemoveOpenFile(ushort fileID)
  96. {
  97. m_openFiles.Remove(fileID);
  98. }
  99. private ushort? AllocateSearchHandle()
  100. {
  101. for (ushort offset = 0; offset < UInt16.MaxValue; offset++)
  102. {
  103. ushort searchHandle = (ushort)(m_nextSearchHandle + offset);
  104. if (searchHandle == 0 || searchHandle == 0xFFFF)
  105. {
  106. continue;
  107. }
  108. if (!m_openSearches.ContainsKey(searchHandle))
  109. {
  110. m_nextSearchHandle = (ushort)(searchHandle + 1);
  111. return searchHandle;
  112. }
  113. }
  114. return null;
  115. }
  116. public ushort? AddOpenSearch(List<QueryDirectoryFileInformation> entries, int enumerationLocation)
  117. {
  118. ushort? searchHandle = AllocateSearchHandle();
  119. if (searchHandle.HasValue)
  120. {
  121. OpenSearch openSearch = new OpenSearch(entries, enumerationLocation);
  122. m_openSearches.Add(searchHandle.Value, openSearch);
  123. }
  124. return searchHandle;
  125. }
  126. public OpenSearch GetOpenSearch(ushort searchHandle)
  127. {
  128. OpenSearch openSearch;
  129. m_openSearches.TryGetValue(searchHandle, out openSearch);
  130. return openSearch;
  131. }
  132. public void RemoveOpenSearch(ushort searchHandle)
  133. {
  134. m_openSearches.Remove(searchHandle);
  135. }
  136. public ushort UserID
  137. {
  138. get
  139. {
  140. return m_userID;
  141. }
  142. }
  143. public SecurityContext SecurityContext
  144. {
  145. get
  146. {
  147. return m_securityContext;
  148. }
  149. }
  150. public string UserName
  151. {
  152. get
  153. {
  154. return m_securityContext.UserName;
  155. }
  156. }
  157. public string MachineName
  158. {
  159. get
  160. {
  161. return m_securityContext.MachineName;
  162. }
  163. }
  164. public DateTime CreationDT
  165. {
  166. get
  167. {
  168. return m_creationDT;
  169. }
  170. }
  171. }
  172. }