Удаление файла при задействовании его в 2-х проектах

shtift

Client
Регистрация
29.07.2015
Сообщения
148
Благодарностей
291
Баллы
63
Привет всем.
Создаю в первом проекте файл и через определенное время, во втором проекте проверяю, что файл был создан и после некоторых манипуляций с данными из файла, удаляю этот файл.
При попытке удалить файл проект завершается и выводится сообщение "Не удается получить доступ к файлу, т.к. файл занят другим процессом". По коду ошибки ругается именно на удаление файла.
Я сделал вывод, что эта проблема возникает из-за того, что одновременно два проекта работают с одним и тем же файлом.
Если это верно, то как сделать так, чтобы первый проект "забыл" про файл и продолжал работать, дабы второй проект мог легко удалить файл?
 

Sergodjan

Administrator
Команда форума
Регистрация
05.09.2012
Сообщения
20 407
Благодарностей
9 115
Баллы
113
Привет всем.
Создаю в первом проекте файл и через определенное время, во втором проекте проверяю, что файл был создан и после некоторых манипуляций с данными из файла, удаляю этот файл.
При попытке удалить файл проект завершается и выводится сообщение "Не удается получить доступ к файлу, т.к. файл занят другим процессом". По коду ошибки ругается именно на удаление файла.
Я сделал вывод, что эта проблема возникает из-за того, что одновременно два проекта работают с одним и тем же файлом.
Если это верно, то как сделать так, чтобы первый проект "забыл" про файл и продолжал работать, дабы второй проект мог легко удалить файл?
файл не привязан к списку в первом или втором проекте?
 

shtift

Client
Регистрация
29.07.2015
Сообщения
148
Благодарностей
291
Баллы
63

LexxWork

Client
Регистрация
31.10.2013
Сообщения
1 190
Благодарностей
791
Баллы
113
Код:
var path = @"D:\downloads\1.txt"; //путь к файлу
var timeout = 10; //секунд

if(!File.Exists(path)){
    project.SendInfoToLog("файл "+path+" не существует");
    return "ok";
}

while(timeout-- > 0){
    try{
        File.Delete(path);
    }catch (IOException e){
        var errorcode = System.Runtime.InteropServices.Marshal.GetHRForException(e) & ((1 << 16) - 1);
        var message = e.Message;
        if(errorcode == 32 || errorcode == 33){
            project.SendWarningToLog("файл "+path+" заблокирован");
            new System.Threading.ManualResetEvent(false).WaitOne(1000);
            continue;
        }
       throw e;
    }
    return "ok";
}

if(timeout <= 0 && File.Exists(path))
    throw new Exception("не удалось удалить файл");
должен признаться, что этот способ очень корявый или лучше сказать наивный. если не понятно почему, то просто пердставьте что этот код ждет секунду и тут же пытается удалить хотя файла может и не быть уже или он обновлен или он как раз только что освободился или другой проект уже начал писать а вы его удалить хотите. Короче на свой страх и риск.
 
Последнее редактирование:

shtift

Client
Регистрация
29.07.2015
Сообщения
148
Благодарностей
291
Баллы
63
Нельзя задействовать такой алгоритм, т.к. мне нужно чтобы задержек по времени не было. Тем более, что после принудительной остановки проекта (первого) файл все еще не удаляется в течение 0:30 - 1 мин, даже простым перемещением в корзину. Выходит, что CaptchaSaver зачем-то "держит" файл, хотя в дальнейшем не контактирует с ним.

Либо проблема может быть еще вот в чем. Во втором проекте этот файл отправляется на сервер, после отправки управление передается первому проекту и он уже пытается удалить файл. Возможно второй проект блокирует? Вот код, отправляющий файл:
Код:
var image = System.Drawing.Image.FromFile(@"C:\Files\1.jpg");

string base64String = String.Empty;

using (System.IO.MemoryStream ms = new System.IO.MemoryStream())
{
    image.Save(ms, System.Drawing.Imaging.ImageFormat.Png);
    byte[] imageBytes = ms.ToArray();
    
    base64String = Convert.ToBase64String(imageBytes);
}

