What isnt surprising is the fact the URI comparison sucks; but that the effect was...
[beagle.git] / Filters / FilterArchive.cs
blob1f7a096e88e7363781c0a969e9547b38fa77767c
1 //
2 // FilterArchive.cs
3 //
4 // Copyright (C) 2004-2006 Novell, Inc.
5 //
6 //
7 //
8 // Permission is hereby granted, free of charge, to any person obtaining a
9 // copy of this software and associated documentation files (the "Software"),
10 // to deal in the Software without restriction, including without limitation
11 // the rights to use, copy, modify, merge, publish, distribute, sublicense,
12 // and/or sell copies of the Software, and to permit persons to whom the
13 // Software is furnished to do so, subject to the following conditions:
15 // The above copyright notice and this permission notice shall be included in
16 // all copies or substantial portions of the Software.
18 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
23 // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
24 // DEALINGS IN THE SOFTWARE.
27 using System;
28 using System.Collections;
29 using System.IO;
31 using ICSharpCode.SharpZipLib;
32 using ICSharpCode.SharpZipLib.Zip;
33 using ICSharpCode.SharpZipLib.GZip;
34 using ICSharpCode.SharpZipLib.BZip2;
35 using ICSharpCode.SharpZipLib.Tar;
37 using Beagle;
38 using Beagle.Daemon;
39 using Beagle.Util;
41 namespace Beagle.Filters {
43 public class FilterArchive : Beagle.Daemon.Filter {
45 internal delegate ArchiveEntry GetNextEntry ();
47 private FileInfo file_info;
48 private Stream archive_stream;
49 private GetNextEntry get_next_entry;
51 // Fairly arbitrary number of files to limit
52 private const int MAX_CHILDREN = 30;
54 public FilterArchive ()
56 // 1: Store entry names as text content
57 SetVersion (1);
59 AddSupportedFlavor (FilterFlavor.NewFromMimeType ("application/zip"));
60 AddSupportedFlavor (FilterFlavor.NewFromMimeType ("application/x-bzip-compressed-tar"));
61 AddSupportedFlavor (FilterFlavor.NewFromMimeType ("application/x-compressed-tar"));
62 AddSupportedFlavor (FilterFlavor.NewFromMimeType ("application/x-tar"));
63 AddSupportedFlavor (FilterFlavor.NewFromMimeType ("application/x-tgz"));
64 AddSupportedFlavor (FilterFlavor.NewFromMimeType ("application/x-gzip"));
65 AddSupportedFlavor (FilterFlavor.NewFromMimeType ("application/x-bzip"));
68 protected override void DoOpen (FileInfo file_info)
70 this.file_info = file_info;
72 switch (MimeType) {
73 case "application/zip":
74 archive_stream = new ZipInputStream (Stream);
75 get_next_entry = GetNextEntryZip;
76 break;
78 case "application/x-bzip-compressed-tar":
79 archive_stream = new TarInputStream (new BZip2InputStream (Stream));
80 get_next_entry = GetNextEntryTar;
81 break;
83 case "application/x-compressed-tar":
84 case "application/x-tgz":
85 archive_stream = new TarInputStream (new GZipInputStream (Stream));
86 get_next_entry = GetNextEntryTar;
87 break;
89 case "application/x-tar":
90 archive_stream = new TarInputStream (Stream);
91 get_next_entry = GetNextEntryTar;
92 break;
94 case "application/x-gzip":
95 archive_stream = new GZipInputStream (Stream);
96 get_next_entry = GetNextEntrySingle;
97 break;
99 case "application/x-bzip":
100 archive_stream = new BZip2InputStream (Stream);
101 get_next_entry = GetNextEntrySingle;
102 break;
104 default:
105 throw new ArgumentException ("Invalid or unsupported mime type.");
110 protected override void DoPullProperties ()
112 // FIXME: Fetch the archive properties.
115 protected override void DoPullSetup ()
117 ArchiveEntry a_entry;
118 int count = 0;
120 while ((a_entry = this.get_next_entry ()) != null && count < MAX_CHILDREN) {
121 // FIXME: For nested archives, create uid:foo#bar
122 // instead of uid:foo#xxx#bar (avoid duplicates ?)
123 Indexable child = new Indexable (new Uri (Uri.ToString () + "#" + a_entry.Name, true));
125 child.CacheContent = true;
126 child.MimeType = a_entry.MimeType;
128 child.DisplayUri = new Uri (DisplayUri.ToString () + "#" + a_entry.Name, true);
129 child.ContentUri = UriFu.PathToFileUri (a_entry.TempFile);
130 child.DeleteContent = true;
132 child.AddProperty (Property.NewBool ("fixme:inside_archive", true));
134 child.AddProperty (Property.NewKeyword ("fixme:relativeuri", a_entry.Name));
136 if (a_entry.Comment != null)
137 child.AddProperty (Property.New ("fixme:comment", a_entry.Comment));
139 foreach (Property prop in Property.StandardFileProperties (Path.GetFileName (a_entry.Name), false))
140 child.AddProperty (prop);
142 AddChildIndexable (child);
144 // Store file names in the archive
145 AppendText (Path.GetFileName (a_entry.Name));
146 AppendText (" ");
148 ++count;
152 internal class ArchiveEntry {
153 public string Name;
154 public string MimeType;
155 public DateTime Modified;
156 public string Comment;
158 public string TempFile;
161 private static string StoreStreamInTempFile (Stream stream, DateTime mtime)
163 if (stream == null)
164 return null;
166 string filename = Path.GetTempFileName ();
167 FileStream file_stream = File.OpenWrite (filename);
169 Mono.Unix.Native.Syscall.chmod (filename, (Mono.Unix.Native.FilePermissions) 384); // 384 is 0600
171 BufferedStream buffered_stream = new BufferedStream (file_stream);
173 byte [] buffer = new byte [8192];
174 int read;
176 do {
177 read = stream.Read (buffer, 0, buffer.Length);
178 if (read > 0)
179 buffered_stream.Write (buffer, 0, read);
180 } while (read > 0);
182 buffered_stream.Close ();
184 File.SetLastWriteTimeUtc (filename, mtime);
186 Mono.Unix.Native.Syscall.chmod (filename, (Mono.Unix.Native.FilePermissions) 256); // 256 is 0400
188 return filename;
191 private ArchiveEntry GetNextEntryZip ()
193 ZipInputStream zip_stream = (ZipInputStream) archive_stream;
194 ZipEntry zip_entry;
196 do {
197 zip_entry = zip_stream.GetNextEntry ();
198 } while (zip_entry != null && zip_entry.IsDirectory);
200 // End of the entries.
201 if (zip_entry == null)
202 return null;
204 ArchiveEntry entry = new ArchiveEntry ();
205 entry.Name = zip_entry.Name;
206 entry.Modified = zip_entry.DateTime; // FIXME: Not sure zip_entry.DateTime is UTC.
207 entry.Comment = zip_entry.Comment;
209 entry.TempFile = StoreStreamInTempFile (archive_stream, entry.Modified);
210 entry.MimeType = XdgMime.GetMimeType (entry.TempFile);
212 return entry;
215 private ArchiveEntry GetNextEntryTar ()
217 TarInputStream tar_stream = (TarInputStream) archive_stream;
218 TarEntry tar_entry;
220 do {
221 tar_entry = tar_stream.GetNextEntry ();
222 } while (tar_entry != null && tar_entry.IsDirectory);
224 // End of the entries;
225 if (tar_entry == null)
226 return null;
228 ArchiveEntry entry = new ArchiveEntry ();
229 entry.Name = tar_entry.Name;
230 entry.Modified = tar_entry.ModTime;
232 entry.TempFile = StoreStreamInTempFile (archive_stream, entry.Modified);
233 entry.MimeType = XdgMime.GetMimeType (entry.TempFile);
235 return entry;
238 private bool handled_single = false;
240 private ArchiveEntry GetNextEntrySingle ()
242 if (handled_single)
243 return null;
245 ArchiveEntry entry = new ArchiveEntry ();
246 entry.Name = Path.GetFileNameWithoutExtension (this.file_info.Name);
247 entry.Modified = this.file_info.LastWriteTimeUtc;
249 entry.TempFile = StoreStreamInTempFile (archive_stream, entry.Modified);
250 entry.MimeType = XdgMime.GetMimeType (entry.TempFile);
252 handled_single = true;
254 return entry;