createfile — C ++ Мониторинг обмена файлами с помощью ReadDirectoryChangesНе вызывать все действия?

У меня есть приложение, которое должно отслеживать любые изменения в определенном каталоге. Я использую следующий код, однако при переименовании файла, например, действие FILE_ACTION_RENAMED_NEW_NAME не срабатывает. На самом деле, UNDISCOVERED ACTION срабатывает get, что приводит к неожиданному поведению? Что я здесь не так делаю?

char Dir[] = "DIRPATH";
HANDLE hDir = CreateFile(
Dir,
FILE_LIST_DIRECTORY,
FILE_SHARE_WRITE | FILE_SHARE_READ | FILE_SHARE_DELETE,
NULL,
OPEN_EXISTING,
FILE_FLAG_BACKUP_SEMANTICS,
NULL);
int nCounter = 0;
FILE_NOTIFY_INFORMATION strFileNotifyInfo[1024];
FILE_NOTIFY_INFORMATION *fni = NULL;

while (TRUE)
{
//strFileNotifyInfo = NULL;
DWORD dwBytesReturned = 0;
if (ReadDirectoryChangesW(hDir, (LPVOID)&strFileNotifyInfo, sizeof(strFileNotifyInfo), TRUE, FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_DIR_NAME | FILE_NOTIFY_CHANGE_LAST_WRITE, &dwBytesReturned, NULL, NULL) == 0)
{
Exit(GetLastErrorAsString());
}
else
{
char fileName[MAX_PATH] = "";
DWORD offset = 0;
do {
fni = (FILE_NOTIFY_INFORMATION*)(&strFileNotifyInfo[offset]);
int ret = ::WideCharToMultiByte(CP_ACP, 0, fni->FileName, fni->FileNameLength / sizeof(WCHAR), fileName, sizeof(fileName), NULL, NULL);
string Test = Dir;
Test += "\\";
Test += fileName;
switch (fni->Action) {
case FILE_ACTION_ADDED:
if (boost::filesystem::is_directory(Test)) {
cout << "Directory added: " << Dir << "\\" << fileName << endl;
}
else {
cout << "File added: " << Dir << "\\" << fileName << endl;
}
break;
case FILE_ACTION_MODIFIED:
if (boost::filesystem::is_directory(Test)) {
cout << "Directory modified: " << Dir << "\\" << fileName << endl;
}
else {
cout << "File modified: " << Dir << "\\" << fileName << endl;
}
break;
case FILE_ACTION_REMOVED:
if (boost::filesystem::is_directory(Test)) {
cout << "Directory removed: " << Dir << "\\" << fileName << endl;
}
else {
cout << "File removed: " << Dir << "\\" << fileName << endl;
}
break;
case FILE_ACTION_RENAMED_NEW_NAME:
if (boost::filesystem::is_directory(Test)) {
cout << "Directory renamend (NEW): " << Dir << "\\" << fileName << endl;
}
else {
cout << "File renamed (NEW): " << Dir << "\\" << fileName << endl;
}
break;
case FILE_ACTION_RENAMED_OLD_NAME:
if (boost::filesystem::is_directory(Test)) {
cout << "Directory renamed (OLD): " << Dir << "\\" << fileName << endl;
}
else {
cout << "File renamed (OLD): " << Dir << "\\" << fileName << endl;
}
break;
default:
if (boost::filesystem::is_directory(Test)) {
cout << "Directory UNDISCOVERED ACTION: " << Dir << "\\" << fileName << endl;
}
else {
cout << "File UNDISCOVERED ACTION: " << Dir << "\\" << fileName << endl;
}
break;
}
::memset(fileName, '\0', sizeof(fileName));
offset += fni->NextEntryOffset;
}
while (fni->NextEntryOffset != 0);
cout << "Loop: " << nCounter++ << endl;
}
}

Пример вывода при переименовании файла из file.txt в file2.txt на карте названо: MAP:

File renamed (OLD): DIRPATH\MAP\file.txt
Directory UNDISCOVERED ACTION: DIRPATH
Loop: 0
Directory modified: DIRPATH\MAP
Loop: 1

1

Решение

FILE_NOTIFY_INFORMATION не является структурой фиксированного размера. Он представляет заголовок фиксированного размера, за которым следует имя файла — строка переменного размера.

&strFileNotifyInfo[1] указывает на некоторое смещение в середине имени файла, которое следует strFileNotifyInfo[0], Его значения — просто неправильно интерпретированные куски памяти, заполненные символами имени файла; по сути случайный мусор.

Вместо этого вы должны использовать FILE_NOTIFY_INFORMATION::NextEntryOffset найти следующий экземпляр FILE_NOTIFY_INFORMATION в буфере.

Ваш код будет выглядеть примерно так:

BYTE buffer[4096];
ReadDirectoryChangesW(hDir, buffer, sizeof(buffer), ...);
BYTE* p = buffer;
for (;;) {
FILE_NOTIFY_INFORMATION* info =
reinterpret_cast<FILE_NOTIFY_INFORMATION*>(p);

// Work with `info` as necessary

if (!info->NextEntryOffset) break;  // this was last entry
p += info->NextEntryOffset;
}
2

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]