// 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 <vector> #include "gin/handle.h" #include "shell/common/asar/archive.h" #include "shell/common/asar/asar_util.h" #include "shell/common/gin_converters/file_path_converter.h" #include "shell/common/gin_helper/dictionary.h" #include "shell/common/node_includes.h" #include "shell/common/node_util.h" namespace { class Archive : public node::ObjectWrap { public: static v8::Local<v8::FunctionTemplate> CreateFunctionTemplate( v8::Isolate* isolate) { auto tpl = v8::FunctionTemplate::New(isolate, Archive::New); tpl->SetClassName( v8::String::NewFromUtf8(isolate, "Archive").ToLocalChecked()); tpl->InstanceTemplate()->SetInternalFieldCount(1); NODE_SET_PROTOTYPE_METHOD(tpl, "getFileInfo", &Archive::GetFileInfo); NODE_SET_PROTOTYPE_METHOD(tpl, "stat", &Archive::Stat); NODE_SET_PROTOTYPE_METHOD(tpl, "readdir", &Archive::Readdir); NODE_SET_PROTOTYPE_METHOD(tpl, "realpath", &Archive::Realpath); NODE_SET_PROTOTYPE_METHOD(tpl, "copyFileOut", &Archive::CopyFileOut); NODE_SET_PROTOTYPE_METHOD(tpl, "getFdAndValidateIntegrityLater", &Archive::GetFD); return tpl; } // disable copy Archive(const Archive&) = delete; Archive& operator=(const Archive&) = delete; protected: explicit Archive(std::shared_ptr<asar::Archive> archive) : archive_(std::move(archive)) {} static void New(const v8::FunctionCallbackInfo<v8::Value>& args) { auto* isolate = args.GetIsolate(); base::FilePath path; if (!gin::ConvertFromV8(isolate, args[0], &path)) { isolate->ThrowException(v8::Exception::Error(node::FIXED_ONE_BYTE_STRING( isolate, "failed to convert path to V8"))); return; } std::shared_ptr<asar::Archive> archive = asar::GetOrCreateAsarArchive(path); if (!archive) { isolate->ThrowException(v8::Exception::Error(node::FIXED_ONE_BYTE_STRING( isolate, "failed to initialize archive"))); return; } auto* archive_wrap = new Archive(std::move(archive)); archive_wrap->Wrap(args.This()); args.GetReturnValue().Set(args.This()); } // Reads the offset and size of file. static void GetFileInfo(const v8::FunctionCallbackInfo<v8::Value>& args) { auto* isolate = args.GetIsolate(); auto* wrap = node::ObjectWrap::Unwrap<Archive>(args.Holder()); base::FilePath path; if (!gin::ConvertFromV8(isolate, args[0], &path)) { args.GetReturnValue().Set(v8::False(isolate)); return; } asar::Archive::FileInfo info; if (!wrap->archive_ || !wrap->archive_->GetFileInfo(path, &info)) { args.GetReturnValue().Set(v8::False(isolate)); return; } gin_helper::Dictionary dict(isolate, v8::Object::New(isolate)); dict.Set("size", info.size); dict.Set("unpacked", info.unpacked); dict.Set("offset", info.offset); if (info.integrity.has_value()) { gin_helper::Dictionary integrity(isolate, v8::Object::New(isolate)); asar::HashAlgorithm algorithm = info.integrity.value().algorithm; switch (algorithm) { case asar::HashAlgorithm::SHA256: integrity.Set("algorithm", "SHA256"); break; case asar::HashAlgorithm::NONE: CHECK(false); break; } integrity.Set("hash", info.integrity.value().hash); dict.Set("integrity", integrity); } args.GetReturnValue().Set(dict.GetHandle()); } // Returns a fake result of fs.stat(path). static void Stat(const v8::FunctionCallbackInfo<v8::Value>& args) { auto* isolate = args.GetIsolate(); auto* wrap = node::ObjectWrap::Unwrap<Archive>(args.Holder()); base::FilePath path; if (!gin::ConvertFromV8(isolate, args[0], &path)) { args.GetReturnValue().Set(v8::False(isolate)); return; } asar::Archive::Stats stats; if (!wrap->archive_ || !wrap->archive_->Stat(path, &stats)) { args.GetReturnValue().Set(v8::False(isolate)); return; } gin_helper::Dictionary dict(isolate, v8::Object::New(isolate)); dict.Set("size", stats.size); dict.Set("offset", stats.offset); dict.Set("isFile", stats.is_file); dict.Set("isDirectory", stats.is_directory); dict.Set("isLink", stats.is_link); args.GetReturnValue().Set(dict.GetHandle()); } // Returns all files under a directory. static void Readdir(const v8::FunctionCallbackInfo<v8::Value>& args) { auto* isolate = args.GetIsolate(); auto* wrap = node::ObjectWrap::Unwrap<Archive>(args.Holder()); base::FilePath path; if (!gin::ConvertFromV8(isolate, args[0], &path)) { args.GetReturnValue().Set(v8::False(isolate)); return; } std::vector<base::FilePath> files; if (!wrap->archive_ || !wrap->archive_->Readdir(path, &files)) { args.GetReturnValue().Set(v8::False(isolate)); return; } args.GetReturnValue().Set(gin::ConvertToV8(isolate, files)); } // Returns the path of file with symbol link resolved. static void Realpath(const v8::FunctionCallbackInfo<v8::Value>& args) { auto* isolate = args.GetIsolate(); auto* wrap = node::ObjectWrap::Unwrap<Archive>(args.Holder()); base::FilePath path; if (!gin::ConvertFromV8(isolate, args[0], &path)) { args.GetReturnValue().Set(v8::False(isolate)); return; } base::FilePath realpath; if (!wrap->archive_ || !wrap->archive_->Realpath(path, &realpath)) { args.GetReturnValue().Set(v8::False(isolate)); return; } args.GetReturnValue().Set(gin::ConvertToV8(isolate, realpath)); } // Copy the file out into a temporary file and returns the new path. static void CopyFileOut(const v8::FunctionCallbackInfo<v8::Value>& args) { auto* isolate = args.GetIsolate(); auto* wrap = node::ObjectWrap::Unwrap<Archive>(args.Holder()); base::FilePath path; if (!gin::ConvertFromV8(isolate, args[0], &path)) { args.GetReturnValue().Set(v8::False(isolate)); return; } base::FilePath new_path; if (!wrap->archive_ || !wrap->archive_->CopyFileOut(path, &new_path)) { args.GetReturnValue().Set(v8::False(isolate)); return; } args.GetReturnValue().Set(gin::ConvertToV8(isolate, new_path)); } // Return the file descriptor. static void GetFD(const v8::FunctionCallbackInfo<v8::Value>& args) { auto* isolate = args.GetIsolate(); auto* wrap = node::ObjectWrap::Unwrap<Archive>(args.Holder()); args.GetReturnValue().Set(gin::ConvertToV8( isolate, wrap->archive_ ? wrap->archive_->GetUnsafeFD() : -1)); } std::shared_ptr<asar::Archive> archive_; }; static void InitAsarSupport(const v8::FunctionCallbackInfo<v8::Value>& args) { auto* isolate = args.GetIsolate(); auto require = args[0]; // Evaluate asar_bundle.js. std::vector<v8::Local<v8::String>> asar_bundle_params = { node::FIXED_ONE_BYTE_STRING(isolate, "require")}; std::vector<v8::Local<v8::Value>> asar_bundle_args = {require}; electron::util::CompileAndCall( isolate->GetCurrentContext(), "electron/js2c/asar_bundle", &asar_bundle_params, &asar_bundle_args, nullptr); } static void SplitPath(const v8::FunctionCallbackInfo<v8::Value>& args) { auto* isolate = args.GetIsolate(); base::FilePath path; if (!gin::ConvertFromV8(isolate, args[0], &path)) { args.GetReturnValue().Set(v8::False(isolate)); return; } gin_helper::Dictionary dict = gin::Dictionary::CreateEmpty(isolate); base::FilePath asar_path, file_path; if (asar::GetAsarArchivePath(path, &asar_path, &file_path, true)) { dict.Set("isAsar", true); dict.Set("asarPath", asar_path); dict.Set("filePath", file_path); } else { dict.Set("isAsar", false); } args.GetReturnValue().Set(dict.GetHandle()); } void Initialize(v8::Local<v8::Object> exports, v8::Local<v8::Value> unused, v8::Local<v8::Context> context, void* priv) { auto* isolate = exports->GetIsolate(); auto cons = Archive::CreateFunctionTemplate(isolate) ->GetFunction(context) .ToLocalChecked(); cons->SetName(node::FIXED_ONE_BYTE_STRING(isolate, "Archive")); exports->Set(context, node::FIXED_ONE_BYTE_STRING(isolate, "Archive"), cons) .Check(); NODE_SET_METHOD(exports, "splitPath", &SplitPath); NODE_SET_METHOD(exports, "initAsarSupport", &InitAsarSupport); } } // namespace NODE_LINKED_BINDING_CONTEXT_AWARE(electron_common_asar, Initialize)