xref: /aosp_15_r20/system/apex/apexd/apexd_metrics.cpp (revision 33f3758387333dbd2962d7edbd98681940d895da)
1*33f37583SAndroid Build Coastguard Worker /*
2*33f37583SAndroid Build Coastguard Worker  * Copyright (C) 2024 The Android Open Source Project
3*33f37583SAndroid Build Coastguard Worker  *
4*33f37583SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*33f37583SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*33f37583SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*33f37583SAndroid Build Coastguard Worker  *
8*33f37583SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*33f37583SAndroid Build Coastguard Worker  *
10*33f37583SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*33f37583SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*33f37583SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*33f37583SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*33f37583SAndroid Build Coastguard Worker  * limitations under the License.
15*33f37583SAndroid Build Coastguard Worker  */
16*33f37583SAndroid Build Coastguard Worker 
17*33f37583SAndroid Build Coastguard Worker #include "apexd_metrics.h"
18*33f37583SAndroid Build Coastguard Worker 
19*33f37583SAndroid Build Coastguard Worker #include <android-base/logging.h>
20*33f37583SAndroid Build Coastguard Worker #include <android-base/result.h>
21*33f37583SAndroid Build Coastguard Worker #include <android-base/strings.h>
22*33f37583SAndroid Build Coastguard Worker #include <sys/stat.h>
23*33f37583SAndroid Build Coastguard Worker 
24*33f37583SAndroid Build Coastguard Worker #include <utility>
25*33f37583SAndroid Build Coastguard Worker 
26*33f37583SAndroid Build Coastguard Worker #include "apex_constants.h"
27*33f37583SAndroid Build Coastguard Worker #include "apex_file.h"
28*33f37583SAndroid Build Coastguard Worker #include "apex_file_repository.h"
29*33f37583SAndroid Build Coastguard Worker #include "apex_sha.h"
30*33f37583SAndroid Build Coastguard Worker #include "apexd_session.h"
31*33f37583SAndroid Build Coastguard Worker #include "apexd_vendor_apex.h"
32*33f37583SAndroid Build Coastguard Worker 
33*33f37583SAndroid Build Coastguard Worker using android::base::Result;
34*33f37583SAndroid Build Coastguard Worker using android::base::StartsWith;
35*33f37583SAndroid Build Coastguard Worker 
36*33f37583SAndroid Build Coastguard Worker namespace android::apex {
37*33f37583SAndroid Build Coastguard Worker 
38*33f37583SAndroid Build Coastguard Worker namespace {
39*33f37583SAndroid Build Coastguard Worker 
40*33f37583SAndroid Build Coastguard Worker std::unique_ptr<Metrics> gMetrics;
41*33f37583SAndroid Build Coastguard Worker 
42*33f37583SAndroid Build Coastguard Worker }  // namespace
43*33f37583SAndroid Build Coastguard Worker 
InitMetrics(std::unique_ptr<Metrics> metrics)44*33f37583SAndroid Build Coastguard Worker std::unique_ptr<Metrics> InitMetrics(std::unique_ptr<Metrics> metrics) {
45*33f37583SAndroid Build Coastguard Worker   std::swap(gMetrics, metrics);
46*33f37583SAndroid Build Coastguard Worker   return metrics;
47*33f37583SAndroid Build Coastguard Worker }
48*33f37583SAndroid Build Coastguard Worker 
SendApexInstallationEndedAtom(const std::string & package_path,InstallResult install_result)49*33f37583SAndroid Build Coastguard Worker void SendApexInstallationEndedAtom(const std::string& package_path,
50*33f37583SAndroid Build Coastguard Worker                                    InstallResult install_result) {
51*33f37583SAndroid Build Coastguard Worker   if (!gMetrics) {
52*33f37583SAndroid Build Coastguard Worker     return;
53*33f37583SAndroid Build Coastguard Worker   }
54*33f37583SAndroid Build Coastguard Worker   Result<std::string> hash = CalculateSha256(package_path);
55*33f37583SAndroid Build Coastguard Worker   if (!hash.ok()) {
56*33f37583SAndroid Build Coastguard Worker     LOG(WARNING) << "Unable to get sha256 of ApexFile: " << hash.error();
57*33f37583SAndroid Build Coastguard Worker     return;
58*33f37583SAndroid Build Coastguard Worker   }
59*33f37583SAndroid Build Coastguard Worker   gMetrics->SendInstallationEnded(*hash, install_result);
60*33f37583SAndroid Build Coastguard Worker }
61*33f37583SAndroid Build Coastguard Worker 
SendSessionApexInstallationEndedAtom(const ApexSession & session,InstallResult install_result)62*33f37583SAndroid Build Coastguard Worker void SendSessionApexInstallationEndedAtom(const ApexSession& session,
63*33f37583SAndroid Build Coastguard Worker                                           InstallResult install_result) {
64*33f37583SAndroid Build Coastguard Worker   if (!gMetrics) {
65*33f37583SAndroid Build Coastguard Worker     return;
66*33f37583SAndroid Build Coastguard Worker   }
67*33f37583SAndroid Build Coastguard Worker 
68*33f37583SAndroid Build Coastguard Worker   for (const auto& hash : session.GetApexFileHashes()) {
69*33f37583SAndroid Build Coastguard Worker     gMetrics->SendInstallationEnded(hash, install_result);
70*33f37583SAndroid Build Coastguard Worker   }
71*33f37583SAndroid Build Coastguard Worker }
72*33f37583SAndroid Build Coastguard Worker 
~InstallRequestedEvent()73*33f37583SAndroid Build Coastguard Worker InstallRequestedEvent::~InstallRequestedEvent() {
74*33f37583SAndroid Build Coastguard Worker   if (!gMetrics) {
75*33f37583SAndroid Build Coastguard Worker     return;
76*33f37583SAndroid Build Coastguard Worker   }
77*33f37583SAndroid Build Coastguard Worker   for (const auto& info : files_) {
78*33f37583SAndroid Build Coastguard Worker     gMetrics->SendInstallationRequested(install_type_, is_rollback_, info);
79*33f37583SAndroid Build Coastguard Worker   }
80*33f37583SAndroid Build Coastguard Worker   // Staged installation ends later. No need to send "end" event now.
81*33f37583SAndroid Build Coastguard Worker   if (succeeded_ && install_type_ == InstallType::Staged) {
82*33f37583SAndroid Build Coastguard Worker     return;
83*33f37583SAndroid Build Coastguard Worker   }
84*33f37583SAndroid Build Coastguard Worker   auto result = succeeded_ ? InstallResult::Success : InstallResult::Failure;
85*33f37583SAndroid Build Coastguard Worker   for (const auto& info : files_) {
86*33f37583SAndroid Build Coastguard Worker     gMetrics->SendInstallationEnded(info.file_hash, result);
87*33f37583SAndroid Build Coastguard Worker   }
88*33f37583SAndroid Build Coastguard Worker }
89*33f37583SAndroid Build Coastguard Worker 
MarkSucceeded()90*33f37583SAndroid Build Coastguard Worker void InstallRequestedEvent::MarkSucceeded() { succeeded_ = true; }
91*33f37583SAndroid Build Coastguard Worker 
AddFiles(std::span<const ApexFile> files)92*33f37583SAndroid Build Coastguard Worker void InstallRequestedEvent::AddFiles(std::span<const ApexFile> files) {
93*33f37583SAndroid Build Coastguard Worker   auto& repo = ApexFileRepository::GetInstance();
94*33f37583SAndroid Build Coastguard Worker   files_.reserve(files.size());
95*33f37583SAndroid Build Coastguard Worker   for (const auto& file : files) {
96*33f37583SAndroid Build Coastguard Worker     Metrics::ApexFileInfo info;
97*33f37583SAndroid Build Coastguard Worker     info.name = file.GetManifest().name();
98*33f37583SAndroid Build Coastguard Worker     info.version = file.GetManifest().version();
99*33f37583SAndroid Build Coastguard Worker     info.shared_libs = file.GetManifest().providesharedapexlibs();
100*33f37583SAndroid Build Coastguard Worker 
101*33f37583SAndroid Build Coastguard Worker     const auto& file_path = file.GetPath();
102*33f37583SAndroid Build Coastguard Worker     struct stat stat_buf;
103*33f37583SAndroid Build Coastguard Worker     if (stat(file_path.c_str(), &stat_buf) == 0) {
104*33f37583SAndroid Build Coastguard Worker       info.file_size = stat_buf.st_size;
105*33f37583SAndroid Build Coastguard Worker     } else {
106*33f37583SAndroid Build Coastguard Worker       PLOG(WARNING) << "Failed to stat " << file_path;
107*33f37583SAndroid Build Coastguard Worker       continue;
108*33f37583SAndroid Build Coastguard Worker     }
109*33f37583SAndroid Build Coastguard Worker 
110*33f37583SAndroid Build Coastguard Worker     if (auto result = CalculateSha256(file_path); result.ok()) {
111*33f37583SAndroid Build Coastguard Worker       info.file_hash = result.value();
112*33f37583SAndroid Build Coastguard Worker     } else {
113*33f37583SAndroid Build Coastguard Worker       LOG(WARNING) << "Unable to get sha256 of " << file_path << ": "
114*33f37583SAndroid Build Coastguard Worker                    << result.error();
115*33f37583SAndroid Build Coastguard Worker       continue;
116*33f37583SAndroid Build Coastguard Worker     }
117*33f37583SAndroid Build Coastguard Worker 
118*33f37583SAndroid Build Coastguard Worker     if (auto result = repo.GetPartition(file); result.ok()) {
119*33f37583SAndroid Build Coastguard Worker       info.partition = result.value();
120*33f37583SAndroid Build Coastguard Worker     } else {
121*33f37583SAndroid Build Coastguard Worker       LOG(WARNING) << "Failed to get partition of " << file_path << ": "
122*33f37583SAndroid Build Coastguard Worker                    << result.error();
123*33f37583SAndroid Build Coastguard Worker       continue;
124*33f37583SAndroid Build Coastguard Worker     }
125*33f37583SAndroid Build Coastguard Worker 
126*33f37583SAndroid Build Coastguard Worker     files_.push_back(std::move(info));
127*33f37583SAndroid Build Coastguard Worker   }
128*33f37583SAndroid Build Coastguard Worker }
129*33f37583SAndroid Build Coastguard Worker 
AddHals(const std::map<std::string,std::vector<std::string>> & hals)130*33f37583SAndroid Build Coastguard Worker void InstallRequestedEvent::AddHals(
131*33f37583SAndroid Build Coastguard Worker     const std::map<std::string, std::vector<std::string>>& hals) {
132*33f37583SAndroid Build Coastguard Worker   for (auto& info : files_) {
133*33f37583SAndroid Build Coastguard Worker     if (auto it = hals.find(info.name); it != hals.end()) {
134*33f37583SAndroid Build Coastguard Worker       info.hals = it->second;
135*33f37583SAndroid Build Coastguard Worker     }
136*33f37583SAndroid Build Coastguard Worker   }
137*33f37583SAndroid Build Coastguard Worker }
138*33f37583SAndroid Build Coastguard Worker 
GetFileHashes() const139*33f37583SAndroid Build Coastguard Worker std::vector<std::string> InstallRequestedEvent::GetFileHashes() const {
140*33f37583SAndroid Build Coastguard Worker   std::vector<std::string> hashes;
141*33f37583SAndroid Build Coastguard Worker   hashes.reserve(files_.size());
142*33f37583SAndroid Build Coastguard Worker   for (const auto& info : files_) {
143*33f37583SAndroid Build Coastguard Worker     hashes.push_back(info.file_hash);
144*33f37583SAndroid Build Coastguard Worker   }
145*33f37583SAndroid Build Coastguard Worker   return hashes;
146*33f37583SAndroid Build Coastguard Worker }
147*33f37583SAndroid Build Coastguard Worker 
148*33f37583SAndroid Build Coastguard Worker }  // namespace android::apex
149