Home | History | Annotate | Download | only in filesystem
      1 /*
      2  * Copyright (C) 2018 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 "src/traced/probes/filesystem/file_scanner.h"
     18 
     19 #include "gmock/gmock.h"
     20 #include "gtest/gtest.h"
     21 
     22 #include <sys/stat.h>
     23 #include <memory>
     24 #include <string>
     25 
     26 #include "perfetto/base/logging.h"
     27 #include "src/base/test/test_task_runner.h"
     28 
     29 namespace perfetto {
     30 namespace {
     31 
     32 using ::testing::Eq;
     33 using ::testing::Contains;
     34 using ::testing::UnorderedElementsAre;
     35 
     36 class TestDelegate : public FileScanner::Delegate {
     37  public:
     38   TestDelegate(
     39       std::function<bool(BlockDeviceID,
     40                          Inode,
     41                          const std::string&,
     42                          protos::pbzero::InodeFileMap_Entry_Type)> callback,
     43       std::function<void()> done_callback)
     44       : callback_(std::move(callback)),
     45         done_callback_(std::move(done_callback)) {}
     46   bool OnInodeFound(BlockDeviceID block_device_id,
     47                     Inode inode,
     48                     const std::string& path,
     49                     protos::pbzero::InodeFileMap_Entry_Type type) override {
     50     return callback_(block_device_id, inode, path, type);
     51   }
     52 
     53   void OnInodeScanDone() { return done_callback_(); }
     54 
     55  private:
     56   std::function<bool(BlockDeviceID,
     57                      Inode,
     58                      const std::string&,
     59                      protos::pbzero::InodeFileMap_Entry_Type)>
     60       callback_;
     61   std::function<void()> done_callback_;
     62 };
     63 
     64 struct FileEntry {
     65   FileEntry(BlockDeviceID block_device_id,
     66             Inode inode,
     67             std::string path,
     68             protos::pbzero::InodeFileMap_Entry_Type type)
     69       : block_device_id_(block_device_id),
     70         inode_(inode),
     71         path_(std::move(path)),
     72         type_(type) {}
     73 
     74   bool operator==(const FileEntry& other) const {
     75     return block_device_id_ == other.block_device_id_ &&
     76            inode_ == other.inode_ && path_ == other.path_ &&
     77            type_ == other.type_;
     78   }
     79 
     80   BlockDeviceID block_device_id_;
     81   Inode inode_;
     82   std::string path_;
     83   protos::pbzero::InodeFileMap_Entry_Type type_;
     84 };
     85 
     86 struct stat CheckStat(const std::string& path) {
     87   struct stat buf;
     88   PERFETTO_CHECK(lstat(path.c_str(), &buf) != -1);
     89   return buf;
     90 }
     91 
     92 FileEntry StatFileEntry(const std::string& path,
     93                         protos::pbzero::InodeFileMap_Entry_Type type) {
     94   struct stat buf = CheckStat(path);
     95   return FileEntry(buf.st_dev, buf.st_ino, path, type);
     96 }
     97 
     98 TEST(FileScannerTest, TestSynchronousStop) {
     99   uint64_t seen = 0;
    100   bool done = false;
    101   TestDelegate delegate(
    102       [&seen](BlockDeviceID, Inode, const std::string&,
    103               protos::pbzero::InodeFileMap_Entry_Type) {
    104         ++seen;
    105         return false;
    106       },
    107       [&done] { done = true; });
    108 
    109   FileScanner fs({"src/traced/probes/filesystem/testdata"}, &delegate);
    110   fs.Scan();
    111 
    112   EXPECT_EQ(seen, 1u);
    113   EXPECT_TRUE(done);
    114 }
    115 
    116 TEST(FileScannerTest, TestAsynchronousStop) {
    117   uint64_t seen = 0;
    118   base::TestTaskRunner task_runner;
    119   TestDelegate delegate(
    120       [&seen](BlockDeviceID, Inode, const std::string&,
    121               protos::pbzero::InodeFileMap_Entry_Type) {
    122         ++seen;
    123         return false;
    124       },
    125       task_runner.CreateCheckpoint("done"));
    126 
    127   FileScanner fs({"src/traced/probes/filesystem/testdata"}, &delegate, 1, 1);
    128   fs.Scan(&task_runner);
    129 
    130   task_runner.RunUntilCheckpoint("done");
    131 
    132   EXPECT_EQ(seen, 1u);
    133 }
    134 
    135 TEST(FileScannerTest, TestSynchronousFindFiles) {
    136   std::vector<FileEntry> file_entries;
    137   TestDelegate delegate(
    138       [&file_entries](BlockDeviceID block_device_id, Inode inode,
    139                       const std::string& path,
    140                       protos::pbzero::InodeFileMap_Entry_Type type) {
    141         file_entries.emplace_back(block_device_id, inode, path, type);
    142         return true;
    143       },
    144       [] {});
    145 
    146   FileScanner fs({"src/traced/probes/filesystem/testdata"}, &delegate);
    147   fs.Scan();
    148 
    149   EXPECT_THAT(
    150       file_entries,
    151       UnorderedElementsAre(
    152           Eq(StatFileEntry("src/traced/probes/filesystem/testdata/dir1/file1",
    153                            protos::pbzero::InodeFileMap_Entry_Type_FILE)),
    154           Eq(StatFileEntry("src/traced/probes/filesystem/testdata/file2",
    155                            protos::pbzero::InodeFileMap_Entry_Type_FILE)),
    156           Eq(StatFileEntry(
    157               "src/traced/probes/filesystem/testdata/dir1",
    158               protos::pbzero::InodeFileMap_Entry_Type_DIRECTORY))));
    159 }
    160 
    161 TEST(FileScannerTest, TestAsynchronousFindFiles) {
    162   base::TestTaskRunner task_runner;
    163   std::vector<FileEntry> file_entries;
    164   TestDelegate delegate(
    165       [&file_entries](BlockDeviceID block_device_id, Inode inode,
    166                       const std::string& path,
    167                       protos::pbzero::InodeFileMap_Entry_Type type) {
    168         file_entries.emplace_back(block_device_id, inode, path, type);
    169         return true;
    170       },
    171       task_runner.CreateCheckpoint("done"));
    172 
    173   FileScanner fs({"src/traced/probes/filesystem/testdata"}, &delegate, 1, 1);
    174   fs.Scan(&task_runner);
    175 
    176   task_runner.RunUntilCheckpoint("done");
    177 
    178   EXPECT_THAT(
    179       file_entries,
    180       UnorderedElementsAre(
    181           Eq(StatFileEntry("src/traced/probes/filesystem/testdata/dir1/file1",
    182                            protos::pbzero::InodeFileMap_Entry_Type_FILE)),
    183           Eq(StatFileEntry("src/traced/probes/filesystem/testdata/file2",
    184                            protos::pbzero::InodeFileMap_Entry_Type_FILE)),
    185           Eq(StatFileEntry(
    186               "src/traced/probes/filesystem/testdata/dir1",
    187               protos::pbzero::InodeFileMap_Entry_Type_DIRECTORY))));
    188 }
    189 
    190 }  // namespace
    191 }  // namespace perfetto
    192