150 lines
3.3 KiB
C
150 lines
3.3 KiB
C
|
#include "mupdf/fitz.h"
|
||
|
|
||
|
/*
|
||
|
Opens an archive entry as a stream.
|
||
|
|
||
|
name: Entry name to look for, this must be an exact match to
|
||
|
the entry name in the archive.
|
||
|
*/
|
||
|
fz_stream *
|
||
|
fz_open_archive_entry(fz_context *ctx, fz_archive *arch, const char *name)
|
||
|
{
|
||
|
if (!arch->open_entry)
|
||
|
fz_throw(ctx, FZ_ERROR_GENERIC, "cannot open archive entry");
|
||
|
return arch->open_entry(ctx, arch, name);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Reads all bytes in an archive entry
|
||
|
into a buffer.
|
||
|
|
||
|
name: Entry name to look for, this must be an exact match to
|
||
|
the entry name in the archive.
|
||
|
*/
|
||
|
fz_buffer *
|
||
|
fz_read_archive_entry(fz_context *ctx, fz_archive *arch, const char *name)
|
||
|
{
|
||
|
if (!arch->read_entry)
|
||
|
fz_throw(ctx, FZ_ERROR_GENERIC, "cannot read archive entry");
|
||
|
return arch->read_entry(ctx, arch, name);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Check if entry by given name exists.
|
||
|
|
||
|
If named entry does not exist 0 will be returned, if it does
|
||
|
exist 1 is returned.
|
||
|
|
||
|
name: Entry name to look for, this must be an exact match to
|
||
|
the entry name in the archive.
|
||
|
*/
|
||
|
int
|
||
|
fz_has_archive_entry(fz_context *ctx, fz_archive *arch, const char *name)
|
||
|
{
|
||
|
if (!arch->has_entry)
|
||
|
fz_throw(ctx, FZ_ERROR_GENERIC, "cannot check if archive has entry");
|
||
|
return arch->has_entry(ctx, arch, name);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Get listed name of entry position idx.
|
||
|
|
||
|
idx: Must be a value >= 0 < return value from
|
||
|
fz_count_archive_entries. If not in range NULL will be
|
||
|
returned.
|
||
|
*/
|
||
|
const char *
|
||
|
fz_list_archive_entry(fz_context *ctx, fz_archive *arch, int idx)
|
||
|
{
|
||
|
if (!arch->list_entry)
|
||
|
fz_throw(ctx, FZ_ERROR_GENERIC, "cannot list archive entries");
|
||
|
return arch->list_entry(ctx, arch, idx);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Number of entries in archive.
|
||
|
|
||
|
Will always return a value >= 0.
|
||
|
*/
|
||
|
int
|
||
|
fz_count_archive_entries(fz_context *ctx, fz_archive *arch)
|
||
|
{
|
||
|
if (!arch->count_entries)
|
||
|
fz_throw(ctx, FZ_ERROR_GENERIC, "cannot count archive entries");
|
||
|
return arch->count_entries(ctx, arch);
|
||
|
}
|
||
|
|
||
|
const char *
|
||
|
fz_archive_format(fz_context *ctx, fz_archive *arch)
|
||
|
{
|
||
|
return arch->format;
|
||
|
}
|
||
|
|
||
|
fz_archive *
|
||
|
fz_new_archive_of_size(fz_context *ctx, fz_stream *file, int size)
|
||
|
{
|
||
|
fz_archive *arch;
|
||
|
arch = Memento_label(fz_calloc(ctx, 1, size), "fz_archive");
|
||
|
arch->file = fz_keep_stream(ctx, file);
|
||
|
return arch;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Open zip or tar archive stream.
|
||
|
|
||
|
Open an archive using a seekable stream object rather than
|
||
|
opening a file or directory on disk.
|
||
|
*/
|
||
|
fz_archive *
|
||
|
fz_open_archive_with_stream(fz_context *ctx, fz_stream *file)
|
||
|
{
|
||
|
fz_archive *arch = NULL;
|
||
|
|
||
|
if (fz_is_zip_archive(ctx, file))
|
||
|
arch = fz_open_zip_archive_with_stream(ctx, file);
|
||
|
else if (fz_is_tar_archive(ctx, file))
|
||
|
arch = fz_open_tar_archive_with_stream(ctx, file);
|
||
|
else
|
||
|
fz_throw(ctx, FZ_ERROR_GENERIC, "cannot recognize archive");
|
||
|
|
||
|
return arch;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Open a zip or tar archive
|
||
|
|
||
|
Open a file and identify its archive type based on the archive
|
||
|
signature contained inside.
|
||
|
|
||
|
filename: a path to a file as it would be given to open(2).
|
||
|
*/
|
||
|
fz_archive *
|
||
|
fz_open_archive(fz_context *ctx, const char *filename)
|
||
|
{
|
||
|
fz_stream *file;
|
||
|
fz_archive *arch = NULL;
|
||
|
|
||
|
file = fz_open_file(ctx, filename);
|
||
|
|
||
|
fz_try(ctx)
|
||
|
arch = fz_open_archive_with_stream(ctx, file);
|
||
|
fz_always(ctx)
|
||
|
fz_drop_stream(ctx, file);
|
||
|
fz_catch(ctx)
|
||
|
fz_rethrow(ctx);
|
||
|
|
||
|
return arch;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
fz_drop_archive(fz_context *ctx, fz_archive *arch)
|
||
|
{
|
||
|
if (!arch)
|
||
|
return;
|
||
|
|
||
|
if (arch->drop_archive)
|
||
|
arch->drop_archive(ctx, arch);
|
||
|
fz_drop_stream(ctx, arch->file);
|
||
|
fz_free(ctx, arch);
|
||
|
}
|