]>
Commit | Line | Data |
---|---|---|
1 | ///////////////////////////////////////////////////////////////////////////// | |
2 | // Name: tarstrm.h | |
3 | // Purpose: Streams for Tar files | |
4 | // Author: Mike Wetherell | |
5 | // RCS-ID: $Id$ | |
6 | // Copyright: (c) 2004 Mike Wetherell | |
7 | // Licence: wxWindows licence | |
8 | ///////////////////////////////////////////////////////////////////////////// | |
9 | ||
10 | #ifndef _WX_WXTARSTREAM_H__ | |
11 | #define _WX_WXTARSTREAM_H__ | |
12 | ||
13 | #include "wx/defs.h" | |
14 | ||
15 | #if wxUSE_TARSTREAM | |
16 | ||
17 | #include "wx/archive.h" | |
18 | #include "wx/hashmap.h" | |
19 | ||
20 | ||
21 | ///////////////////////////////////////////////////////////////////////////// | |
22 | // Constants | |
23 | ||
24 | // TypeFlag values | |
25 | enum { | |
26 | wxTAR_REGTYPE = '0', // regular file | |
27 | wxTAR_LNKTYPE = '1', // hard link | |
28 | wxTAR_SYMTYPE = '2', // symbolic link | |
29 | wxTAR_CHRTYPE = '3', // character special | |
30 | wxTAR_BLKTYPE = '4', // block special | |
31 | wxTAR_DIRTYPE = '5', // directory | |
32 | wxTAR_FIFOTYPE = '6', // named pipe | |
33 | wxTAR_CONTTYPE = '7' // contiguous file | |
34 | }; | |
35 | ||
36 | // Archive Formats (use wxTAR_PAX, it's backward compatible) | |
37 | enum wxTarFormat | |
38 | { | |
39 | wxTAR_USTAR, // POSIX.1-1990 tar format | |
40 | wxTAR_PAX // POSIX.1-2001 tar format | |
41 | }; | |
42 | ||
43 | ||
44 | ///////////////////////////////////////////////////////////////////////////// | |
45 | // wxTarNotifier | |
46 | ||
47 | class WXDLLIMPEXP_BASE wxTarNotifier | |
48 | { | |
49 | public: | |
50 | virtual ~wxTarNotifier() { } | |
51 | ||
52 | virtual void OnEntryUpdated(class wxTarEntry& entry) = 0; | |
53 | }; | |
54 | ||
55 | ||
56 | ///////////////////////////////////////////////////////////////////////////// | |
57 | // Tar Entry - hold the meta data for a file in the tar | |
58 | ||
59 | class WXDLLIMPEXP_BASE wxTarEntry : public wxArchiveEntry | |
60 | { | |
61 | public: | |
62 | wxTarEntry(const wxString& name = wxEmptyString, | |
63 | const wxDateTime& dt = wxDateTime::Now(), | |
64 | wxFileOffset size = wxInvalidOffset); | |
65 | virtual ~wxTarEntry(); | |
66 | ||
67 | wxTarEntry(const wxTarEntry& entry); | |
68 | wxTarEntry& operator=(const wxTarEntry& entry); | |
69 | ||
70 | // Get accessors | |
71 | wxString GetName(wxPathFormat format = wxPATH_NATIVE) const; | |
72 | wxString GetInternalName() const { return m_Name; } | |
73 | wxPathFormat GetInternalFormat() const { return wxPATH_UNIX; } | |
74 | int GetMode() const; | |
75 | int GetUserId() const { return m_UserId; } | |
76 | int GetGroupId() const { return m_GroupId; } | |
77 | wxFileOffset GetSize() const { return m_Size; } | |
78 | wxFileOffset GetOffset() const { return m_Offset; } | |
79 | wxDateTime GetDateTime() const { return m_ModifyTime; } | |
80 | wxDateTime GetAccessTime() const { return m_AccessTime; } | |
81 | wxDateTime GetCreateTime() const { return m_CreateTime; } | |
82 | int GetTypeFlag() const { return m_TypeFlag; } | |
83 | wxString GetLinkName() const { return m_LinkName; } | |
84 | wxString GetUserName() const { return m_UserName; } | |
85 | wxString GetGroupName() const { return m_GroupName; } | |
86 | int GetDevMajor() const { return m_DevMajor; } | |
87 | int GetDevMinor() const { return m_DevMinor; } | |
88 | ||
89 | // is accessors | |
90 | bool IsDir() const; | |
91 | bool IsReadOnly() const { return !(m_Mode & 0222); } | |
92 | ||
93 | // set accessors | |
94 | void SetName(const wxString& name, wxPathFormat format = wxPATH_NATIVE); | |
95 | void SetUserId(int id) { m_UserId = id; } | |
96 | void SetGroupId(int id) { m_GroupId = id; } | |
97 | void SetMode(int mode); | |
98 | void SetSize(wxFileOffset size) { m_Size = size; } | |
99 | void SetDateTime(const wxDateTime& dt) { m_ModifyTime = dt; } | |
100 | void SetAccessTime(const wxDateTime& dt) { m_AccessTime = dt; } | |
101 | void SetCreateTime(const wxDateTime& dt) { m_CreateTime = dt; } | |
102 | void SetTypeFlag(int type) { m_TypeFlag = type; } | |
103 | void SetLinkName(const wxString& link) { m_LinkName = link; } | |
104 | void SetUserName(const wxString& user) { m_UserName = user; } | |
105 | void SetGroupName(const wxString& group) { m_GroupName = group; } | |
106 | void SetDevMajor(int dev) { m_DevMajor = dev; } | |
107 | void SetDevMinor(int dev) { m_DevMinor = dev; } | |
108 | ||
109 | // set is accessors | |
110 | void SetIsDir(bool isDir = true); | |
111 | void SetIsReadOnly(bool isReadOnly = true); | |
112 | ||
113 | static wxString GetInternalName(const wxString& name, | |
114 | wxPathFormat format = wxPATH_NATIVE, | |
115 | bool *pIsDir = NULL); | |
116 | ||
117 | wxTarEntry *Clone() const { return new wxTarEntry(*this); } | |
118 | ||
119 | void SetNotifier(wxTarNotifier& WXUNUSED(notifier)) { } | |
120 | ||
121 | private: | |
122 | void SetOffset(wxFileOffset offset) { m_Offset = offset; } | |
123 | ||
124 | virtual wxArchiveEntry* DoClone() const { return Clone(); } | |
125 | ||
126 | wxString m_Name; | |
127 | int m_Mode; | |
128 | bool m_IsModeSet; | |
129 | int m_UserId; | |
130 | int m_GroupId; | |
131 | wxFileOffset m_Size; | |
132 | wxFileOffset m_Offset; | |
133 | wxDateTime m_ModifyTime; | |
134 | wxDateTime m_AccessTime; | |
135 | wxDateTime m_CreateTime; | |
136 | wxChar m_TypeFlag; | |
137 | wxString m_LinkName; | |
138 | wxString m_UserName; | |
139 | wxString m_GroupName; | |
140 | int m_DevMajor; | |
141 | int m_DevMinor; | |
142 | ||
143 | friend class wxTarInputStream; | |
144 | ||
145 | DECLARE_DYNAMIC_CLASS(wxTarEntry) | |
146 | }; | |
147 | ||
148 | ||
149 | ///////////////////////////////////////////////////////////////////////////// | |
150 | // wxTarInputStream | |
151 | ||
152 | WX_DECLARE_STRING_HASH_MAP(wxString, wxTarHeaderRecords); | |
153 | ||
154 | class WXDLLIMPEXP_BASE wxTarInputStream : public wxArchiveInputStream | |
155 | { | |
156 | public: | |
157 | typedef wxTarEntry entry_type; | |
158 | ||
159 | wxTarInputStream(wxInputStream& stream, wxMBConv& conv = wxConvLocal); | |
160 | wxTarInputStream(wxInputStream *stream, wxMBConv& conv = wxConvLocal); | |
161 | virtual ~wxTarInputStream(); | |
162 | ||
163 | bool OpenEntry(wxTarEntry& entry); | |
164 | bool CloseEntry(); | |
165 | ||
166 | wxTarEntry *GetNextEntry(); | |
167 | ||
168 | wxFileOffset GetLength() const { return m_size; } | |
169 | bool IsSeekable() const { return m_parent_i_stream->IsSeekable(); } | |
170 | ||
171 | protected: | |
172 | size_t OnSysRead(void *buffer, size_t size); | |
173 | wxFileOffset OnSysTell() const { return m_pos; } | |
174 | wxFileOffset OnSysSeek(wxFileOffset seek, wxSeekMode mode); | |
175 | ||
176 | private: | |
177 | void Init(); | |
178 | ||
179 | wxArchiveEntry *DoGetNextEntry() { return GetNextEntry(); } | |
180 | bool OpenEntry(wxArchiveEntry& entry); | |
181 | bool IsOpened() const { return m_pos != wxInvalidOffset; } | |
182 | ||
183 | wxStreamError ReadHeaders(); | |
184 | bool ReadExtendedHeader(wxTarHeaderRecords*& recs); | |
185 | ||
186 | wxString GetExtendedHeader(const wxString& key) const; | |
187 | wxString GetHeaderPath() const; | |
188 | wxFileOffset GetHeaderNumber(int id) const; | |
189 | wxString GetHeaderString(int id) const; | |
190 | wxDateTime GetHeaderDate(const wxString& key) const; | |
191 | ||
192 | wxFileOffset m_pos; // position within the current entry | |
193 | wxFileOffset m_offset; // offset to the start of the entry's data | |
194 | wxFileOffset m_size; // size of the current entry's data | |
195 | ||
196 | int m_sumType; | |
197 | int m_tarType; | |
198 | class wxTarHeaderBlock *m_hdr; | |
199 | wxTarHeaderRecords *m_HeaderRecs; | |
200 | wxTarHeaderRecords *m_GlobalHeaderRecs; | |
201 | ||
202 | DECLARE_NO_COPY_CLASS(wxTarInputStream) | |
203 | }; | |
204 | ||
205 | ||
206 | ///////////////////////////////////////////////////////////////////////////// | |
207 | // wxTarOutputStream | |
208 | ||
209 | class WXDLLIMPEXP_BASE wxTarOutputStream : public wxArchiveOutputStream | |
210 | { | |
211 | public: | |
212 | wxTarOutputStream(wxOutputStream& stream, | |
213 | wxTarFormat format = wxTAR_PAX, | |
214 | wxMBConv& conv = wxConvLocal); | |
215 | wxTarOutputStream(wxOutputStream *stream, | |
216 | wxTarFormat format = wxTAR_PAX, | |
217 | wxMBConv& conv = wxConvLocal); | |
218 | virtual ~wxTarOutputStream(); | |
219 | ||
220 | bool PutNextEntry(wxTarEntry *entry); | |
221 | ||
222 | bool PutNextEntry(const wxString& name, | |
223 | const wxDateTime& dt = wxDateTime::Now(), | |
224 | wxFileOffset size = wxInvalidOffset); | |
225 | ||
226 | bool PutNextDirEntry(const wxString& name, | |
227 | const wxDateTime& dt = wxDateTime::Now()); | |
228 | ||
229 | bool CopyEntry(wxTarEntry *entry, wxTarInputStream& inputStream); | |
230 | bool CopyArchiveMetaData(wxTarInputStream& WXUNUSED(s)) { return true; } | |
231 | ||
232 | void Sync(); | |
233 | bool CloseEntry(); | |
234 | bool Close(); | |
235 | ||
236 | bool IsSeekable() const { return m_parent_o_stream->IsSeekable(); } | |
237 | ||
238 | void SetBlockingFactor(int factor) { m_BlockingFactor = factor; } | |
239 | int GetBlockingFactor() const { return m_BlockingFactor; } | |
240 | ||
241 | protected: | |
242 | size_t OnSysWrite(const void *buffer, size_t size); | |
243 | wxFileOffset OnSysTell() const { return m_pos; } | |
244 | wxFileOffset OnSysSeek(wxFileOffset pos, wxSeekMode mode); | |
245 | ||
246 | private: | |
247 | void Init(wxTarFormat format); | |
248 | ||
249 | bool PutNextEntry(wxArchiveEntry *entry); | |
250 | bool CopyEntry(wxArchiveEntry *entry, wxArchiveInputStream& stream); | |
251 | bool CopyArchiveMetaData(wxArchiveInputStream& WXUNUSED(s)) { return true; } | |
252 | bool IsOpened() const { return m_pos != wxInvalidOffset; } | |
253 | ||
254 | bool WriteHeaders(wxTarEntry& entry); | |
255 | bool ModifyHeader(); | |
256 | wxString PaxHeaderPath(const wxString& format, const wxString& path); | |
257 | ||
258 | void SetExtendedHeader(const wxString& key, const wxString& value); | |
259 | void SetHeaderPath(const wxString& name); | |
260 | bool SetHeaderNumber(int id, wxFileOffset n); | |
261 | void SetHeaderString(int id, const wxString& str); | |
262 | void SetHeaderDate(const wxString& key, const wxDateTime& datetime); | |
263 | ||
264 | wxFileOffset m_pos; // position within the current entry | |
265 | wxFileOffset m_maxpos; // max pos written | |
266 | wxFileOffset m_size; // expected entry size | |
267 | ||
268 | wxFileOffset m_headpos; // offset within the file to the entry's header | |
269 | wxFileOffset m_datapos; // offset within the file to the entry's data | |
270 | ||
271 | wxFileOffset m_tarstart;// offset within the file to the tar | |
272 | wxFileOffset m_tarsize; // size of tar so far | |
273 | ||
274 | bool m_pax; | |
275 | int m_BlockingFactor; | |
276 | wxUint32 m_chksum; | |
277 | bool m_large; | |
278 | class wxTarHeaderBlock *m_hdr; | |
279 | class wxTarHeaderBlock *m_hdr2; | |
280 | char *m_extendedHdr; | |
281 | size_t m_extendedSize; | |
282 | wxString m_badfit; | |
283 | ||
284 | DECLARE_NO_COPY_CLASS(wxTarOutputStream) | |
285 | }; | |
286 | ||
287 | ||
288 | ///////////////////////////////////////////////////////////////////////////// | |
289 | // Iterators | |
290 | ||
291 | #if wxUSE_STL || defined WX_TEST_ARCHIVE_ITERATOR | |
292 | typedef wxArchiveIterator<wxTarInputStream> wxTarIter; | |
293 | typedef wxArchiveIterator<wxTarInputStream, | |
294 | std::pair<wxString, wxTarEntry*> > wxTarPairIter; | |
295 | #endif | |
296 | ||
297 | ||
298 | ///////////////////////////////////////////////////////////////////////////// | |
299 | // wxTarClassFactory | |
300 | ||
301 | class WXDLLIMPEXP_BASE wxTarClassFactory : public wxArchiveClassFactory | |
302 | { | |
303 | public: | |
304 | typedef wxTarEntry entry_type; | |
305 | typedef wxTarInputStream instream_type; | |
306 | typedef wxTarOutputStream outstream_type; | |
307 | typedef wxTarNotifier notifier_type; | |
308 | #if wxUSE_STL || defined WX_TEST_ARCHIVE_ITERATOR | |
309 | typedef wxTarIter iter_type; | |
310 | typedef wxTarPairIter pairiter_type; | |
311 | #endif | |
312 | ||
313 | wxTarClassFactory(); | |
314 | ||
315 | wxTarEntry *NewEntry() const | |
316 | { return new wxTarEntry; } | |
317 | wxTarInputStream *NewStream(wxInputStream& stream) const | |
318 | { return new wxTarInputStream(stream, GetConv()); } | |
319 | wxTarOutputStream *NewStream(wxOutputStream& stream) const | |
320 | { return new wxTarOutputStream(stream, wxTAR_PAX, GetConv()); } | |
321 | wxTarInputStream *NewStream(wxInputStream *stream) const | |
322 | { return new wxTarInputStream(stream, GetConv()); } | |
323 | wxTarOutputStream *NewStream(wxOutputStream *stream) const | |
324 | { return new wxTarOutputStream(stream, wxTAR_PAX, GetConv()); } | |
325 | ||
326 | wxString GetInternalName(const wxString& name, | |
327 | wxPathFormat format = wxPATH_NATIVE) const | |
328 | { return wxTarEntry::GetInternalName(name, format); } | |
329 | ||
330 | const wxChar * const *GetProtocols(wxStreamProtocolType type | |
331 | = wxSTREAM_PROTOCOL) const; | |
332 | ||
333 | protected: | |
334 | wxArchiveEntry *DoNewEntry() const | |
335 | { return NewEntry(); } | |
336 | wxArchiveInputStream *DoNewStream(wxInputStream& stream) const | |
337 | { return NewStream(stream); } | |
338 | wxArchiveOutputStream *DoNewStream(wxOutputStream& stream) const | |
339 | { return NewStream(stream); } | |
340 | wxArchiveInputStream *DoNewStream(wxInputStream *stream) const | |
341 | { return NewStream(stream); } | |
342 | wxArchiveOutputStream *DoNewStream(wxOutputStream *stream) const | |
343 | { return NewStream(stream); } | |
344 | ||
345 | private: | |
346 | DECLARE_DYNAMIC_CLASS(wxTarClassFactory) | |
347 | }; | |
348 | ||
349 | ||
350 | #endif // wxUSE_TARSTREAM | |
351 | ||
352 | #endif // _WX_WXTARSTREAM_H__ |