You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
180 lines
8.3 KiB
C
180 lines
8.3 KiB
C
2 years ago
|
// Copyright 2008, Google Inc. All rights reserved.
|
||
|
//
|
||
|
// Redistribution and use in source and binary forms, with or without
|
||
|
// modification, are permitted provided that the following conditions are met:
|
||
|
//
|
||
|
// 1. Redistributions of source code must retain the above copyright notice,
|
||
|
// this list of conditions and the following disclaimer.
|
||
|
// 2. Redistributions in binary form must reproduce the above copyright notice,
|
||
|
// this list of conditions and the following disclaimer in the documentation
|
||
|
// and/or other materials provided with the distribution.
|
||
|
// 3. Neither the name of Google Inc. nor the names of its contributors may be
|
||
|
// used to endorse or promote products derived from this software without
|
||
|
// specific prior written permission.
|
||
|
//
|
||
|
// THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||
|
// WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||
|
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
|
||
|
// EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
||
|
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
||
|
// OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
||
|
// WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
||
|
// OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
||
|
// ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
|
||
|
// This file contains the declaration of the KmzFile class.
|
||
|
|
||
|
#ifndef KML_ENGINE_KMZ_FILE_H__
|
||
|
#define KML_ENGINE_KMZ_FILE_H__
|
||
|
|
||
|
#include <vector>
|
||
|
#include "boost/intrusive_ptr.hpp"
|
||
|
#include "boost/scoped_ptr.hpp"
|
||
|
#include "kml/base/referent.h"
|
||
|
#include "kml/base/util.h"
|
||
|
#include "kml/engine/kml_file.h"
|
||
|
|
||
|
// ZipFile hides the implementation details of the underlying zip library from
|
||
|
// this interface.
|
||
|
namespace kmlbase {
|
||
|
class ZipFile;
|
||
|
}
|
||
|
|
||
|
namespace kmlengine {
|
||
|
|
||
|
// The Kmz class represents an instance of a KMZ file. It contains methods
|
||
|
// for reading and writing KMZ files. By default, there is an upper limit of
|
||
|
// 2 GB on uncompressed file sizes. If you need to lower this limit, use
|
||
|
// the set_max_uncompressed_size method.
|
||
|
class KmzFile : public kmlbase::Referent {
|
||
|
public:
|
||
|
~KmzFile();
|
||
|
|
||
|
// Open a KMZ file from a file path. Returns a pointer to a KmzFile object
|
||
|
// if the file could be opened and read, and the data was recognizably KMZ.
|
||
|
// Otherwise returns NULL.
|
||
|
static KmzFile* OpenFromFile(const char* kmz_filepath);
|
||
|
|
||
|
// Open a KMZ file from a string. Returns a pointer to a KmzFile object if a
|
||
|
// temporary file could be created, the data was recognizably KMZ. Otherwise
|
||
|
// returns NULL.
|
||
|
static KmzFile* OpenFromString(const string& kmz_data);
|
||
|
|
||
|
static KmzFile* CreateFromString(const string& kmz_data) {
|
||
|
return OpenFromString(kmz_data);
|
||
|
}
|
||
|
|
||
|
// Sets the upper limit for the largest uncompressed file size (in bytes)
|
||
|
// for the underlying Zip implementation to handle. By default it is 2 GB.
|
||
|
// If this is exceeded, any attempt to read the archived file will return
|
||
|
// false.
|
||
|
void set_max_uncompressed_file_size(unsigned int i);
|
||
|
|
||
|
// Returns the maximum uncompressed file size that the underlying Zip
|
||
|
// implementation will handle in bytes.
|
||
|
unsigned int get_max_uncompressed_file_size();
|
||
|
|
||
|
// Checks to see if kmz_data looks like a PK ZIP file.
|
||
|
static bool IsKmz(const string& kmz_data);
|
||
|
|
||
|
// Read the default KML file from a KMZ archive. This is defined as the first
|
||
|
// entry in the ZIP table of contents that ends in ".kml". Note that it may
|
||
|
// NOT be at the root level of the archive. The ZIP archives table of
|
||
|
// contents is exactly the order in which the source files were added to the
|
||
|
// archive. Returns false if no KML file. The output string is not cleared
|
||
|
// before being written to.
|
||
|
bool ReadKml(string* output) const;
|
||
|
|
||
|
// This does the same as ReadKml() and in addition returns the path of the
|
||
|
// KML file within the KMZ archive if a non-NULL kml_path is supplied.
|
||
|
// NOTE: While it is considered a best practice to have The KML file of
|
||
|
// a KMZ archive be "doc.kml" this is not always the case.
|
||
|
bool ReadKmlAndGetPath(string* output, string* kml_path) const;
|
||
|
|
||
|
// Read a specific file from a KMZ archive. Returns false if subfile was not
|
||
|
// found, or if subfile could not be read. Note: subfile must be a full path
|
||
|
// from the archive root. Relative references of "../../foo" are not handled.
|
||
|
// The output string is not cleared before being written to.
|
||
|
bool ReadFile(const char* subfile, string* output) const;
|
||
|
|
||
|
// Fills a vector of strings of the files contained in the opened KMZ archive.
|
||
|
// The vector is not cleared, only appended to. The string is the full path
|
||
|
// name of the KML file from the archive root, with '/' as the separator.
|
||
|
// Returns false upon error.
|
||
|
bool List(std::vector<string>* subfiles);
|
||
|
|
||
|
// Saves the raw bytes of the in-memory KMZ file.
|
||
|
bool SaveToString(string* kmz_bytes);
|
||
|
|
||
|
// These are for the creation of KMZ files:
|
||
|
|
||
|
// Creates an empty KmzFile at kmz_filepath on which AddFile may be called.
|
||
|
// Returns NULL if the file could not be created for writing.
|
||
|
static KmzFile* Create(const char* kmz_filepath);
|
||
|
|
||
|
// Writes data to path_in_kmz. The path must be relative to the root of the
|
||
|
// archive. e.g. AddFile(data, "somedir/file.png"). If not, false is returned.
|
||
|
// False is also returned on any interal zipfile error.
|
||
|
bool AddFile(const string& data, const string& path_in_kmz);
|
||
|
|
||
|
// Adds a StringVector of hrefs to the KMZ file, resolved against a base
|
||
|
// URL. The base URL is usually from kmz_file->get_url() and the hrefs
|
||
|
// are most easily generated from GetRelativeLinks. All paths are normalized
|
||
|
// prior to writing.
|
||
|
// Returns the number of errors encountered during processing.
|
||
|
// Errors may result from failure to normalize an href, an href that points
|
||
|
// above the base url, or failure to read the resolved file prior to writing.
|
||
|
// Duplicate entries are ignored and not considered errors.
|
||
|
size_t AddFileList(const string& base_url,
|
||
|
const kmlbase::StringVector& file_paths);
|
||
|
|
||
|
// Creates a KMZ file from a string of KML data. Returns true if
|
||
|
// kmz_filepath could be successfully created and written.
|
||
|
// TODO: Permit adding resources (images, models, etc.) to the KMZ archive.
|
||
|
static bool WriteKmz(const char* kmz_filepath, const string& kml);
|
||
|
|
||
|
// Creates a KMZ file at kmz_filepath from a string of KML. Any local
|
||
|
// references in the file are written to the KMZ as archived resources
|
||
|
// according to the rules explained in CreateFromElement.
|
||
|
// TODO: handle <Model> references.
|
||
|
// TODO: handle references in <description>.
|
||
|
static bool CreateFromKmlFilepath(const string& kml_filepath,
|
||
|
const string& kmz_filepath);
|
||
|
|
||
|
// Creates a KMZ file at kmz_filepath from an ElementPtr and a base url. Any
|
||
|
// local references in the file are written to the KMZ as archived resources
|
||
|
// if and only if the resource URI is relative to and below the base_url.
|
||
|
// i.e. <href>/etc/passwd</href> is not valid because it is absolute, and
|
||
|
// from a base url of "/home/libkml/" <href>../../etc/passwd</href> is
|
||
|
// invalid because it does not point below /home/libkml.
|
||
|
// TODO: handle <Model> references.
|
||
|
// TODO: handle references in <description>.
|
||
|
static bool CreateFromElement(const kmldom::ElementPtr& element,
|
||
|
const string& base_url,
|
||
|
const string& kmz_filepath);
|
||
|
|
||
|
// Creates a KMZ file at kmz_filepath from a KmlFile. Any local
|
||
|
// references in the file are written to the KMZ as archived resources
|
||
|
// according to the rules laid out above for CreateFromElement.
|
||
|
// The KmlFile _must_ have been created with its base URL set to the
|
||
|
// local path where the KML file can be found, i.e.
|
||
|
// KmlPtr kml_file = KmlFile::CreateFromStringWithUrl(...).
|
||
|
// TODO: handle <Model> references.
|
||
|
// TODO: handle references in <description>.
|
||
|
static bool CreateFromKmlFile(const KmlFilePtr& kml_file,
|
||
|
const string& kmz_filepath);
|
||
|
|
||
|
private:
|
||
|
// Class can only be created from static methods.
|
||
|
KmzFile(kmlbase::ZipFile* zip_file);
|
||
|
boost::scoped_ptr<kmlbase::ZipFile> zip_file_;
|
||
|
LIBKML_DISALLOW_EVIL_CONSTRUCTORS(KmzFile);
|
||
|
};
|
||
|
|
||
|
typedef boost::intrusive_ptr<KmzFile> KmzFilePtr;
|
||
|
|
||
|
} // end namespace kmlengine
|
||
|
|
||
|
#endif // KML_ENGINE_KMZ_FILE_H__
|