TestHelpers.cpp revision 7ad1110ecd6a840fcd2895c62668828a1ca029c6
1/*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "TestHelpers.h"
18
19#include <libgen.h>
20#include <unistd.h>
21
22#include <memory>
23#include <string>
24
25#include "android-base/file.h"
26#include "android-base/logging.h"
27#include "android-base/strings.h"
28#include "ziparchive/zip_archive.h"
29
30namespace android {
31
32static std::string sTestDataPath;
33
34// Extract the directory of the current executable path.
35static std::string GetExecutableDir() {
36  const std::string path = base::GetExecutablePath();
37  std::unique_ptr<char, decltype(&std::free)> mutable_path = {strdup(path.c_str()), std::free};
38  std::string executable_dir = dirname(mutable_path.get());
39  return executable_dir;
40}
41
42void InitializeTest(int* argc, char** argv) {
43  // Set the default test data path to be the executable path directory.
44  SetTestDataPath(GetExecutableDir());
45
46  for (int i = 1; i < *argc; i++) {
47    const std::string arg = argv[i];
48    if (base::StartsWith(arg, "--testdata=")) {
49      SetTestDataPath(arg.substr(strlen("--testdata=")));
50      for (int j = i; j != *argc; j++) {
51        argv[j] = argv[j + 1];
52      }
53      --(*argc);
54      --i;
55    } else if (arg == "-h" || arg == "--help") {
56      std::cerr << "\nAdditional options specific to this test:\n"
57                   "  --testdata=[PATH]\n"
58                   "      Specify the location of test data used within the tests.\n";
59      exit(1);
60    }
61  }
62}
63
64void SetTestDataPath(const std::string& path) { sTestDataPath = path; }
65
66const std::string& GetTestDataPath() {
67  CHECK(!sTestDataPath.empty()) << "no test data path set.";
68  return sTestDataPath;
69}
70
71::testing::AssertionResult ReadFileFromZipToString(const std::string& zip_path,
72                                                   const std::string& file,
73                                                   std::string* out_contents) {
74  out_contents->clear();
75  ::ZipArchiveHandle handle;
76  int32_t result = OpenArchive(zip_path.c_str(), &handle);
77  if (result != 0) {
78    return ::testing::AssertionFailure() << "Failed to open zip '" << zip_path
79                                         << "': " << ::ErrorCodeString(result);
80  }
81
82  ::ZipString name(file.c_str());
83  ::ZipEntry entry;
84  result = ::FindEntry(handle, name, &entry);
85  if (result != 0) {
86    ::CloseArchive(handle);
87    return ::testing::AssertionFailure() << "Could not find file '" << file << "' in zip '"
88                                         << zip_path << "' : " << ::ErrorCodeString(result);
89  }
90
91  out_contents->resize(entry.uncompressed_length);
92  result = ::ExtractToMemory(
93      handle, &entry, const_cast<uint8_t*>(reinterpret_cast<const uint8_t*>(out_contents->data())),
94      out_contents->size());
95  if (result != 0) {
96    ::CloseArchive(handle);
97    return ::testing::AssertionFailure() << "Failed to extract file '" << file << "' from zip '"
98                                         << zip_path << "': " << ::ErrorCodeString(result);
99  }
100
101  ::CloseArchive(handle);
102  return ::testing::AssertionSuccess();
103}
104
105::testing::AssertionResult IsStringEqual(const ResTable& table, uint32_t resource_id,
106                                         const char* expected_str) {
107  Res_value val;
108  ssize_t block = table.getResource(resource_id, &val, MAY_NOT_BE_BAG);
109  if (block < 0) {
110    return ::testing::AssertionFailure() << "could not find resource";
111  }
112
113  if (val.dataType != Res_value::TYPE_STRING) {
114    return ::testing::AssertionFailure() << "resource is not a string";
115  }
116
117  const ResStringPool* pool = table.getTableStringBlock(block);
118  if (pool == NULL) {
119    return ::testing::AssertionFailure() << "table has no string pool for block " << block;
120  }
121
122  const String8 actual_str = pool->string8ObjectAt(val.data);
123  if (String8(expected_str) != actual_str) {
124    return ::testing::AssertionFailure() << actual_str.string();
125  }
126  return ::testing::AssertionSuccess() << actual_str.string();
127}
128
129std::string GetStringFromPool(const ResStringPool* pool, uint32_t idx) {
130  String8 str = pool->string8ObjectAt(idx);
131  return std::string(str.string(), str.length());
132}
133
134}  // namespace android
135