blob: 19263de66881924ab4ac36c48c4fd917e159df63 [file] [log] [blame]
Andrew de los Reyes819fef22010-12-17 11:33:58 -08001// Copyright (c) 2010 The Chromium OS Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Alex Vakulenko75039d72014-03-25 12:36:28 -07005#include <base/strings/stringprintf.h>
Andrew de los Reyes819fef22010-12-17 11:33:58 -08006
Alex Vakulenkod2779df2014-06-16 13:19:00 -07007#include <algorithm>
8#include <string>
9
Gilad Arnolde4ad2502011-12-29 17:08:54 -080010#include "update_engine/multi_range_http_fetcher.h"
Andrew de los Reyes819fef22010-12-17 11:33:58 -080011#include "update_engine/utils.h"
12
13namespace chromeos_update_engine {
14
15// Begins the transfer to the specified URL.
16// State change: Stopped -> Downloading
17// (corner case: Stopped -> Stopped for an empty request)
Gilad Arnold9bedeb52011-11-17 16:19:57 -080018void MultiRangeHttpFetcher::BeginTransfer(const std::string& url) {
Andrew de los Reyes819fef22010-12-17 11:33:58 -080019 CHECK(!base_fetcher_active_) << "BeginTransfer but already active.";
20 CHECK(!pending_transfer_ended_) << "BeginTransfer but pending.";
21 CHECK(!terminating_) << "BeginTransfer but terminating.";
22
23 if (ranges_.empty()) {
24 // Note that after the callback returns this object may be destroyed.
25 if (delegate_)
26 delegate_->TransferComplete(this, true);
27 return;
28 }
29 url_ = url;
30 current_index_ = 0;
31 bytes_received_this_range_ = 0;
32 LOG(INFO) << "starting first transfer";
33 base_fetcher_->set_delegate(this);
34 StartTransfer();
35}
36
37// State change: Downloading -> Pending transfer ended
Gilad Arnold9bedeb52011-11-17 16:19:57 -080038void MultiRangeHttpFetcher::TerminateTransfer() {
Andrew de los Reyes819fef22010-12-17 11:33:58 -080039 if (!base_fetcher_active_) {
40 LOG(INFO) << "Called TerminateTransfer but not active.";
41 // Note that after the callback returns this object may be destroyed.
42 if (delegate_)
43 delegate_->TransferTerminated(this);
44 return;
45 }
46 terminating_ = true;
Alex Vakulenkod2779df2014-06-16 13:19:00 -070047
Andrew de los Reyes819fef22010-12-17 11:33:58 -080048 if (!pending_transfer_ended_) {
49 base_fetcher_->TerminateTransfer();
50 }
51}
52
53// State change: Stopped or Downloading -> Downloading
Gilad Arnold9bedeb52011-11-17 16:19:57 -080054void MultiRangeHttpFetcher::StartTransfer() {
Andrew de los Reyes819fef22010-12-17 11:33:58 -080055 if (current_index_ >= ranges_.size()) {
56 return;
57 }
Gilad Arnolde4ad2502011-12-29 17:08:54 -080058
59 Range range = ranges_[current_index_];
60 LOG(INFO) << "starting transfer of range " << range.ToString();
61
Andrew de los Reyes819fef22010-12-17 11:33:58 -080062 bytes_received_this_range_ = 0;
Gilad Arnolde4ad2502011-12-29 17:08:54 -080063 base_fetcher_->SetOffset(range.offset());
64 if (range.HasLength())
65 base_fetcher_->SetLength(range.length());
66 else
67 base_fetcher_->UnsetLength();
Andrew de los Reyes819fef22010-12-17 11:33:58 -080068 if (delegate_)
Gilad Arnolde4ad2502011-12-29 17:08:54 -080069 delegate_->SeekToOffset(range.offset());
Andrew de los Reyes819fef22010-12-17 11:33:58 -080070 base_fetcher_active_ = true;
71 base_fetcher_->BeginTransfer(url_);
72}
73
74// State change: Downloading -> Downloading or Pending transfer ended
Gilad Arnold9bedeb52011-11-17 16:19:57 -080075void MultiRangeHttpFetcher::ReceivedBytes(HttpFetcher* fetcher,
Andrew de los Reyes819fef22010-12-17 11:33:58 -080076 const char* bytes,
77 int length) {
78 CHECK_LT(current_index_, ranges_.size());
79 CHECK_EQ(fetcher, base_fetcher_.get());
80 CHECK(!pending_transfer_ended_);
Gilad Arnolde4ad2502011-12-29 17:08:54 -080081 size_t next_size = length;
82 Range range = ranges_[current_index_];
83 if (range.HasLength()) {
Andrew de los Reyes819fef22010-12-17 11:33:58 -080084 next_size = std::min(next_size,
Gilad Arnolde4ad2502011-12-29 17:08:54 -080085 range.length() - bytes_received_this_range_);
Andrew de los Reyes819fef22010-12-17 11:33:58 -080086 }
87 LOG_IF(WARNING, next_size <= 0) << "Asked to write length <= 0";
88 if (delegate_) {
89 delegate_->ReceivedBytes(this, bytes, next_size);
90 }
91 bytes_received_this_range_ += length;
Gilad Arnolde4ad2502011-12-29 17:08:54 -080092 if (range.HasLength() && bytes_received_this_range_ >= range.length()) {
Andrew de los Reyes819fef22010-12-17 11:33:58 -080093 // Terminates the current fetcher. Waits for its TransferTerminated
94 // callback before starting the next range so that we don't end up
95 // signalling the delegate that the whole multi-transfer is complete
96 // before all fetchers are really done and cleaned up.
97 pending_transfer_ended_ = true;
98 LOG(INFO) << "terminating transfer";
99 fetcher->TerminateTransfer();
100 }
101}
102
103// State change: Downloading or Pending transfer ended -> Stopped
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800104void MultiRangeHttpFetcher::TransferEnded(HttpFetcher* fetcher,
Andrew de los Reyes819fef22010-12-17 11:33:58 -0800105 bool successful) {
106 CHECK(base_fetcher_active_) << "Transfer ended unexpectedly.";
107 CHECK_EQ(fetcher, base_fetcher_.get());
108 pending_transfer_ended_ = false;
109 http_response_code_ = fetcher->http_response_code();
110 LOG(INFO) << "TransferEnded w/ code " << http_response_code_;
111 if (terminating_) {
112 LOG(INFO) << "Terminating.";
113 Reset();
114 // Note that after the callback returns this object may be destroyed.
115 if (delegate_)
116 delegate_->TransferTerminated(this);
117 return;
118 }
119
120 // If we didn't get enough bytes, it's failure
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800121 Range range = ranges_[current_index_];
122 if (range.HasLength()) {
123 if (bytes_received_this_range_ < range.length()) {
Andrew de los Reyes819fef22010-12-17 11:33:58 -0800124 // Failure
125 LOG(INFO) << "Didn't get enough bytes. Ending w/ failure.";
126 Reset();
127 // Note that after the callback returns this object may be destroyed.
128 if (delegate_)
129 delegate_->TransferComplete(this, false);
130 return;
131 }
132 // We got enough bytes and there were bytes specified, so this is success.
133 successful = true;
134 }
135
Andrew de los Reyesffcb6d12011-04-04 09:57:38 -0700136 // If we have another transfer, do that.
Andrew de los Reyes819fef22010-12-17 11:33:58 -0800137 if (current_index_ + 1 < ranges_.size()) {
138 current_index_++;
139 LOG(INFO) << "Starting next transfer (" << current_index_ << ").";
140 StartTransfer();
141 return;
142 }
143
144 LOG(INFO) << "Done w/ all transfers";
145 Reset();
146 // Note that after the callback returns this object may be destroyed.
147 if (delegate_)
148 delegate_->TransferComplete(this, successful);
149}
150
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800151void MultiRangeHttpFetcher::TransferComplete(HttpFetcher* fetcher,
Andrew de los Reyes819fef22010-12-17 11:33:58 -0800152 bool successful) {
153 LOG(INFO) << "Received transfer complete.";
154 TransferEnded(fetcher, successful);
155}
156
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800157void MultiRangeHttpFetcher::TransferTerminated(HttpFetcher* fetcher) {
Andrew de los Reyes819fef22010-12-17 11:33:58 -0800158 LOG(INFO) << "Received transfer terminated.";
159 TransferEnded(fetcher, false);
160}
161
Gilad Arnold9bedeb52011-11-17 16:19:57 -0800162void MultiRangeHttpFetcher::Reset() {
Andrew de los Reyes819fef22010-12-17 11:33:58 -0800163 base_fetcher_active_ = pending_transfer_ended_ = terminating_ = false;
164 current_index_ = 0;
165 bytes_received_this_range_ = 0;
166}
167
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800168std::string MultiRangeHttpFetcher::Range::ToString() const {
Alex Vakulenko75039d72014-03-25 12:36:28 -0700169 std::string range_str = base::StringPrintf("%jd+", offset());
Gilad Arnolde4ad2502011-12-29 17:08:54 -0800170 if (HasLength())
171 base::StringAppendF(&range_str, "%zu", length());
172 else
173 base::StringAppendF(&range_str, "?");
174 return range_str;
175}
176
Andrew de los Reyes819fef22010-12-17 11:33:58 -0800177} // namespace chromeos_update_engine