QueryDirectoryHelper.cs 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. /* Copyright (C) 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 SMBLibrary.Authentication;
  10. using SMBLibrary.SMB2;
  11. using Utilities;
  12. namespace SMBLibrary.Server.SMB2
  13. {
  14. internal class QueryDirectoryHelper
  15. {
  16. internal static SMB2Command GetQueryDirectoryResponse(QueryDirectoryRequest request, ISMBShare share, SMB2ConnectionState state)
  17. {
  18. SMB2Session session = state.GetSession(request.Header.SessionID);
  19. OpenFileObject openFile = session.GetOpenFileObject(request.FileId);
  20. if (openFile == null)
  21. {
  22. return new ErrorResponse(request.CommandName, NTStatus.STATUS_FILE_CLOSED);
  23. }
  24. if (!((FileSystemShare)share).HasReadAccess(session.SecurityContext, openFile.Path))
  25. {
  26. state.LogToServer(Severity.Verbose, "Query Directory on '{0}{1}' failed. User '{2}' was denied access.", share.Name, openFile.Path, session.UserName);
  27. return new ErrorResponse(request.CommandName, NTStatus.STATUS_ACCESS_DENIED);
  28. }
  29. FileSystemShare fileSystemShare = (FileSystemShare)share;
  30. FileID fileID = request.FileId;
  31. OpenSearch openSearch = session.GetOpenSearch(fileID);
  32. if (openSearch == null || request.Reopen)
  33. {
  34. if (request.Reopen)
  35. {
  36. session.RemoveOpenSearch(fileID);
  37. }
  38. List<QueryDirectoryFileInformation> entries;
  39. NTStatus searchStatus = share.FileStore.QueryDirectory(out entries, openFile.Handle, request.FileName, request.FileInformationClass);
  40. if (searchStatus != NTStatus.STATUS_SUCCESS)
  41. {
  42. state.LogToServer(Severity.Verbose, "Query Directory on '{0}{1}', Searched for '{2}', NTStatus: {3}", share.Name, openFile.Path, request.FileName, searchStatus.ToString());
  43. return new ErrorResponse(request.CommandName, searchStatus);
  44. }
  45. state.LogToServer(Severity.Information, "Query Directory on '{0}{1}', Searched for '{2}', found {3} matching entries", share.Name, openFile.Path, request.FileName, entries.Count);
  46. openSearch = session.AddOpenSearch(fileID, entries, 0);
  47. }
  48. if (request.Restart || request.Reopen)
  49. {
  50. openSearch.EnumerationLocation = 0;
  51. }
  52. if (openSearch.Entries.Count == 0)
  53. {
  54. // [MS-SMB2] If there are no entries to return [..] the server MUST fail the request with STATUS_NO_SUCH_FILE.
  55. session.RemoveOpenSearch(fileID);
  56. return new ErrorResponse(request.CommandName, NTStatus.STATUS_NO_SUCH_FILE);
  57. }
  58. if (openSearch.EnumerationLocation == openSearch.Entries.Count)
  59. {
  60. return new ErrorResponse(request.CommandName, NTStatus.STATUS_NO_MORE_FILES);
  61. }
  62. List<QueryDirectoryFileInformation> page = new List<QueryDirectoryFileInformation>();
  63. int pageLength = 0;
  64. for (int index = openSearch.EnumerationLocation; index < openSearch.Entries.Count; index++)
  65. {
  66. QueryDirectoryFileInformation fileInformation = openSearch.Entries[index];
  67. if (fileInformation.FileInformationClass != request.FileInformationClass)
  68. {
  69. // We do not support changing FileInformationClass during a search (unless SMB2_REOPEN is set).
  70. return new ErrorResponse(request.CommandName, NTStatus.STATUS_INVALID_PARAMETER);
  71. }
  72. int entryLength = fileInformation.Length;
  73. if (pageLength + entryLength <= request.OutputBufferLength)
  74. {
  75. page.Add(fileInformation);
  76. int paddedLength = (int)Math.Ceiling((double)entryLength / 8) * 8;
  77. pageLength += paddedLength;
  78. openSearch.EnumerationLocation = index + 1;
  79. }
  80. else
  81. {
  82. break;
  83. }
  84. if (request.ReturnSingleEntry)
  85. {
  86. break;
  87. }
  88. }
  89. QueryDirectoryResponse response = new QueryDirectoryResponse();
  90. response.SetFileInformationList(page);
  91. return response;
  92. }
  93. }
  94. }