blob: 67dd18f5f2d70f4f16cc7135dcb19ce5b23cd3e4 [file] [log] [blame]
/*
* Copyright 2018 The WebRTC Project Authors. All rights reserved.
*
* Use of this source code is governed by a BSD-style license
* that can be found in the LICENSE file in the root of the source
* tree. An additional intellectual property rights grant can be found
* in the file PATENTS. All contributing project authors may
* be found in the AUTHORS file in the root of the source tree.
*/
#include "p2p/base/basic_async_resolver_factory.h"
#include <memory>
#include <utility>
#include "absl/memory/memory.h"
#include "api/async_dns_resolver.h"
#include "rtc_base/async_resolver.h"
namespace webrtc {
rtc::AsyncResolverInterface* BasicAsyncResolverFactory::Create() {
return new rtc::AsyncResolver();
}
class WrappingAsyncDnsResolver;
class WrappingAsyncDnsResolverResult : public AsyncDnsResolverResult {
public:
explicit WrappingAsyncDnsResolverResult(WrappingAsyncDnsResolver* owner)
: owner_(owner) {}
~WrappingAsyncDnsResolverResult() {}
// Note: Inline declaration not possible, since it refers to
// WrappingAsyncDnsResolver.
bool GetResolvedAddress(int family, rtc::SocketAddress* addr) const override;
int GetError() const override;
private:
WrappingAsyncDnsResolver* const owner_;
};
class WrappingAsyncDnsResolver : public AsyncDnsResolverInterface,
public sigslot::has_slots<> {
public:
explicit WrappingAsyncDnsResolver(rtc::AsyncResolverInterface* wrapped)
: wrapped_(absl::WrapUnique(wrapped)), result_(this) {}
~WrappingAsyncDnsResolver() override { wrapped_.release()->Destroy(false); }
void Start(const rtc::SocketAddress& addr,
std::function<void()> callback) override {
RTC_DCHECK(state_ == State::kNotStarted);
state_ = State::kStarted;
callback_ = callback;
wrapped_->SignalDone.connect(this,
&WrappingAsyncDnsResolver::OnResolveResult);
wrapped_->Start(addr);
}
const AsyncDnsResolverResult& result() const override {
RTC_DCHECK(state_ == State::kResolved);
return result_;
}
// For use by WrappingAsyncDnsResolverResult
rtc::AsyncResolverInterface* wrapped() const { return wrapped_.get(); }
private:
enum class State { kNotStarted, kStarted, kResolved };
void OnResolveResult(rtc::AsyncResolverInterface* ref) {
RTC_DCHECK(state_ == State::kStarted);
RTC_DCHECK_EQ(ref, wrapped_.get());
state_ = State::kResolved;
callback_();
}
std::function<void()> callback_;
std::unique_ptr<rtc::AsyncResolverInterface> wrapped_;
State state_ = State::kNotStarted;
WrappingAsyncDnsResolverResult result_;
};
bool WrappingAsyncDnsResolverResult::GetResolvedAddress(
int family,
rtc::SocketAddress* addr) const {
if (!owner_->wrapped()) {
return false;
}
return owner_->wrapped()->GetResolvedAddress(family, addr);
}
int WrappingAsyncDnsResolverResult::GetError() const {
if (!owner_->wrapped()) {
return -1; // FIXME: Find a code that makes sense.
}
return owner_->wrapped()->GetError();
}
std::unique_ptr<webrtc::AsyncDnsResolverInterface>
WrappingAsyncDnsResolverFactory::Create() {
return std::make_unique<WrappingAsyncDnsResolver>(wrapped_factory_->Create());
}
std::unique_ptr<webrtc::AsyncDnsResolverInterface>
WrappingAsyncDnsResolverFactory::CreateAndResolve(
const rtc::SocketAddress& addr,
std::function<void()> callback) {
std::unique_ptr<webrtc::AsyncDnsResolverInterface> resolver = Create();
resolver->Start(addr, callback);
return resolver;
}
} // namespace webrtc