| // Copyright 2014 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 "lib/app/cpp/application_context.h" |
| #include "lib/app/cpp/connect.h" |
| #include "lib/network/fidl/network_service.fidl.h" |
| #include "lib/network/fidl/url_loader.fidl.h" |
| #include "lib/fxl/macros.h" |
| #include "lib/fsl/tasks/message_loop.h" |
| |
| #include <string> |
| |
| namespace examples { |
| |
| class ResponsePrinter { |
| public: |
| void Run(network::URLResponsePtr response) const { |
| if (response->error) { |
| printf("Got error: %d (%s)\n", response->error->code, |
| response->error->description.get().c_str()); |
| exit(1); |
| } else { |
| PrintResponse(response); |
| PrintResponseBody(std::move(response->body->get_stream())); |
| } |
| |
| fsl::MessageLoop::GetCurrent()->QuitNow(); // All done! |
| } |
| |
| void PrintResponse(const network::URLResponsePtr& response) const { |
| printf(">>> Headers <<< \n"); |
| printf(" %s\n", response->status_line.get().c_str()); |
| if (response->headers) { |
| for (size_t i = 0; i < response->headers.size(); ++i) |
| printf(" %s=%s\n", |
| response->headers[i]->name.To<std::string>().c_str(), |
| response->headers[i]->value.To<std::string>().c_str()); |
| } |
| } |
| |
| void PrintResponseBody(zx::socket body) const { |
| // Read response body in blocking fashion. |
| printf(">>> Body <<<\n"); |
| |
| for (;;) { |
| char buf[512]; |
| size_t num_bytes = sizeof(buf); |
| zx_status_t result = body.read(0u, buf, num_bytes, &num_bytes); |
| |
| if (result == ZX_ERR_SHOULD_WAIT) { |
| body.wait_one(ZX_SOCKET_READABLE | ZX_SOCKET_PEER_CLOSED, |
| zx::time::infinite(), nullptr); |
| } else if (result == ZX_ERR_PEER_CLOSED) { |
| // not an error |
| break; |
| } else if (result == ZX_OK) { |
| if (fwrite(buf, num_bytes, 1, stdout) != 1) { |
| printf("\nUnexpected error writing to file\n"); |
| break; |
| } |
| } else { |
| printf("\nUnexpected error reading response %d\n", result); |
| break; |
| } |
| } |
| |
| printf("\n>>> EOF <<<\n"); |
| } |
| }; |
| |
| class WGetApp { |
| public: |
| WGetApp() : context_(app::ApplicationContext::CreateFromStartupInfo()) { |
| network_service_ = |
| context_->ConnectToEnvironmentService<network::NetworkService>(); |
| FXL_DCHECK(network_service_); |
| } |
| |
| bool Start(const std::vector<std::string>& args) { |
| if (args.size() == 1) { |
| printf("usage: %s url\n", args[0].c_str()); |
| return false; |
| } |
| std::string url(args[1]); |
| if (url.find("://") == std::string::npos) { |
| url.insert(0, "http://"); |
| } |
| printf("Loading: %s\n", url.c_str()); |
| |
| network_service_->CreateURLLoader(url_loader_.NewRequest()); |
| |
| network::URLRequestPtr request(network::URLRequest::New()); |
| request->url = url; |
| request->method = "GET"; |
| request->auto_follow_redirects = true; |
| |
| url_loader_->Start(std::move(request), |
| [this](network::URLResponsePtr response) { |
| ResponsePrinter printer; |
| printer.Run(std::move(response)); |
| }); |
| return true; |
| } |
| |
| private: |
| std::unique_ptr<app::ApplicationContext> context_; |
| |
| network::NetworkServicePtr network_service_; |
| network::URLLoaderPtr url_loader_; |
| }; |
| |
| } // namespace examples |
| |
| int main(int argc, const char** argv) { |
| std::vector<std::string> args(argv, argv + argc); |
| fsl::MessageLoop loop; |
| |
| examples::WGetApp app; |
| if (app.Start(args)) |
| loop.Run(); |
| |
| return 0; |
| } |