Home | History | Annotate | Download | only in android
      1 package org.robolectric.res.android;
      2 
      3 import static org.robolectric.res.android.Asset.toIntExact;
      4 import static org.robolectric.res.android.Errors.NAME_NOT_FOUND;
      5 import static org.robolectric.res.android.Errors.NO_ERROR;
      6 import static org.robolectric.res.android.Util.ALOGW;
      7 import static org.robolectric.res.android.Util.isTruthy;
      8 
      9 import java.io.IOException;
     10 import java.util.Enumeration;
     11 import java.util.zip.ZipEntry;
     12 import java.util.zip.ZipFile;
     13 
     14 public class ZipFileRO {
     15 
     16   static final int kCompressStored = 0;
     17   static final int kCompressDeflated = 8;
     18 
     19   final ZipArchiveHandle mHandle;
     20   final String mFileName;
     21 
     22   ZipFileRO(ZipArchiveHandle handle, String fileName) {
     23     this.mHandle = handle;
     24     this.mFileName = fileName;
     25   }
     26 
     27   static class ZipEntryRO {
     28         ZipEntry entry;
     29     String name;
     30     Object cookie;
     31 
     32     ZipEntryRO() {
     33     }
     34 
     35     //    ~ZipEntryRO() {
     36     @Override
     37     protected void finalize() {
     38 //      EndIteration(cookie);
     39     }
     40 
     41 //    private:
     42 //    ZipEntryRO(final ZipEntryRO& other);
     43 //    ZipEntryRO& operator=(final ZipEntryRO& other);
     44   };
     45 
     46 //  ~ZipFileRO() {
     47   @Override
     48   protected void finalize() {
     49     CloseArchive(mHandle);
     50 //    free(mFileName);
     51   }
     52 
     53   static int OpenArchive(String zipFileName, Ref<ZipArchiveHandle> mHandle) {
     54     try {
     55       mHandle.set(new ZipArchiveHandle(new ZipFile(zipFileName)));
     56       return NO_ERROR;
     57     } catch (IOException e) {
     58       return NAME_NOT_FOUND;
     59     }
     60   }
     61 
     62   private static void CloseArchive(ZipArchiveHandle mHandle) {
     63     throw new UnsupportedOperationException();
     64   }
     65 
     66   private static String ErrorCodeString(int error) {
     67     return "error " + error;
     68   }
     69 
     70   static int FindEntry(ZipArchiveHandle mHandle, String name, Ref<ZipEntry> zipEntryRef) {
     71     ZipEntry entry = mHandle.zipFile.getEntry(name);
     72     zipEntryRef.set(entry);
     73     if (entry == null) {
     74       return NAME_NOT_FOUND;
     75     }
     76     return NO_ERROR;
     77   }
     78 
     79   /*
     80  * Open the specified file read-only.  We memory-map the entire thing and
     81  * close the file before returning.
     82  */
     83 /* static */
     84   static ZipFileRO open(final String zipFileName)
     85   {
     86     final Ref<ZipArchiveHandle> handle = new Ref<>(null);
     87     final int error = OpenArchive(zipFileName, handle);
     88     if (isTruthy(error)) {
     89       ALOGW("Error opening archive %s: %s", zipFileName, ErrorCodeString(error));
     90       CloseArchive(handle.get());
     91       return null;
     92     }
     93 
     94     return new ZipFileRO(handle.get(), zipFileName);
     95   }
     96 
     97   // /* static */ ZipFileRO* ZipFileRO::openFd(int fd, String debugFileName,
     98   //     boolean assume_ownership)
     99   // {
    100   //   ZipArchiveHandle handle;
    101   //   int error = OpenArchiveFd(fd, debugFileName, &handle, assume_ownership);
    102   //   if (error) {
    103   //     ALOGW("Error opening archive fd %d %s: %s", fd, debugFileName, ErrorCodeString(error));
    104   //     CloseArchive(handle);
    105   //     return NULL;
    106   //   }
    107   //
    108   //   return new ZipFileRO(handle, strdup(debugFileName));
    109   // }
    110 
    111   org.robolectric.res.android.ZipFileRO.ZipEntryRO findEntryByName(final String entryName)
    112   {
    113     ZipEntryRO data = new ZipEntryRO();
    114 
    115     data.name = String(entryName);
    116 
    117     final Ref<ZipEntry> zipEntryRef = new Ref<>(data.entry);
    118     final int error = FindEntry(mHandle, data.name, zipEntryRef);
    119     if (isTruthy(error)) {
    120       return null;
    121     }
    122 
    123     data.entry = zipEntryRef.get();
    124     return data;
    125   }
    126 
    127   /*
    128    * Get the useful fields from the zip entry.
    129    *
    130    * Returns "false" if the offsets to the fields or the contents of the fields
    131    * appear to be bogus.
    132    */
    133   boolean getEntryInfo(org.robolectric.res.android.ZipFileRO.ZipEntryRO entry, Ref<Short> pMethod,
    134       final Ref<Long> pUncompLen, Ref<Long> pCompLen, Ref<Long> pOffset,
    135       final Ref<Long> pModWhen, Ref<Long> pCrc32)
    136   {
    137     final ZipEntryRO zipEntry = /*reinterpret_cast<ZipEntryRO*>*/(entry);
    138     final ZipEntry ze = zipEntry.entry;
    139 
    140     if (pMethod != null) {
    141       pMethod.set((short) ze.getMethod());
    142     }
    143     if (pUncompLen != null) {
    144         pUncompLen.set(ze.getSize()); // uncompressed_length
    145     }
    146     if (pCompLen != null) {
    147         pCompLen.set(ze.getCompressedSize());
    148     }
    149     if (pOffset != null) {
    150       throw new UnsupportedOperationException("Figure out offset");
    151       //        pOffset = ze.offset;
    152     }
    153     if (pModWhen != null) {
    154         // todo pModWhen.set(ze.getLastModifiedTime().toMillis());
    155     }
    156     if (pCrc32 != null) {
    157       pCrc32.set(ze.getCrc());
    158     }
    159 
    160     return true;
    161   }
    162 
    163   boolean startIteration(Ref<Enumeration<? extends ZipEntry>> cookie) {
    164     return startIteration(cookie, null, null);
    165   }
    166 
    167   boolean startIteration(/* void** */ Ref<Enumeration<? extends ZipEntry>> cookie, final String prefix, final String suffix)
    168   {
    169     cookie.set(this.mHandle.zipFile.entries());
    170 //    ZipEntryRO* ze = new ZipEntryRO;
    171 //    String pe(prefix ? prefix : "");
    172 //    String se(suffix ? suffix : "");
    173 //    int error = StartIteration(mHandle, &(ze.cookie),
    174 //    prefix ? &pe : null,
    175 //      suffix ? &se : null);
    176 //    if (error) {
    177 //      ALOGW("Could not start iteration over %s: %s", mFileName, ErrorCodeString(error));
    178 //      delete ze;
    179 //      return false;
    180 //    }
    181 //
    182 //    *cookie = ze;
    183     return true;
    184   }
    185 
    186   org.robolectric.res.android.ZipFileRO.ZipEntryRO nextEntry(/*void* */ Enumeration<? extends ZipEntry> cookie)
    187   {
    188     if (!cookie.hasMoreElements()) {
    189       return null;
    190     }
    191     ZipEntryRO zipEntryRO = new ZipEntryRO();
    192     zipEntryRO.entry = cookie.nextElement();
    193     return zipEntryRO;
    194 //    ZipEntryRO ze = /*reinterpret_cast<ZipEntryRO*>*/(ZipEntryRO) cookie;
    195 //    int error = Next(ze.cookie, &(ze.entry), &(ze.name));
    196 //    if (error) {
    197 //      if (error != -1) {
    198 //        ALOGW("Error iteration over %s: %s", mFileName, ErrorCodeString(error));
    199 //      }
    200 //      return null;
    201 //    }
    202 //
    203 //    return &(ze.entry);
    204   }
    205 
    206   void endIteration(/*void**/ Object cookie)
    207   {
    208 //    delete reinterpret_cast<ZipEntryRO*>(cookie);
    209   }
    210 
    211   void releaseEntry(org.robolectric.res.android.ZipFileRO.ZipEntryRO entry)
    212   {
    213 //    delete reinterpret_cast<ZipEntryRO*>(entry);
    214   }
    215 
    216   /*
    217    * Copy the entry's filename to the buffer.
    218    */
    219   int getEntryFileName(org.robolectric.res.android.ZipFileRO.ZipEntryRO entry, Ref<String> buffer)
    220   {
    221     buffer.set(entry.entry.getName());
    222 
    223 //    final ZipEntryRO* zipEntry = reinterpret_cast<ZipEntryRO*>(entry);
    224 //    final uint16_t requiredSize = zipEntry.name.name_length + 1;
    225 //
    226 //    if (bufLen < requiredSize) {
    227 //      ALOGW("Buffer too short, requires %d bytes for entry name", requiredSize);
    228 //      return requiredSize;
    229 //    }
    230 //
    231 //    memcpy(buffer, zipEntry.name.name, requiredSize - 1);
    232 //    buffer[requiredSize - 1] = '\0';
    233 //
    234     return 0;
    235   }
    236 
    237 /*
    238  * Create a new FileMap object that spans the data in "entry".
    239  */
    240   /*FileMap*/ ZipFileRO(org.robolectric.res.android.ZipFileRO.ZipEntryRO entry)
    241   {
    242     throw new UnsupportedOperationException("Implememnt me");
    243 
    244 //    final ZipEntryRO *zipEntry = reinterpret_cast<ZipEntryRO*>(entry);
    245 //    final ZipEntry& ze = zipEntry.entry;
    246 //    int fd = GetFileDescriptor(mHandle);
    247 //    size_t actualLen = 0;
    248 //
    249 //    if (ze.method == kCompressStored) {
    250 //      actualLen = ze.uncompressed_length;
    251 //    } else {
    252 //      actualLen = ze.compressed_length;
    253 //    }
    254 //
    255 //    FileMap* newMap = new FileMap();
    256 //    if (!newMap.create(mFileName, fd, ze.offset, actualLen, true)) {
    257 //      delete newMap;
    258 //      return null;
    259 //    }
    260 //
    261 //    return newMap;
    262   }
    263 
    264   /*
    265  * Create a new FileMap object that spans the data in "entry".
    266  */
    267   FileMap createEntryFileMap(ZipEntryRO entry)
    268   {
    269     // final _ZipEntryRO *zipEntry = reinterpret_cast<_ZipEntryRO*>(entry);
    270     // const ZipEntry& ze = zipEntry->entry;
    271     ZipEntry ze = entry.entry;
    272     // int fd = GetFileDescriptor(mHandle);
    273     int fd = -1;
    274     int actualLen = 0;
    275 
    276     if (ze.getMethod() == kCompressStored) {
    277       actualLen = toIntExact(ze.getSize());
    278     } else {
    279       actualLen = toIntExact(ze.getCompressedSize());
    280     }
    281 
    282     FileMap newMap = new FileMap();
    283     if (!newMap.createFromZip(mFileName, mHandle.zipFile, entry.entry, actualLen, true)) {
    284       // delete newMap;
    285       return null;
    286     }
    287 
    288     return newMap;
    289   }
    290 
    291   /*
    292    * Uncompress an entry, in its entirety, into the provided output buffer.
    293    *
    294    * This doesn't verify the data's CRC, which might be useful for
    295    * uncompressed data.  The caller should be able to manage it.
    296    */
    297   boolean uncompressEntry(org.robolectric.res.android.ZipFileRO.ZipEntryRO entry, Object buffer, int size)
    298   {
    299     throw new UnsupportedOperationException("Implememnt me");
    300 //    ZipEntryRO *zipEntry = reinterpret_cast<ZipEntryRO*>(entry);
    301 //    final int error = ExtractToMemory(mHandle, &(zipEntry.entry),
    302 //    (uint8_t*) buffer, size);
    303 //    if (error) {
    304 //      ALOGW("ExtractToMemory failed with %s", ErrorCodeString(error));
    305 //      return false;
    306 //    }
    307 //
    308 //    return true;
    309   }
    310 
    311   /*
    312    * Uncompress an entry, in its entirety, to an open file descriptor.
    313    *
    314    * This doesn't verify the data's CRC, but probably should.
    315    */
    316   boolean uncompressEntry(org.robolectric.res.android.ZipFileRO.ZipEntryRO entry, int fd)
    317   {
    318     throw new UnsupportedOperationException("Implememnt me");
    319 //    ZipEntryRO *zipEntry = reinterpret_cast<ZipEntryRO*>(entry);
    320 //    final int error = ExtractEntryToFile(mHandle, &(zipEntry.entry), fd);
    321 //    if (error) {
    322 //      ALOGW("ExtractToMemory failed with %s", ErrorCodeString(error));
    323 //      return false;
    324 //    }
    325 //
    326 //    return true;
    327   }
    328 
    329   static String String(String string) {
    330     return string;
    331   }
    332 }
    333