1*ec63e07aSXin Li // Copyright 2022 Google LLC
2*ec63e07aSXin Li //
3*ec63e07aSXin Li // Licensed under the Apache License, Version 2.0 (the "License");
4*ec63e07aSXin Li // you may not use this file except in compliance with the License.
5*ec63e07aSXin Li // You may obtain a copy of the License at
6*ec63e07aSXin Li //
7*ec63e07aSXin Li // https://www.apache.org/licenses/LICENSE-2.0
8*ec63e07aSXin Li //
9*ec63e07aSXin Li // Unless required by applicable law or agreed to in writing, software
10*ec63e07aSXin Li // distributed under the License is distributed on an "AS IS" BASIS,
11*ec63e07aSXin Li // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*ec63e07aSXin Li // See the License for the specific language governing permissions and
13*ec63e07aSXin Li // limitations under the License.
14*ec63e07aSXin Li
15*ec63e07aSXin Li #include <fcntl.h>
16*ec63e07aSXin Li #include <unistd.h>
17*ec63e07aSXin Li
18*ec63e07aSXin Li #include <cstdlib>
19*ec63e07aSXin Li #include <fstream>
20*ec63e07aSXin Li #include <iostream>
21*ec63e07aSXin Li #include <string>
22*ec63e07aSXin Li #include <vector>
23*ec63e07aSXin Li
24*ec63e07aSXin Li #include "absl/flags/flag.h"
25*ec63e07aSXin Li #include "absl/flags/parse.h"
26*ec63e07aSXin Li #include "absl/log/globals.h"
27*ec63e07aSXin Li #include "absl/log/initialize.h"
28*ec63e07aSXin Li #include "contrib/libzip/sandboxed.h"
29*ec63e07aSXin Li #include "contrib/libzip/utils/utils_zip.h"
30*ec63e07aSXin Li
31*ec63e07aSXin Li ABSL_FLAG(bool, list, false, "list files");
32*ec63e07aSXin Li ABSL_FLAG(std::string, unzip, "", "unzip");
33*ec63e07aSXin Li ABSL_FLAG(std::string, add_file, "", "add file");
34*ec63e07aSXin Li ABSL_FLAG(std::string, delete, "", "delete");
35*ec63e07aSXin Li
ListFiles(LibZip & zip)36*ec63e07aSXin Li absl::Status ListFiles(LibZip& zip) {
37*ec63e07aSXin Li SAPI_ASSIGN_OR_RETURN(int64_t num, zip.GetNumberEntries());
38*ec63e07aSXin Li for (uint64_t i = 0; i < num; i++) {
39*ec63e07aSXin Li SAPI_ASSIGN_OR_RETURN(std::string name, zip.GetName(i));
40*ec63e07aSXin Li std::cout << name << "\n";
41*ec63e07aSXin Li }
42*ec63e07aSXin Li
43*ec63e07aSXin Li return absl::OkStatus();
44*ec63e07aSXin Li }
45*ec63e07aSXin Li
UnzipToStdout(LibZip & zip,const std::string & filename)46*ec63e07aSXin Li absl::Status UnzipToStdout(LibZip& zip, const std::string& filename) {
47*ec63e07aSXin Li SAPI_ASSIGN_OR_RETURN(std::vector<uint8_t> buf, zip.ReadFile(filename));
48*ec63e07aSXin Li
49*ec63e07aSXin Li std::cout << buf.data();
50*ec63e07aSXin Li
51*ec63e07aSXin Li return absl::OkStatus();
52*ec63e07aSXin Li }
53*ec63e07aSXin Li
AddFile(LibZip & zip,const std::string & filename)54*ec63e07aSXin Li absl::Status AddFile(LibZip& zip, const std::string& filename) {
55*ec63e07aSXin Li int fd = open(filename.c_str(), O_RDONLY);
56*ec63e07aSXin Li if (fd < 0) {
57*ec63e07aSXin Li return absl::UnavailableError(
58*ec63e07aSXin Li absl::StrCat("Unable to open file ", filename));
59*ec63e07aSXin Li }
60*ec63e07aSXin Li
61*ec63e07aSXin Li // The fd will be consumed
62*ec63e07aSXin Li SAPI_ASSIGN_OR_RETURN(uint64_t index, zip.AddFile(filename, fd));
63*ec63e07aSXin Li
64*ec63e07aSXin Li return absl::OkStatus();
65*ec63e07aSXin Li }
66*ec63e07aSXin Li
DeleteFile(LibZip & zip,const std::string & filename)67*ec63e07aSXin Li absl::Status DeleteFile(LibZip& zip, const std::string& filename) {
68*ec63e07aSXin Li int64_t index = -1;
69*ec63e07aSXin Li
70*ec63e07aSXin Li SAPI_ASSIGN_OR_RETURN(int64_t num, zip.GetNumberEntries());
71*ec63e07aSXin Li for (uint64_t i = 0; i < num; i++) {
72*ec63e07aSXin Li SAPI_ASSIGN_OR_RETURN(std::string name, zip.GetName(i));
73*ec63e07aSXin Li if (name == filename) {
74*ec63e07aSXin Li index = i;
75*ec63e07aSXin Li break;
76*ec63e07aSXin Li }
77*ec63e07aSXin Li }
78*ec63e07aSXin Li if (index == -1) {
79*ec63e07aSXin Li return absl::UnavailableError(
80*ec63e07aSXin Li absl::StrCat("Unable to remove file ", filename));
81*ec63e07aSXin Li }
82*ec63e07aSXin Li
83*ec63e07aSXin Li return zip.DeleteFile(index);
84*ec63e07aSXin Li }
85*ec63e07aSXin Li
main(int argc,char * argv[])86*ec63e07aSXin Li int main(int argc, char* argv[]) {
87*ec63e07aSXin Li std::string prog_name(argv[0]);
88*ec63e07aSXin Li absl::SetStderrThreshold(absl::LogSeverityAtLeast::kInfo);
89*ec63e07aSXin Li std::vector<char*> args = absl::ParseCommandLine(argc, argv);
90*ec63e07aSXin Li absl::InitializeLog();
91*ec63e07aSXin Li
92*ec63e07aSXin Li if (args.size() < 2 || args.size() > 3) {
93*ec63e07aSXin Li std::cerr << "Usage:\n " << prog_name << " ZIPFILE [OUTFILE]\n";
94*ec63e07aSXin Li return EXIT_FAILURE;
95*ec63e07aSXin Li }
96*ec63e07aSXin Li
97*ec63e07aSXin Li std::string filename(args[1]);
98*ec63e07aSXin Li ZipSapiSandbox sandbox;
99*ec63e07aSXin Li if (!sandbox.Init().ok()) {
100*ec63e07aSXin Li std::cerr << "Unable to start sandbox\n";
101*ec63e07aSXin Li return EXIT_FAILURE;
102*ec63e07aSXin Li }
103*ec63e07aSXin Li
104*ec63e07aSXin Li LibZip zip(&sandbox, filename, 0);
105*ec63e07aSXin Li
106*ec63e07aSXin Li if (!zip.IsOpen()) {
107*ec63e07aSXin Li std::cerr << "Unable to open file " << filename << "\n";
108*ec63e07aSXin Li return EXIT_FAILURE;
109*ec63e07aSXin Li }
110*ec63e07aSXin Li
111*ec63e07aSXin Li int outfd = -1;
112*ec63e07aSXin Li if (args.size() == 3) {
113*ec63e07aSXin Li outfd = open(args[2], O_WRONLY | O_CREAT);
114*ec63e07aSXin Li if (outfd < 0) {
115*ec63e07aSXin Li std::cerr << "Unable to open file " << args[2] << "\n";
116*ec63e07aSXin Li return EXIT_FAILURE;
117*ec63e07aSXin Li }
118*ec63e07aSXin Li }
119*ec63e07aSXin Li
120*ec63e07aSXin Li bool needs_saving = false;
121*ec63e07aSXin Li absl::Status status;
122*ec63e07aSXin Li if (absl::GetFlag(FLAGS_list)) {
123*ec63e07aSXin Li status = ListFiles(zip);
124*ec63e07aSXin Li }
125*ec63e07aSXin Li if (!absl::GetFlag(FLAGS_unzip).empty()) {
126*ec63e07aSXin Li status = UnzipToStdout(zip, absl::GetFlag(FLAGS_unzip));
127*ec63e07aSXin Li }
128*ec63e07aSXin Li if (!absl::GetFlag(FLAGS_add_file).empty()) {
129*ec63e07aSXin Li status = AddFile(zip, absl::GetFlag(FLAGS_add_file));
130*ec63e07aSXin Li needs_saving = true;
131*ec63e07aSXin Li }
132*ec63e07aSXin Li if (!absl::GetFlag(FLAGS_delete).empty()) {
133*ec63e07aSXin Li status = DeleteFile(zip, absl::GetFlag(FLAGS_delete));
134*ec63e07aSXin Li needs_saving = true;
135*ec63e07aSXin Li }
136*ec63e07aSXin Li
137*ec63e07aSXin Li if (!status.ok()) {
138*ec63e07aSXin Li std::cerr << status << "\n";
139*ec63e07aSXin Li return EXIT_FAILURE;
140*ec63e07aSXin Li }
141*ec63e07aSXin Li
142*ec63e07aSXin Li status = zip.Finish();
143*ec63e07aSXin Li if (!status.ok()) {
144*ec63e07aSXin Li std::cerr << status << "\n";
145*ec63e07aSXin Li return EXIT_FAILURE;
146*ec63e07aSXin Li }
147*ec63e07aSXin Li
148*ec63e07aSXin Li if (needs_saving) {
149*ec63e07aSXin Li if (outfd == -1) {
150*ec63e07aSXin Li status = zip.Save();
151*ec63e07aSXin Li } else {
152*ec63e07aSXin Li status = zip.Save(outfd);
153*ec63e07aSXin Li }
154*ec63e07aSXin Li if (!status.ok()) {
155*ec63e07aSXin Li std::cerr << status << "\n";
156*ec63e07aSXin Li return EXIT_FAILURE;
157*ec63e07aSXin Li }
158*ec63e07aSXin Li }
159*ec63e07aSXin Li
160*ec63e07aSXin Li return EXIT_SUCCESS;
161*ec63e07aSXin Li }
162