summaryrefslogtreecommitdiff
path: root/plugins/FTPFileYM/job_packer.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'plugins/FTPFileYM/job_packer.cpp')
-rw-r--r--plugins/FTPFileYM/job_packer.cpp354
1 files changed, 354 insertions, 0 deletions
diff --git a/plugins/FTPFileYM/job_packer.cpp b/plugins/FTPFileYM/job_packer.cpp
new file mode 100644
index 0000000000..4d5c9ce4d5
--- /dev/null
+++ b/plugins/FTPFileYM/job_packer.cpp
@@ -0,0 +1,354 @@
+/*
+FTP File YM plugin
+Copyright (C) 2007-2010 Jan Holub
+
+This program is free software; you can redistribute it and/or
+modify it under the terms of the GNU General Public License
+as published by the Free Software Foundation version 2
+of the License.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with this program. If not, see <http://www.gnu.org/licenses/>.
+*/
+
+#include "job_packer.h"
+#include "job_upload.h"
+#include "dialog.h"
+#include "options.h"
+#include "utils.h"
+
+Event PackerJob::jobDone;
+Mutex PackerJob::mutexJobCount;
+int PackerJob::iRunningJobCount = 0;
+
+extern UploadDialog *uDlg;
+extern Options &opt;
+
+PackerJob::PackerJob(HANDLE _hContact, int _iFtpNum, EMode _mode)
+:GenericJob(_hContact, _iFtpNum, _mode),uiFileSize(0),uiReaded(0),lastUpdateTick(0)
+{ }
+
+void PackerJob::getZipFilePath()
+{
+ TCHAR buff[256], stzFileName[256] = {0};
+ TCHAR *pch;
+
+ if (this->files.size() == 1)
+ {
+ _tcscpy(stzFileName, Utils::getFileNameFromPath(this->files[0]));
+ pch = _tcsrchr(stzFileName, '.');
+ if (pch) *pch = 0;
+ }
+ else
+ {
+ _tcscpy(buff, this->files[0]);
+ pch = _tcsrchr(buff, '\\');
+ if (pch)
+ {
+ *pch = 0;
+ pch = _tcsrchr(buff, '\\');
+ if (pch) _tcscpy(stzFileName, pch + 1);
+ }
+ }
+
+ if (_tcslen(stzFileName) == 0)
+ _tcscpy(stzFileName, _T("archive"));
+
+ GetTempPath(SIZEOF(buff), buff);
+
+ mir_sntprintf(this->stzFilePath, SIZEOF(this->stzFilePath), _T("%s%s.zip"), buff, stzFileName);
+ _tcscpy(this->stzFileName, Utils::getFileNameFromPath(this->stzFilePath));
+
+ if (opt.bSetZipName)
+ Utils::setFileNameDlg(this->stzFileName);
+}
+
+void PackerJob::addToUploadDlg()
+{
+ this->getZipFilePath();
+
+ UploadDialog::Tab *newTab = new UploadDialog::Tab(this);
+ this->tab = newTab;
+ this->start();
+}
+
+void PackerJob::waitingThread(void *arg)
+{
+ PackerJob *job = (PackerJob *)arg;
+
+ while(!Miranda_Terminated())
+ {
+ Lock *lock = new Lock(mutexJobCount);
+ if (iRunningJobCount < MAX_RUNNING_JOBS)
+ {
+ iRunningJobCount++;
+ delete lock;
+ job->pack();
+ delete job;
+
+ Lock *lock = new Lock(mutexJobCount);
+ iRunningJobCount--;
+ delete lock;
+
+ jobDone.release();
+ return;
+ }
+
+ delete lock;
+ jobDone.wait();
+ job->status = GenericJob::STATUS_WAITING;
+ }
+}
+
+void PackerJob::start()
+{
+ mir_forkthread(&PackerJob::waitingThread, this);
+}
+
+void PackerJob::pack()
+{
+ struct _stat fileInfo;
+ for (UINT i = 0; i < this->files.size(); i++)
+ {
+ if (_tstat(this->files[i], &fileInfo) == 0)
+ this->uiFileSize += (UINT64)fileInfo.st_size;
+ }
+
+ this->setStatus(STATUS_PACKING);
+ this->startTS = time(NULL);
+
+ int res = this->createZipFile();
+ if (res == ZIP_OK)
+ {
+ UploadJob *ujob = new UploadJob(this);
+ ujob->tab->job = ujob;
+ ujob->start();
+ }
+ else
+ {
+ if (res == ZIP_ERRNO)
+ {
+ Utils::msgBox(TranslateT("Error occured when zipping the file(s)."), MB_OK | MB_ICONERROR);
+ delete this->tab;
+ }
+
+ DeleteFile(this->stzFilePath);
+ }
+}
+
+int PackerJob::createZipFile()
+{
+ int result = ZIP_ERRNO;
+
+ char *filePath = mir_t2a(this->stzFilePath);
+ zipFile zf = zipOpen2(filePath, 0, NULL, NULL);
+ FREE(filePath);
+
+ if (zf != NULL)
+ {
+ result = ZIP_OK;
+
+ int size_buf = 65536;
+ void *buff = (void *)mir_alloc(size_buf);
+
+ for (UINT i = 0; i < this->files.size(); i++)
+ {
+ int size_read;
+ zip_fileinfo zi;
+
+ zi.tmz_date.tm_sec = zi.tmz_date.tm_min = zi.tmz_date.tm_hour = 0;
+ zi.tmz_date.tm_mday = zi.tmz_date.tm_mon = zi.tmz_date.tm_year = 0;
+ zi.dosDate = 0;
+ zi.internal_fa = 0;
+ zi.external_fa = 0;
+
+ getFileTime(this->files[i], &zi.tmz_date, &zi.dosDate);
+
+ char *file = mir_t2a(Utils::getFileNameFromPath(this->files[i]));
+ int err = zipOpenNewFileInZip(zf, file, &zi, NULL, 0, NULL, 0, NULL, Z_DEFLATED, opt.iCompressionLevel);
+ FREE(file);
+
+ if (err == ZIP_OK)
+ {
+ FILE *fin = _tfopen(this->files[i], _T("rb"));
+ if (fin)
+ {
+ do
+ {
+ if (this->isCanceled())
+ {
+ fclose(fin);
+ result = STATUS_CANCELED;
+ goto Cleanup;
+ }
+
+ err = ZIP_OK;
+ size_read = (int)fread(buff, 1, size_buf, fin);
+ if (size_read < size_buf && feof(fin) == 0)
+ {
+ fclose(fin);
+ result = ZIP_ERRNO;
+ goto Cleanup;
+ }
+
+ if (size_read > 0)
+ {
+ err = zipWriteInFileInZip(zf, buff, size_read);
+ this->uiReaded += size_read;
+ }
+
+ this->updateStats();
+ }
+ while ((err == ZIP_OK) && (size_read > 0));
+ fclose(fin);
+ }
+ else
+ {
+ err = ZIP_ERRNO;
+ }
+
+ err = zipCloseFileInZip(zf);
+ if (err < 0)
+ {
+ result = ZIP_ERRNO;
+ goto Cleanup;
+ }
+ }
+ else
+ {
+ result = ZIP_ERRNO;
+ break;
+ }
+ }
+
+Cleanup:
+ zipClose(zf, NULL);
+ FREE(buff);
+ }
+
+ return result;
+}
+
+uLong PackerJob::getFileTime(TCHAR *file, tm_zip *tmzip, uLong *dt)
+{
+ FILETIME ftLocal;
+ HANDLE hFind;
+ WIN32_FIND_DATA ff32;
+
+ hFind = FindFirstFile(file, &ff32);
+ if (hFind != INVALID_HANDLE_VALUE)
+ {
+ FileTimeToLocalFileTime(&(ff32.ftLastWriteTime),&ftLocal);
+ FileTimeToDosDateTime(&ftLocal,((LPWORD)dt)+1,((LPWORD)dt)+0);
+ FindClose(hFind);
+ return 1;
+ }
+
+ return 0;
+}
+
+void PackerJob::updateStats()
+{
+ DWORD dwNewTick = GetTickCount();
+ if (this->uiReaded && (time(NULL) > this->startTS) && (dwNewTick > this->lastUpdateTick + 100))
+ {
+ this->lastUpdateTick = dwNewTick;
+
+ double speed = ((double)this->uiReaded / 1024)/(time(NULL) - this->startTS);
+ mir_sntprintf(this->tab->stzSpeed, SIZEOF(this->tab->stzSpeed), _T("%0.1f kB/s"), speed);
+
+ double perc = this->uiFileSize ? ((double)this->uiReaded / this->uiFileSize) * 100 : 0;
+ mir_sntprintf(this->tab->stzComplet, SIZEOF(this->tab->stzComplet), _T("%0.1f%% (%d kB/%d kB)"), perc, (int)this->uiReaded/1024, (int)this->uiFileSize/1024);
+
+ TCHAR buff[256];
+ long s = (this->uiFileSize - this->uiReaded) / (long)(speed * 1024);
+ int d = (s / 60 / 60 / 24);
+ int h = (s - d * 60 * 60 * 24) / 60 / 60;
+ int m = (s - d * 60 * 60 * 24 - h * 60 * 60) / 60;
+ s = s - (d * 24 * 60 * 60) - (h * 60 * 60) - (m * 60);
+
+ if (d > 0) mir_sntprintf(buff, SIZEOF(buff), _T("%dd %02d:%02d:%02d"), d, h, m, s);
+ else mir_sntprintf(buff, SIZEOF(buff), _T("%02d:%02d:%02d"), h, m, s);
+ mir_sntprintf(this->tab->stzRemain, SIZEOF(this->tab->stzRemain), _T("%s (%d kB/%d kB)"), buff, (this->uiFileSize - this->uiReaded)/1024, this->uiFileSize/1024);
+
+ this->refreshTab(false);
+ }
+}
+
+void PackerJob::refreshTab(bool bTabChanged)
+{
+ if (uDlg->activeTab == this->tab->index())
+ {
+ GenericJob::refreshTab(bTabChanged);
+
+ SetDlgItemText(uDlg->hwnd, IDC_UP_SPEED, this->tab->stzSpeed);
+ SetDlgItemText(uDlg->hwnd, IDC_UP_COMPLETED, this->tab->stzComplet);
+ SetDlgItemText(uDlg->hwnd, IDC_UP_REMAIN, this->tab->stzRemain);
+
+ SendDlgItemMessage(uDlg->hwnd, IDC_PB_UPLOAD, PBM_SETRANGE32, 0, (LPARAM)this->uiFileSize);
+ SendDlgItemMessage(uDlg->hwnd, IDC_PB_UPLOAD, PBM_SETPOS, (WPARAM)this->uiReaded, 0);
+
+ if (bTabChanged)
+ {
+ SetDlgItemText(uDlg->hwnd, IDC_STATUSBAR, TranslateT("PACKING..."));
+ EnableWindow(GetDlgItem(uDlg->hwnd, IDC_BTN_PAUSE), FALSE);
+ }
+ }
+}
+
+bool PackerJob::isCanceled()
+{
+ return this->status == STATUS_CANCELED;
+}
+
+void PackerJob::pauseHandler()
+{
+ /* Not implemented */
+}
+
+void PackerJob::pause()
+{
+ /* Not implemented */
+}
+
+void PackerJob::resume()
+{
+ /* Not implemented */
+}
+
+void PackerJob::cancel()
+{
+ this->setStatus(STATUS_CANCELED);
+}
+
+void PackerJob::closeTab()
+{
+ if (Utils::msgBox(TranslateT("Do you really want to cancel this upload?"), MB_YESNO | MB_ICONQUESTION) == IDYES)
+ {
+ this->cancel();
+ delete this->tab;
+ }
+}
+
+void PackerJob::closeAllTabs()
+{
+ this->cancel();
+ delete this->tab;
+}
+
+void PackerJob::createToolTip()
+{
+ TCHAR *server = mir_a2t(this->ftp->szServer);
+ mir_sntprintf(uDlg->stzToolTipText, SIZEOF(uDlg->stzToolTipText),
+ TranslateT("Status: %s\r\nFile: %s\r\nServer: %s"),
+ this->getStatusString(),
+ this->stzFileName,
+ server);
+
+ FREE(server);
+} \ No newline at end of file