// Copyright (c) 2014 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "atom/common/asar/archive.h" #if defined(OS_WIN) #include #endif #include #include #include "atom/common/asar/scoped_temporary_file.h" #include "base/files/file.h" #include "base/logging.h" #include "base/pickle.h" #include "base/json/json_reader.h" #include "base/strings/string_number_conversions.h" #include "base/values.h" namespace asar { namespace { #if defined(OS_WIN) const char kSeparators[] = "\\/"; #else const char kSeparators[] = "/"; #endif bool GetNodeFromPath(std::string path, const base::DictionaryValue* root, const base::DictionaryValue** out); // Gets the "files" from "dir". bool GetFilesNode(const base::DictionaryValue* root, const base::DictionaryValue* dir, const base::DictionaryValue** out) { // Test for symbol linked directory. std::string link; if (dir->GetStringWithoutPathExpansion("link", &link)) { const base::DictionaryValue* linked_node = NULL; if (!GetNodeFromPath(link, root, &linked_node)) return false; dir = linked_node; } return dir->GetDictionaryWithoutPathExpansion("files", out); } // Gets sub-file "name" from "dir". bool GetChildNode(const base::DictionaryValue* root, const std::string& name, const base::DictionaryValue* dir, const base::DictionaryValue** out) { if (name == "") { *out = root; return true; } const base::DictionaryValue* files = NULL; return GetFilesNode(root, dir, &files) && files->GetDictionaryWithoutPathExpansion(name, out); } // Gets the node of "path" from "root". bool GetNodeFromPath(std::string path, const base::DictionaryValue* root, const base::DictionaryValue** out) { if (path == "") { *out = root; return true; } const base::DictionaryValue* dir = root; for (size_t delimiter_position = path.find_first_of(kSeparators); delimiter_position != std::string::npos; delimiter_position = path.find_first_of(kSeparators)) { const base::DictionaryValue* child = NULL; if (!GetChildNode(root, path.substr(0, delimiter_position), dir, &child)) return false; dir = child; path.erase(0, delimiter_position + 1); } return GetChildNode(root, path, dir, out); } bool FillFileInfoWithNode(Archive::FileInfo* info, uint32 header_size, const base::DictionaryValue* node) { int size; if (!node->GetInteger("size", &size)) return false; info->size = static_cast(size); info->unpacked = false; if (node->GetBoolean("unpacked", &info->unpacked) && info->unpacked) return true; std::string offset; if (!node->GetString("offset", &offset)) return false; if (!base::StringToUint64(offset, &info->offset)) return false; info->offset += header_size; return true; } } // namespace Archive::Archive(const base::FilePath& path) : path_(path), file_(path_, base::File::FLAG_OPEN | base::File::FLAG_READ), #if defined(OS_WIN) fd_(_open_osfhandle( reinterpret_cast(file_.GetPlatformFile()), 0)), #elif defined(OS_POSIX) fd_(file_.GetPlatformFile()), #else fd_(-1), #endif header_size_(0) { } Archive::~Archive() { } bool Archive::Init() { if (!file_.IsValid()) return false; std::vector buf; int len; buf.resize(8); len = file_.ReadAtCurrentPos(buf.data(), buf.size()); if (len != static_cast(buf.size())) { PLOG(ERROR) << "Failed to read header size from " << path_.value(); return false; } uint32 size; if (!base::PickleIterator(base::Pickle(buf.data(), buf.size())).ReadUInt32( &size)) { LOG(ERROR) << "Failed to parse header size from " << path_.value(); return false; } buf.resize(size); len = file_.ReadAtCurrentPos(buf.data(), buf.size()); if (len != static_cast(buf.size())) { PLOG(ERROR) << "Failed to read header from " << path_.value(); return false; } std::string header; if (!base::PickleIterator(base::Pickle(buf.data(), buf.size())).ReadString( &header)) { LOG(ERROR) << "Failed to parse header from " << path_.value(); return false; } std::string error; base::JSONReader reader; scoped_ptr value(reader.ReadToValue(header)); if (!value || !value->IsType(base::Value::TYPE_DICTIONARY)) { LOG(ERROR) << "Failed to parse header: " << error; return false; } header_size_ = 8 + size; header_.reset(static_cast(value.release())); return true; } bool Archive::GetFileInfo(const base::FilePath& path, FileInfo* info) { if (!header_) return false; const base::DictionaryValue* node; if (!GetNodeFromPath(path.AsUTF8Unsafe(), header_.get(), &node)) return false; std::string link; if (node->GetString("link", &link)) return GetFileInfo(base::FilePath::FromUTF8Unsafe(link), info); return FillFileInfoWithNode(info, header_size_, node); } bool Archive::Stat(const base::FilePath& path, Stats* stats) { if (!header_) return false; const base::DictionaryValue* node; if (!GetNodeFromPath(path.AsUTF8Unsafe(), header_.get(), &node)) return false; if (node->HasKey("link")) { stats->is_file = false; stats->is_link = true; return true; } if (node->HasKey("files")) { stats->is_file = false; stats->is_directory = true; return true; } return FillFileInfoWithNode(stats, header_size_, node); } bool Archive::Readdir(const base::FilePath& path, std::vector* list) { if (!header_) return false; const base::DictionaryValue* node; if (!GetNodeFromPath(path.AsUTF8Unsafe(), header_.get(), &node)) return false; const base::DictionaryValue* files; if (!GetFilesNode(header_.get(), node, &files)) return false; base::DictionaryValue::Iterator iter(*files); while (!iter.IsAtEnd()) { list->push_back(base::FilePath::FromUTF8Unsafe(iter.key())); iter.Advance(); } return true; } bool Archive::Realpath(const base::FilePath& path, base::FilePath* realpath) { if (!header_) return false; const base::DictionaryValue* node; if (!GetNodeFromPath(path.AsUTF8Unsafe(), header_.get(), &node)) return false; std::string link; if (node->GetString("link", &link)) { *realpath = base::FilePath::FromUTF8Unsafe(link); return true; } *realpath = path; return true; } bool Archive::CopyFileOut(const base::FilePath& path, base::FilePath* out) { if (external_files_.contains(path)) { *out = external_files_.get(path)->path(); return true; } FileInfo info; if (!GetFileInfo(path, &info)) return false; if (info.unpacked) { *out = path_.AddExtension(FILE_PATH_LITERAL("unpacked")).Append(path); return true; } scoped_ptr temp_file(new ScopedTemporaryFile); if (!temp_file->InitFromFile(&file_, info.offset, info.size)) return false; *out = temp_file->path(); external_files_.set(path, temp_file.Pass()); return true; } int Archive::GetFD() const { return fd_; } } // namespace asar