blob: de01d5534ead135e0d36d416cb78f30c9b8b85ce [file] [log] [blame]
/*
* Copyright (c) 2021, The OpenThread Authors.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of the copyright holder nor the
* names of its contributors may be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef DNS_SERVER_HPP_
#define DNS_SERVER_HPP_
#include "openthread-core-config.h"
#if OPENTHREAD_CONFIG_DNSSD_SERVER_ENABLE
#include <openthread/dnssd_server.h>
#include "common/as_core_type.hpp"
#include "common/callback.hpp"
#include "common/message.hpp"
#include "common/non_copyable.hpp"
#include "common/owned_ptr.hpp"
#include "common/timer.hpp"
#include "net/dns_types.hpp"
#include "net/ip6.hpp"
#include "net/netif.hpp"
#include "net/srp_server.hpp"
/**
* @file
* This file includes definitions for the DNS-SD server.
*/
struct otPlatDnsUpstreamQuery
{
};
namespace ot {
namespace Srp {
class Server;
}
namespace Dns {
namespace ServiceDiscovery {
/**
* Implements DNS-SD server.
*
*/
class Server : public InstanceLocator, private NonCopyable
{
friend class Srp::Server;
public:
/**
* Contains the counters of the DNS-SD server.
*
*/
class Counters : public otDnssdCounters, public Clearable<Counters>
{
};
#if OPENTHREAD_CONFIG_DNS_UPSTREAM_QUERY_ENABLE
/**
* Represents an upstream query transaction. The methods should only be used by
* `Dns::ServiceDiscovery::Server`.
*
*/
class UpstreamQueryTransaction : public otPlatDnsUpstreamQuery
{
public:
/**
* Returns whether the transaction is valid.
*
* @retval TRUE The transaction is valid.
* @retval FALSE The transaction is not valid.
*
*/
bool IsValid(void) const { return mValid; }
/**
* Returns the time when the transaction expires.
*
* @returns The expire time of the transaction.
*
*/
TimeMilli GetExpireTime(void) const { return mExpireTime; }
/**
* Resets the transaction with a reason. The transaction will be invalid and can be reused for
* another upstream query after this call.
*
*/
void Reset(void) { mValid = false; }
/**
* Initializes the transaction.
*
* @param[in] aMessageInfo The IP message info of the query.
*
*/
void Init(const Ip6::MessageInfo &aMessageInfo);
/**
* Returns the message info of the query.
*
* @returns The message info of the query.
*
*/
const Ip6::MessageInfo &GetMessageInfo(void) const { return mMessageInfo; }
private:
Ip6::MessageInfo mMessageInfo;
TimeMilli mExpireTime;
bool mValid;
};
#endif
/**
* Specifies a DNS-SD query type.
*
*/
enum DnsQueryType : uint8_t
{
kDnsQueryNone = OT_DNSSD_QUERY_TYPE_NONE, ///< Service type unspecified.
kDnsQueryBrowse = OT_DNSSD_QUERY_TYPE_BROWSE, ///< Service type browse service.
kDnsQueryResolve = OT_DNSSD_QUERY_TYPE_RESOLVE, ///< Service type resolve service instance.
kDnsQueryResolveHost = OT_DNSSD_QUERY_TYPE_RESOLVE_HOST, ///< Service type resolve hostname.
};
typedef otDnssdServiceInstanceInfo ServiceInstanceInfo; ///< A discovered service instance for a DNS-SD query.
typedef otDnssdHostInfo HostInfo; ///< A discover host for a DNS-SD query.
typedef otDnssdQuerySubscribeCallback SubscribeCallback;
typedef otDnssdQueryUnsubscribeCallback UnsubscribeCallback;
static constexpr uint16_t kPort = OPENTHREAD_CONFIG_DNSSD_SERVER_PORT; ///< The DNS-SD server port.
/**
* Initializes the object.
*
* @param[in] aInstance A reference to the OpenThread instance.
*
*/
explicit Server(Instance &aInstance);
/**
* Starts the DNS-SD server.
*
* @retval kErrorNone Successfully started the DNS-SD server.
* @retval kErrorFailed If failed to open or bind the UDP socket.
*
*/
Error Start(void);
/**
* Stops the DNS-SD server.
*
*/
void Stop(void);
/**
* Sets DNS-SD query callbacks.
*
* @param[in] aSubscribe A pointer to the callback function to subscribe a service or service instance.
* @param[in] aUnsubscribe A pointer to the callback function to unsubscribe a service or service instance.
* @param[in] aContext A pointer to the application-specific context.
*
*/
void SetQueryCallbacks(SubscribeCallback aSubscribe, UnsubscribeCallback aUnsubscribe, void *aContext);
/**
* Notifies a discovered service instance.
*
* @param[in] aServiceFullName The null-terminated full service name.
* @param[in] aInstanceInfo A reference to the discovered service instance information.
*
*/
void HandleDiscoveredServiceInstance(const char *aServiceFullName, const ServiceInstanceInfo &aInstanceInfo);
#if OPENTHREAD_CONFIG_DNS_UPSTREAM_QUERY_ENABLE
/**
* Notifies an answer of an upstream DNS query.
*
* The Transaction will be released.
*
* @param[in] aQueryTransaction A reference to upstream DNS query transaction.
* @param[in] aResponseMessage A pointer to response UDP message, should be allocated from Udp::NewMessage.
* Passing a nullptr means close the transaction without a response.
*
*/
void OnUpstreamQueryDone(UpstreamQueryTransaction &aQueryTransaction, Message *aResponseMessage);
/**
* Indicates whether the server will forward DNS queries to platform DNS upstream API.
*
* @retval TRUE If the server will forward DNS queries.
* @retval FALSE If the server will not forward DNS queries.
*
*/
bool IsUpstreamQueryEnabled(void) const { return mEnableUpstreamQuery; }
/**
* Enables or disables forwarding DNS queries to platform DNS upstream API.
*
* @param[in] aEnabled A boolean to enable/disable forwarding DNS queries to upstream.
*
*/
void SetUpstreamQueryEnabled(bool aEnabled) { mEnableUpstreamQuery = aEnabled; }
#endif // OPENTHREAD_CONFIG_DNS_UPSTREAM_QUERY_ENABLE
/**
* Notifies a discovered host.
*
* @param[in] aHostFullName The null-terminated full host name.
* @param[in] aHostInfo A reference to the discovered host information.
*
*/
void HandleDiscoveredHost(const char *aHostFullName, const HostInfo &aHostInfo);
/**
* Acquires the next query in the server.
*
* @param[in] aQuery The query pointer. Pass `nullptr` to get the first query.
*
* @returns A pointer to the query or `nullptr` if no more queries.
*
*/
const otDnssdQuery *GetNextQuery(const otDnssdQuery *aQuery) const;
/**
* Acquires the DNS-SD query type and name for a specific query.
*
* @param[in] aQuery The query pointer.
* @param[out] aName The name output buffer.
*
* @returns The DNS-SD query type.
*
*/
static DnsQueryType GetQueryTypeAndName(const otDnssdQuery *aQuery, char (&aName)[Name::kMaxNameSize]);
/**
* Returns the counters of the DNS-SD server.
*
* @returns A reference to the `Counters` instance.
*
*/
const Counters &GetCounters(void) const { return mCounters; };
/**
* Represents different test mode flags for use in `SetTestMode()`.
*
*/
enum TestModeFlags : uint8_t
{
kTestModeSingleQuestionOnly = 1 << 0, ///< Allow single question in query, send `FormatError` otherwise.
kTestModeEmptyAdditionalSection = 1 << 1, ///< Do not include any RR in additional section.
};
static constexpr uint8_t kTestModeDisabled = 0; ///< Test mode is disabled (no flags).
/**
* Sets the test mode for `Server`.
*
* The test mode flags are intended for testing the client by having server behave in certain ways, e.g., reject
* messages with certain format (e.g., more than one question in query).
*
* @param[in] aTestMode The new test mode (combination of `TestModeFlags`).
*
*/
void SetTestMode(uint8_t aTestMode) { mTestMode = aTestMode; }
private:
static constexpr bool kBindUnspecifiedNetif = OPENTHREAD_CONFIG_DNSSD_SERVER_BIND_UNSPECIFIED_NETIF;
static constexpr uint32_t kQueryTimeout = OPENTHREAD_CONFIG_DNSSD_QUERY_TIMEOUT;
static constexpr uint16_t kMaxConcurrentUpstreamQueries = 32;
typedef Header::Response ResponseCode;
typedef char DnsName[Name::kMaxNameSize];
typedef char DnsLabel[Name::kMaxLabelSize];
typedef Message ProxyQuery;
typedef MessageQueue ProxyQueryList;
enum QueryType : uint8_t
{
kPtrQuery,
kSrvQuery,
kTxtQuery,
kSrvTxtQuery,
kAaaaQuery,
};
enum Section : uint8_t
{
kAnswerSection,
kAdditionalDataSection,
};
struct Request
{
ResponseCode ParseQuestions(uint8_t aTestMode);
const Message *mMessage;
const Ip6::MessageInfo *mMessageInfo;
Header mHeader;
QueryType mType;
};
struct ProxyQueryInfo;
struct NameOffsets : public Clearable<NameOffsets>
{
uint16_t mDomainName;
uint16_t mServiceName;
uint16_t mInstanceName;
uint16_t mHostName;
};
class Response : public InstanceLocator, private NonCopyable
{
public:
explicit Response(Instance &aInstance);
Error AllocateAndInitFrom(const Request &aRequest);
void InitFrom(ProxyQuery &aQuery, const ProxyQueryInfo &aInfo);
void SetResponseCode(ResponseCode aResponseCode) { mHeader.SetResponseCode(aResponseCode); }
ResponseCode AddQuestionsFrom(const Request &aRequest);
Error ParseQueryName(void);
void ReadQueryName(DnsName &aName) const;
bool QueryNameMatches(const char *aName) const;
Error AppendQueryName(void);
Error AppendPtrRecord(const char *aInstanceLabel, uint32_t aTtl);
Error AppendSrvRecord(const ServiceInstanceInfo &aInstanceInfo);
Error AppendSrvRecord(const char *aHostName,
uint32_t aTtl,
uint16_t aPriority,
uint16_t aWeight,
uint16_t aPort);
Error AppendTxtRecord(const ServiceInstanceInfo &aInstanceInfo);
Error AppendTxtRecord(const void *aTxtData, uint16_t aTxtLength, uint32_t aTtl);
Error AppendHostAddresses(const HostInfo &aHostInfo);
Error AppendHostAddresses(const ServiceInstanceInfo &aInstanceInfo);
Error AppendHostAddresses(const Ip6::Address *aAddrs, uint16_t aAddrsLength, uint32_t aTtl);
void UpdateRecordLength(ResourceRecord &aRecord, uint16_t aOffset);
void IncResourceRecordCount(void);
void Send(const Ip6::MessageInfo &aMessageInfo);
void Answer(const HostInfo &aHostInfo, const Ip6::MessageInfo &aMessageInfo);
void Answer(const ServiceInstanceInfo &aInstanceInfo, const Ip6::MessageInfo &aMessageInfo);
Error ExtractServiceInstanceLabel(const char *aInstanceName, DnsLabel &aLabel);
#if OPENTHREAD_CONFIG_SRP_SERVER_ENABLE
Error ResolveBySrp(void);
bool QueryNameMatchesService(const Srp::Server::Service &aService) const;
Error AppendSrvRecord(const Srp::Server::Service &aService);
Error AppendTxtRecord(const Srp::Server::Service &aService);
Error AppendHostAddresses(const Srp::Server::Host &aHost);
#endif
#if OT_SHOULD_LOG_AT(OT_LOG_LEVEL_INFO)
void Log(void) const;
static const char *QueryTypeToString(QueryType aType);
#endif
OwnedPtr<Message> mMessage;
Header mHeader;
QueryType mType;
Section mSection;
NameOffsets mOffsets;
};
struct ProxyQueryInfo
{
void ReadFrom(const ProxyQuery &aQuery);
void RemoveFrom(ProxyQuery &aQuery) const;
void UpdateIn(ProxyQuery &aQuery) const;
QueryType mType;
Ip6::MessageInfo mMessageInfo;
TimeMilli mExpireTime;
NameOffsets mOffsets;
};
bool IsRunning(void) const { return mSocket.IsBound(); }
static void HandleUdpReceive(void *aContext, otMessage *aMessage, const otMessageInfo *aMessageInfo);
void HandleUdpReceive(Message &aMessage, const Ip6::MessageInfo &aMessageInfo);
void ProcessQuery(Request &aRequest);
static uint8_t GetNameLength(const char *aName);
void ResolveByProxy(Response &aResponse, const Ip6::MessageInfo &aMessageInfo);
void RemoveQueryAndPrepareResponse(ProxyQuery &aQuery, const ProxyQueryInfo &aInfo, Response &aResponse);
void Finalize(ProxyQuery &aQuery, ResponseCode aResponseCode);
static void ReadQueryName(const Message &aQuery, DnsName &aName);
static bool QueryNameMatches(const Message &aQuery, const char *aName);
#if OPENTHREAD_CONFIG_DNS_UPSTREAM_QUERY_ENABLE
static bool ShouldForwardToUpstream(const Request &aRequest);
UpstreamQueryTransaction *AllocateUpstreamQueryTransaction(const Ip6::MessageInfo &aMessageInfo);
void ResetUpstreamQueryTransaction(UpstreamQueryTransaction &aTxn, Error aError);
Error ResolveByUpstream(const Request &aRequest);
#endif
void HandleTimer(void);
void ResetTimer(void);
void UpdateResponseCounters(ResponseCode aResponseCode);
using ServerTimer = TimerMilliIn<Server, &Server::HandleTimer>;
static const char kDefaultDomainName[];
static const char kSubLabel[];
#if OPENTHREAD_CONFIG_DNS_UPSTREAM_QUERY_ENABLE
static const char *kBlockedDomains[];
#endif
Ip6::Udp::Socket mSocket;
ProxyQueryList mProxyQueries;
Callback<SubscribeCallback> mQuerySubscribe;
Callback<UnsubscribeCallback> mQueryUnsubscribe;
#if OPENTHREAD_CONFIG_DNS_UPSTREAM_QUERY_ENABLE
bool mEnableUpstreamQuery;
UpstreamQueryTransaction mUpstreamQueryTransactions[kMaxConcurrentUpstreamQueries];
#endif
ServerTimer mTimer;
Counters mCounters;
uint8_t mTestMode;
};
} // namespace ServiceDiscovery
} // namespace Dns
DefineMapEnum(otDnssdQueryType, Dns::ServiceDiscovery::Server::DnsQueryType);
#if OPENTHREAD_CONFIG_DNS_UPSTREAM_QUERY_ENABLE
DefineCoreType(otPlatDnsUpstreamQuery, Dns::ServiceDiscovery::Server::UpstreamQueryTransaction);
#endif
} // namespace ot
#endif // OPENTHREAD_CONFIG_DNSSD_SERVER_ENABLE
#endif // DNS_SERVER_HPP_