var result = ZennoPoster.CaptchaRecognition("Anti-Captcha.dll", base64String, "");

var split = result.Split(new [] {"-|-"}, StringSplitOptions.RemoveEmptyEntries);
if (split.Length == 2) return split[0];
 

LexxWork

Client
Регистрация
31.10.2013
Сообщения
1 190
Благодарностей
791
Баллы
113
зачем так извращаться? отправляйте капчу там же где ее получили. При том вобще не обязательно хранить капчу в файле, производите распозновоание в стандартном экшене.
 

shtift

Client
Регистрация
29.07.2015
Сообщения
148
Благодарностей
291
Баллы
63
зачем так извращаться? отправляйте капчу там же где ее получили. При том вобще не обязательно хранить капчу в файле, производите распозновоание в стандартном экшене.
Согласен, но нужно чтобы, пока капча отправляется, в это время выполнялись другие манипуляции. А т.к. нельзя параллельно выполнять в одном проекте 2 действия, отсюда и делаю 2 проекта - один работает по своему плану, второй в это время отправляет картинку стандартным экшеном.

Делаю через файлы, т.к. капча получается в base64, а декодировать ее не выходит. Может тогда подскажете как декодировать? Или можно сразу в таком виде отправлять?

Вот в таком виде приходит:
HTML:
<div style="background-image: url(&quot;&quot;);" id="captchaimg" class="image"></div>
 

shtift

Client
Регистрация
29.07.2015
Сообщения
148
Благодарностей
291
Баллы
63
Вроде разобрался. Парсю регуляркой строку и кладу в переменную gBase64String. Отправляю на разгадывание:
Код:
var result = ZennoPoster.CaptchaRecognition("Anti-Captcha.dll", gBase64String, "");
var split = result.Split(new [] {"-|-"}, StringSplitOptions.RemoveEmptyEntries);
if (split.Length == 2) return split[0];
Подскажите, все верно? Возможно код содержит ошибки.
 

LexxWork

Client
Регистрация
31.10.2013
Сообщения
1 190
Благодарностей
791
Баллы
113
можно попробовать хранить b64 строки капч в привязаном списк/таблице, но я точно не знаю как они будут согласовываться между проектами.

формат списка будет айди капчи разделитель b64 разделитель ответ. Если ненравится хранить строку картинки в списке, можно просто хранить ее в файле без каких либо перекодировок как обычный текст, а в таблице просто хранить путь к капче.
также нужно сделать так чтобы айди капчи (название файла) не пересекались.
при добавлении в список я думаю лочить не нужно
после того как главный шаблон выполнит все нужные действия, нужно будет остановится и поискать в списке свою капчу с ответом. Если ошибка - разгадывать уже в шаблоне. Если ответ пустой - ждать пока прийдет ответ. Здесь прийдется лочить список при удалении.

шаблон-разгадыватель должен работать в несколько (столько же сколько других проектов его используют) потоков, потому что если меньше то будет задержка и время выиграть неудастся. Так же нужно предусмотреть чтобы после рагадывания этот шаблон не удалял из списка разгаданую капчу, потому что не знает взял ли соответсвующий проект ответ или еще нет. Файл можно будет удалить сразу же после первого разгадывания. Здесь список нужно редактировать поэтому список нужно лочить.


Вобще я писал на форуме что можно разгадывать капчу и даже уже сделал и протестировал универсальный снипет. Его плюс в том что он полностю независим от других шаблонов, не сохраняет левых файлов и не делает лишних перекодировок, абсолютно универсальный, может использовать любые доступные в зенопостере модули разгадывания капчи, не использует поток зенопостера и может работать даже в демке (не тестил но предполагаю). Так же по желанию может быть предусмотрена возможность сжимания картинки до минимально приемлемого размера. Большой минус в том что он сделан только для продажи.
ps
лучше так
var result = split[0];
if (split.Length >1) result = split[1];
return result;
 
Последнее редактирование:
  • Спасибо
Реакции: shtift

Кто просматривает тему: (Всего: 2, Пользователи: 0, Гости: 2)