1 /*******************************************************************************
\r
2 TPI - flexible but useless plug-in framework.
\r
3 Copyright (C) 2002-2009 Silky
\r
5 This library is free software; you can redistribute it and/or modify it under
\r
6 the terms of the GNU Lesser General Public License as published by the Free
\r
7 Software Foundation; either version 2.1 of the License, or (at your option)
\r
10 This library is distributed in the hope that it will be useful, but WITHOUT
\r
11 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
\r
12 FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
\r
15 You should have received a copy of the GNU Lesser General Public License along
\r
16 with this library; if not, write to the Free Software Foundation, Inc.,
\r
17 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
\r
20 *******************************************************************************/
\r
22 //******************************************************************************
\r
24 //******************************************************************************
\r
26 #include "../../common/header/plugin.h"
\r
27 #include "../../common/header/plugin-extra.h"
\r
28 #include "../../common/library/library.h"
\r
29 #include <wx/config.h>
\r
30 #include <wx/stdpaths.h>
\r
31 #include <wx/xml/xml.h>
\r
32 #include <wx/tokenzr.h>
\r
33 #include "cuiWrapper.h"
\r
35 //******************************************************************************
\r
37 //******************************************************************************
\r
42 wxString szExeFileAlt;
\r
43 wxString szListCommand;
\r
44 wxULongLong_t nLibIndex;
\r
49 wxString g_szCurrentArchive;
\r
50 wxArrayString g_asOutput;
\r
52 //******************************************************************************
\r
54 //******************************************************************************
\r
56 int myExecute(wxString szCommandLine, wxString * szOutput, wxString szCwd)
\r
59 wxString sz = ::wxGetCwd();
\r
60 ::wxSetWorkingDirectory(szCwd);
\r
61 FILE * fp = popen(szCommandLine.ToUTF8(), "r");
\r
62 ::wxSetWorkingDirectory(sz);
\r
65 ::wxLogError(L"Error :\n\nCommandLine:\n%s", szCommandLine.c_str());
\r
66 return TPI_ERROR_U_USE_LIBRARY;
\r
69 if (szOutput != NULL)
\r
74 memset(sz, 0, sizeof(sz));
\r
75 fread(sz, sizeof(char), sizeof(sz) - 1, fp);
\r
76 * szOutput += UTF82String(sz);
\r
78 // ::wxMessageBox(* szOutput);
\r
81 // 127を返した場合はコマンドが存在しない。
\r
82 int nErrorCode = pclose(fp);
\r
83 return (WIFEXITED(nErrorCode) && WEXITSTATUS(nErrorCode) != 127) ? TPI_ERROR_SUCCESS : TPI_ERROR_U_USE_LIBRARY;
\r
85 SECURITY_ATTRIBUTES sa;
\r
86 memset(& sa, 0, sizeof(SECURITY_ATTRIBUTES));
\r
87 sa.bInheritHandle = TRUE;
\r
88 sa.nLength = sizeof(SECURITY_ATTRIBUTES);
\r
89 HANDLE hRead, hWrite;
\r
90 if (! ::CreatePipe(& hRead, & hWrite, & sa, 0))
\r
92 return TPI_ERROR_U_USE_LIBRARY;
\r
95 memset(& si, 0, sizeof(STARTUPINFO));
\r
96 si.cb = sizeof(STARTUPINFO);
\r
97 si.dwFlags = STARTF_USESTDHANDLES;
\r
98 si.hStdOutput = hWrite;
\r
99 si.hStdError = hWrite;
\r
100 PROCESS_INFORMATION pi;
\r
101 if (! ::CreateProcess(NULL, szCommandLine.wchar_str(), NULL, NULL, TRUE, CREATE_NO_WINDOW, NULL, szCwd.IsEmpty() ? NULL : szCwd.wchar_str(), & si, & pi))
\r
103 return TPI_ERROR_U_USE_LIBRARY;
\r
105 ::CloseHandle(pi.hThread);
\r
107 if (szOutput != NULL)
\r
113 if (! ::PeekNamedPipe(hRead, NULL, 0, NULL, & dwSize, NULL))
\r
120 memset(sz, 0, sizeof(sz));
\r
121 ::ReadFile(hRead, & sz, sizeof(sz), & dwSize, NULL);
\r
122 * szOutput += UTF82String(sz);
\r
123 // ::MessageBoxA(NULL, sz, NULL, 0);
\r
125 else if (::WaitForSingleObject(pi.hProcess, 0) == WAIT_OBJECT_0)
\r
132 ::CloseHandle(pi.hProcess);
\r
133 ::CloseHandle(hRead);
\r
134 ::CloseHandle(hWrite);
\r
135 return TPI_ERROR_SUCCESS;
\r
139 PosInfo MakePosInfo(wxString szPrefix)
\r
142 g_LibInfo.node.GetAttribute(wxT("list-") + szPrefix + wxT("-s"), wxEmptyString).ToULongLong(& pi.nStart);
\r
143 g_LibInfo.node.GetAttribute(wxT("list-") + szPrefix + wxT("-c"), wxEmptyString).ToULongLong(& pi.nCount);
\r
144 g_LibInfo.node.GetAttribute(wxT("list-") + szPrefix + wxT("-l"), wxT("0")).ToULongLong(& pi.nLine);
\r
148 wxULongLong_t GetSize(PosInfo & pi, wxULongLong_t nCurrent, wxArrayString & as)
\r
150 if (pi.nStart == 0 && pi.nCount == 0)
\r
155 wxULongLong_t nTemp, nPos = nCurrent + pi.nLine;
\r
156 wxString sz = pi.nCount == 0 ? as[nPos].Mid(pi.nStart) : as[nPos].Mid(pi.nStart, pi.nCount);
\r
157 sz.ToULongLong(& nTemp);
\r
161 //******************************************************************************
\r
163 //******************************************************************************
\r
170 int __stdcall GetPluginInformation
\r
172 unsigned int _uInfoId,
\r
179 return TPI_ERROR_D_PARAMETER;
\r
183 case TPI_INFO_VERSION_MAJOR:
\r
184 case TPI_INFO_VERSION_MINOR:
\r
185 * (int *) _pPtr = 0;
\r
187 case TPI_INFO_VERSION_API:
\r
188 * (int *) _pPtr = 2;
\r
191 return TPI_ERROR_D_UNSUPPORTED;
\r
193 return TPI_ERROR_SUCCESS;
\r
196 int __stdcall GetFormatInformation(TPI_FORMATINFO * _fiInfo, bool _bFirst)
\r
198 static wxULongLong_t s_nFileId;
\r
200 wxXmlDocument config(wxPathOnly(p.GetExecutablePath()) + wxT("/lib/cuiWrapper.xml"));
\r
201 // 一気に先頭のライブラリの情報を取得。
\r
202 wxXmlNode * xmlLibrary = config.GetRoot()->GetChildren();
\r
211 for (wxULongLong_t i = 0; i < s_nFileId && xmlLibrary != NULL; i++)
\r
213 xmlLibrary = xmlLibrary->GetNext();
\r
217 if (xmlLibrary == NULL || xmlLibrary->GetName() != wxT("library"))
\r
220 return TPI_ERROR_S_ENDOFDATA;
\r
223 MakeFormatInfo(wxT("cuiWrapper"), _fiInfo, xmlLibrary, s_nFileId++);
\r
224 if (myExecute(xmlLibrary->GetAttribute(wxT("name"), wxEmptyString), NULL, wxEmptyString) != TPI_ERROR_SUCCESS)
\r
226 _fiInfo->eSupportedCommand = 0;
\r
229 return TPI_ERROR_SUCCESS;
\r
232 int __stdcall LoadPlugin
\r
234 const wxString & _szArcName,
\r
235 wxULongLong_t _llTypeId
\r
240 wxXmlDocument config(wxPathOnly(p.GetExecutablePath()) + wxT("/lib/cuiWrapper.xml"));
\r
241 if (! config.IsOk())
\r
243 return TPI_ERROR_UNDEFINED;
\r
245 // 一気に先頭のライブラリの情報を取得。
\r
246 wxXmlNode * xmlLibrary = config.GetRoot()->GetChildren();
\r
248 // 対象が存在するならば対応するライブラリを調査、
\r
249 // 対象が存在しないならば指示されたライブラリをロード。
\r
250 if (! ::wxFileExists(_szArcName))
\r
253 for (g_LibInfo.nLibIndex = 0; g_LibInfo.nLibIndex < _llTypeId && xmlLibrary != NULL; g_LibInfo.nLibIndex++)
\r
255 xmlLibrary = xmlLibrary->GetNext();
\r
257 if (xmlLibrary == NULL || xmlLibrary->GetName() != wxT("library"))
\r
260 return TPI_ERROR_UNDEFINED;
\r
262 g_LibInfo.szExeFile = xmlLibrary->GetAttribute(wxT("name"), wxEmptyString);
\r
263 g_LibInfo.szExeFileAlt = xmlLibrary->GetAttribute(wxT("name-alt"), wxEmptyString);
\r
264 g_LibInfo.node = * xmlLibrary;
\r
266 return TPI_ERROR_SUCCESS;
\r
269 // 無限ループに陥らないよう上限を設定。
\r
270 for (g_LibInfo.nLibIndex = 0; g_LibInfo.nLibIndex < 300 && xmlLibrary != NULL; g_LibInfo.nLibIndex++)
\r
273 g_LibInfo.szExeFile = xmlLibrary->GetAttribute(wxT("name"), wxEmptyString);
\r
274 g_LibInfo.szExeFileAlt = xmlLibrary->GetAttribute(wxT("name-alt"), wxEmptyString);
\r
275 g_LibInfo.node = * xmlLibrary;
\r
278 if (CheckArchive(_szArcName, NULL) == TPI_ERROR_SUCCESS)
\r
281 return TPI_ERROR_SUCCESS;
\r
284 xmlLibrary = xmlLibrary->GetNext();
\r
287 return TPI_ERROR_U_LOAD_LIBRARY;
\r
290 int __stdcall FreePlugin
\r
292 void * // _pReserved
\r
295 return TPI_ERROR_SUCCESS;
\r
298 int __stdcall CheckArchive
\r
300 const wxString & _szArcName,
\r
301 wxULongLong_t * _llFileCount
\r
304 wxFileName fnArchive(_szArcName);
\r
305 wxArrayString asExt = ::wxStringTokenize(g_LibInfo.node.GetAttribute(wxT("suffix"), wxEmptyString), wxT(";"));
\r
306 if (! g_LibInfo.node.HasAttribute(wxT("list")))
\r
308 return TPI_ERROR_ARC_UNSUPPORTED;
\r
311 for (size_t i = 0; i < asExt.GetCount(); i++)
\r
313 // .tar.XXXなど二重判定への対応。
\r
314 // if (asExt[i].IsSameAs(fnArchive.GetExt(), false))
\r
315 if (fnArchive.GetFullName().EndsWith(wxT('.') + asExt[i]))
\r
317 // 開いて確認。先行してデータを取得しておく。
\r
318 int nErrorCode = OpenArchive(_szArcName, NULL);
\r
321 if (_llFileCount != NULL)
\r
324 * _llFileCount = g_asOutput.GetCount();
\r
330 return TPI_ERROR_ARC_UNSUPPORTED;
\r
333 int __stdcall OpenArchive
\r
335 const wxString & _szArcName,
\r
339 if (g_szCurrentArchive != _szArcName)
\r
342 if (myExecute(g_LibInfo.szExeFile + wxT(" ") + MakeCommandLineSend(g_LibInfo.node.GetAttribute(wxT("list"), wxEmptyString), _szArcName, NULL, NULL, wxEmptyString), & szOutput, wxEmptyString) != TPI_ERROR_SUCCESS)
\r
344 return TPI_ERROR_U_USE_LIBRARY;
\r
346 g_szCurrentArchive = _szArcName;
\r
347 g_asOutput = ::wxStringTokenize(szOutput, wxT("\r\n"));
\r
348 g_asOutput.Shrink();
\r
350 if (_hArchive != NULL)
\r
352 * _hArchive = & g_asOutput;
\r
354 return g_asOutput.IsEmpty() ? TPI_ERROR_UNDEFINED : TPI_ERROR_SUCCESS;
\r
357 int __stdcall CloseArchive
\r
362 ((wxArrayString *) _hArchive)->Clear();
\r
363 return TPI_ERROR_SUCCESS;
\r
366 int __stdcall GetFileInformation
\r
369 TPI_FILEINFO * _fiInfo,
\r
373 static size_t s_nCurrentLine;
\r
374 static wxULongLong_t s_nFileId;
\r
375 if (_hArchive == NULL)
\r
377 return TPI_ERROR_UNDEFINED;
\r
379 wxArrayString asOutput = * (wxArrayString *) _hArchive;
\r
381 // XMLからの読み込みは初回に行う。
\r
382 static wxString szEndLine, szDateFormat;
\r
383 static wxULongLong_t nProcessPerLine;
\r
384 static PosInfo piFName, piPSize, piUSize, piDate;
\r
388 wxString szStartLine = g_LibInfo.node.GetAttribute(wxT("list-line-s"), wxEmptyString);
\r
389 if (! szStartLine.IsEmpty())
\r
392 s_nCurrentLine = asOutput.Index(szStartLine) + 1;
\r
393 if (s_nCurrentLine == wxNOT_FOUND + 1)
\r
396 return TPI_ERROR_ARC_UNSUPPORTED;
\r
401 g_LibInfo.node.GetAttribute(wxT("list-line-c"), wxT("1")).ToULongLong(& nProcessPerLine);
\r
402 szEndLine = g_LibInfo.node.GetAttribute(wxT("list-line-e"), szStartLine);
\r
403 szDateFormat = g_LibInfo.node.GetAttribute(wxT("list-date-f"), wxDefaultDateTimeFormat);
\r
404 piFName = MakePosInfo(wxT("fname"));
\r
405 piPSize = MakePosInfo(wxT("psize"));
\r
406 piUSize = MakePosInfo(wxT("usize"));
\r
407 piDate = MakePosInfo(wxT("date"));
\r
410 if (s_nCurrentLine >= asOutput.GetCount())
\r
412 // 空行で終わるとき以外はエラーとする。
\r
413 return szEndLine.IsEmpty() ? TPI_ERROR_S_ENDOFDATA : TPI_ERROR_ARC_UNSUPPORTED;
\r
417 if (asOutput[s_nCurrentLine] == szEndLine)
\r
419 return TPI_ERROR_S_ENDOFDATA;
\r
423 _fiInfo->szStoredName = piFName.nCount == 0 ? asOutput[s_nCurrentLine + piFName.nLine].Mid(piFName.nStart) : asOutput[s_nCurrentLine + piFName.nLine].Mid(piFName.nStart, piFName.nCount);
\r
424 _fiInfo->szStoredName.Trim();
\r
425 _fiInfo->fnFileName = wxFileName::wxFileName(_fiInfo->szStoredName);
\r
428 _fiInfo->nPackedSize = GetSize(piPSize, s_nCurrentLine, asOutput);
\r
429 _fiInfo->nUnpackedSize = GetSize(piUSize, s_nCurrentLine, asOutput);
\r
432 if (piDate.nStart != 0 || piDate.nCount != 0)
\r
434 _fiInfo->tmModified.ParseFormat(piDate.nCount == 0 ? asOutput[s_nCurrentLine + piDate.nLine].Mid(piDate.nStart) : asOutput[s_nCurrentLine + piDate.nLine].Mid(piDate.nStart, piDate.nCount), szDateFormat);
\r
438 _fiInfo->nFileId = s_nFileId++;
\r
439 s_nCurrentLine += nProcessPerLine;
\r
441 return TPI_ERROR_SUCCESS;
\r
444 int __stdcall GetArchiveInformation
\r
447 TPI_ARCHIVEINFO * _aiInfo
\r
451 MakeFormatInfo(wxT("cuiWrapper"), & _aiInfo->fiInfo, & g_LibInfo.node, 0);
\r
452 return TPI_ERROR_SUCCESS;
\r
455 int __stdcall Command
\r
457 wxULongLong_t _eCommand,
\r
458 TPI_SWITCHES * _swInfo,
\r
459 const wxString & _szArcName,
\r
460 const wxArrayString & _szFiles
\r
463 // xmlからコマンドラインを取得。
\r
464 wxString szPath, szCommandLine;
\r
467 if (! g_LibInfo.node.GetAttribute(
\r
468 _eCommand == TPI_COMMAND_CREATE ? wxT("create") :
\r
469 _eCommand == TPI_COMMAND_ADD ? wxT("add") :
\r
470 _eCommand == TPI_COMMAND_EXTRACT ? wxT("extract") :
\r
471 _eCommand == TPI_COMMAND_DELETE ? wxT("delete") :
\r
472 _eCommand == TPI_COMMAND_UPDATE ? wxT("update") :
\r
473 _eCommand == TPI_COMMAND_TEST ? wxT("test") :
\r
474 _eCommand == TPI_COMMAND_REPAIR ? wxT("repair") :
\r
475 _eCommand == TPI_COMMAND_MOVE ? wxT("move") :
\r
476 _eCommand == TPI_COMMAND_SFX ? wxT("sfx") :
\r
477 _eCommand == TPI_COMMAND_UNSFX ? wxT("unsfx") : wxEmptyString, & szCommandLine))
\r
479 g_LibInfo.node.GetAttribute(
\r
480 _eCommand == TPI_COMMAND_CREATE ? wxT("create-alt") :
\r
481 _eCommand == TPI_COMMAND_ADD ? wxT("add-alt") :
\r
482 _eCommand == TPI_COMMAND_EXTRACT ? wxT("extract-alt") :
\r
483 _eCommand == TPI_COMMAND_DELETE ? wxT("delete-alt") :
\r
484 _eCommand == TPI_COMMAND_UPDATE ? wxT("update-alt") :
\r
485 _eCommand == TPI_COMMAND_TEST ? wxT("test-alt") :
\r
486 _eCommand == TPI_COMMAND_REPAIR ? wxT("repair-alt") :
\r
487 _eCommand == TPI_COMMAND_MOVE ? wxT("move-alt") :
\r
488 _eCommand == TPI_COMMAND_SFX ? wxT("sfx-alt") :
\r
489 _eCommand == TPI_COMMAND_UNSFX ? wxT("unsfx-alt") : wxEmptyString, & szCommandLine);
\r
492 if (szCommandLine.IsEmpty())
\r
494 return TPI_ERROR_U_USE_LIBRARY;
\r
497 // コマンドライン・レスポンスファイル作成。
\r
499 szResponceFileName = MakeResponceFile(_szFiles, g_LibInfo.node.GetAttribute(wxT("quote-resp"), wxT("1")) == wxT("1")),
\r
500 szCommandLineSend = MakeCommandLineSend(szCommandLine, _szArcName, _swInfo, _szFiles, szResponceFileName);
\r
504 int nErrorCode = myExecute(g_LibInfo.szExeFile + wxT(" ") + szCommandLineSend, & szOutput, _swInfo->fnDestinationDirectory.GetFullPath());
\r
507 ::wxRemoveFile(szResponceFileName);
\r
509 if (nErrorCode != TPI_ERROR_SUCCESS)
\r
511 ::wxLogError(L"Error :\n%x\n\nCommandLine:\n%s\n\nOutput:\n%s", nErrorCode, szCommandLineSend.c_str(), szOutput.c_str());
\r
516 int __stdcall SetCallbackProc
\r
518 TPI_PROC _prArcProc
\r
522 if (_prArcProc == NULL)
\r
524 return TPI_ERROR_D_PARAMETER;
\r
526 g_prProc = * _prArcProc;
\r
528 return TPI_ERROR_SUCCESS;
\r