Arquivos
chromium/chrome/browser/extensions/extension_creator.cc
T
thestig@chromium.org 431dc5a0d1 Remove svn:executable property from some more files that don't need it.
TBR=evmar
Review URL: http://codereview.chromium.org/149471

git-svn-id: svn://svn.chromium.org/chrome/trunk/src@20406 0039d316-1c4b-4281-b951-d872f2087c98
2009-07-10 20:14:40 +00:00

226 linhas
7.1 KiB
C++

// Copyright (c) 2009 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/extensions/extension_creator.h"
#include <vector>
#include <string>
#include "base/crypto/rsa_private_key.h"
#include "base/crypto/signature_creator.h"
#include "base/file_util.h"
#include "base/scoped_handle.h"
#include "base/string_util.h"
#include "chrome/browser/extensions/extensions_service.h"
#include "chrome/common/extensions/extension.h"
#include "chrome/common/zip.h"
#include "net/base/base64.h"
namespace {
const int kRSAKeySize = 1024;
};
bool ExtensionCreator::InitializeInput(
const FilePath& extension_dir,
const FilePath& private_key_path,
const FilePath& private_key_output_path) {
// Validate input |extension_dir|.
if (extension_dir.value().empty() ||
!file_util::DirectoryExists(extension_dir)) {
error_message_ = "Input directory must exist.";
return false;
}
// Validate input |private_key| (if provided).
if (!private_key_path.value().empty() &&
!file_util::PathExists(private_key_path)) {
error_message_ = "Input value for private key must be a valid path.";
return false;
}
// If an |output_private_key| path is given, make sure it doesn't over-write
// an existing private key.
if (private_key_path.value().empty() &&
!private_key_output_path.value().empty() &&
file_util::PathExists(private_key_output_path)) {
error_message_ = "Private key exists next to input directory. Try using "
"--pack-extension-key";
return false;
}
return true;
}
base::RSAPrivateKey* ExtensionCreator::ReadInputKey(const FilePath&
private_key_path) {
if (!file_util::PathExists(private_key_path)) {
error_message_ = "Input value for private key must exist.";
return false;
}
std::string private_key_contents;
if (!file_util::ReadFileToString(private_key_path,
&private_key_contents)) {
error_message_ = "Failed to read private key.";
return false;
}
std::string private_key_bytes;
if (!Extension::ParsePEMKeyBytes(private_key_contents,
&private_key_bytes)) {
error_message_ = "Invalid private key.";
return false;
}
return base::RSAPrivateKey::CreateFromPrivateKeyInfo(
std::vector<uint8>(private_key_bytes.begin(), private_key_bytes.end()));
}
base::RSAPrivateKey* ExtensionCreator::GenerateKey(const FilePath&
output_private_key_path) {
scoped_ptr<base::RSAPrivateKey> key_pair(
base::RSAPrivateKey::Create(kRSAKeySize));
if (!key_pair.get()) {
error_message_ = "Yikes! Failed to generate random RSA private key.";
return NULL;
}
std::vector<uint8> private_key_vector;
if (!key_pair->ExportPrivateKey(&private_key_vector)) {
error_message_ = "Failed to export private key.";
return NULL;
}
std::string private_key_bytes(
reinterpret_cast<char*>(&private_key_vector.front()),
private_key_vector.size());
std::string private_key;
if (!Extension::ProducePEM(private_key_bytes, &private_key)) {
error_message_ = "Failed to output private key.";
return NULL;
}
std::string pem_output;
if (!Extension::FormatPEMForFileOutput(private_key, &pem_output,
false)) {
error_message_ = "Failed to output private key.";
return NULL;
}
if (!output_private_key_path.empty()) {
if (-1 == file_util::WriteFile(output_private_key_path,
pem_output.c_str(), pem_output.size())) {
error_message_ = "Failed to write private key.";
return NULL;
}
}
return key_pair.release();
}
bool ExtensionCreator::CreateZip(const FilePath& extension_dir,
FilePath* zip_path) {
file_util::CreateNewTempDirectory(FILE_PATH_LITERAL("chrome_"), zip_path);
*zip_path = zip_path->Append(FILE_PATH_LITERAL("extension.zip"));
if (!Zip(extension_dir, *zip_path)) {
error_message_ = "Failed to create temporary zip file during packaging.";
return false;
}
return true;
}
bool ExtensionCreator::SignZip(const FilePath& zip_path,
base::RSAPrivateKey* private_key,
std::vector<uint8>* signature) {
scoped_ptr<base::SignatureCreator> signature_creator(
base::SignatureCreator::Create(private_key));
ScopedStdioHandle zip_handle(file_util::OpenFile(zip_path, "rb"));
uint8 buffer[1 << 16];
int bytes_read = -1;
while ((bytes_read = fread(buffer, 1, sizeof(buffer),
zip_handle.get())) > 0) {
if (!signature_creator->Update(buffer, bytes_read)) {
error_message_ = "Error while signing extension.";
return false;
}
}
zip_handle.Close();
signature_creator->Final(signature);
return true;
}
bool ExtensionCreator::WriteCRX(const FilePath& zip_path,
base::RSAPrivateKey* private_key,
const std::vector<uint8>& signature,
const FilePath& crx_path) {
if (file_util::PathExists(crx_path))
file_util::Delete(crx_path, false);
ScopedStdioHandle crx_handle(file_util::OpenFile(crx_path, "wb"));
std::vector<uint8> public_key;
if (!private_key->ExportPublicKey(&public_key)) {
error_message_ = "Failed to export public key.";
return false;
}
ExtensionsService::ExtensionHeader header;
memcpy(&header.magic, ExtensionsService::kExtensionHeaderMagic,
ExtensionsService::kExtensionHeaderMagicSize);
header.version = ExtensionsService::kCurrentVersion;
header.key_size = public_key.size();
header.signature_size = signature.size();
fwrite(&header, sizeof(ExtensionsService::ExtensionHeader), 1,
crx_handle.get());
fwrite(&public_key.front(), sizeof(uint8), public_key.size(),
crx_handle.get());
fwrite(&signature.front(), sizeof(uint8), signature.size(),
crx_handle.get());
uint8 buffer[1 << 16];
int bytes_read = -1;
ScopedStdioHandle zip_handle(file_util::OpenFile(zip_path, "rb"));
while ((bytes_read = fread(buffer, 1, sizeof(buffer),
zip_handle.get())) > 0) {
fwrite(buffer, sizeof(char), bytes_read, crx_handle.get());
}
return true;
}
bool ExtensionCreator::Run(const FilePath& extension_dir,
const FilePath& crx_path,
const FilePath& private_key_path,
const FilePath& output_private_key_path) {
// Check input diretory and read manifest.
if (!InitializeInput(extension_dir, private_key_path,
output_private_key_path)) {
return false;
}
// Initialize Key Pair
scoped_ptr<base::RSAPrivateKey> key_pair;
if (!private_key_path.value().empty())
key_pair.reset(ReadInputKey(private_key_path));
else
key_pair.reset(GenerateKey(output_private_key_path));
if (!key_pair.get())
return false;
// Zip up the extension.
FilePath zip_path;
std::vector<uint8> signature;
bool result = false;
if (CreateZip(extension_dir, &zip_path) &&
SignZip(zip_path, key_pair.get(), &signature) &&
WriteCRX(zip_path, key_pair.get(), signature, crx_path)) {
result = true;
}
file_util::Delete(zip_path, false);
return result;
}