1#![doc = include_str!("../README.md")]
2
3extern crate alloc;
4
5const PKG_VERSION: &str = if let Some(version) = option_env!("CARGO_PKG_VERSION") {
9 version
10} else {
11 ""
12};
13
14pub(crate) const MAX_CONTROL_RESPONSE: usize = 1024 * 1024;
24
25#[cfg(feature = "acme")]
27pub mod acme;
28#[cfg(feature = "async_tokio")]
29mod cert;
30mod config;
31mod control_dialer;
32mod derp;
33mod dial_plan;
34mod dns;
35mod hostinfo;
36#[cfg_attr(not(feature = "async_tokio"), expect(dead_code))]
37mod map_request_builder;
38mod node;
39#[cfg(feature = "async_tokio")]
40mod serve;
41mod service;
42mod ssh_policy;
43mod tka;
44#[cfg(feature = "async_tokio")]
45mod tokio;
46#[cfg(feature = "identity-federation")]
47pub mod wif;
48
49use std::fmt;
50
51#[cfg(feature = "async_tokio")]
52pub use cert::{
53 CertError, MISSING_CERT_RPC, certified_key_from_pem, get_certificate, is_tailnet_name,
54};
55#[cfg(feature = "acme")]
56pub use cert::{
57 PublishTxt, SetDnsPublisher, issue_cert_pair_via_setdns, issue_certificate_via_setdns,
58};
59#[doc(inline)]
60pub use config::{
61 Config, DEFAULT_CONTROL_SERVER, DEFAULT_PERSISTENT_KEEPALIVE, ExitProxyConfig, ExitProxyScheme,
62 TransportMode, TunConfig, services_hash,
63};
64pub use control_dialer::{ControlDialer, TcpDialer, complete_connection};
65pub use derp::{Map as DerpMap, Region as DerpRegion, convert_derp_map};
66pub use dial_plan::{DialCandidate, DialMode, DialPlan};
67pub use dns::{DnsConfig, ExtraRecord, Resolver as DnsResolver, ResolverTransport};
68pub use node::{
69 ExitNodeSelector, Id as NodeId, Node, NodeCapMap, PeerChange, StableId as StableNodeId,
70 TailnetAddress, UserProfile, is_tailscale_ip, validate_service_name,
71};
72#[cfg(feature = "async_tokio")]
73pub use serve::{
74 FunnelError, FunnelOptions, MISSING_FUNNEL_RELAY, ServeConfig, ServeState, ServeTarget,
75 accept_tls, funnel_access, listen_funnel, listen_tls, tls_acceptor,
76};
77pub use service::{ServiceError, ServiceMode, resolve_service_listen};
78pub use ssh_policy::{
79 SshAccept, SshAction, SshConnIdentity, SshDecision, SshDenyReason, SshPolicy, SshPrincipal,
80 SshRule,
81};
82pub use tka::TkaStatus;
83pub use ts_control_serde::{
84 Endpoint, EndpointType, NODE_ATTR_SUGGEST_EXIT_NODE, TkaBootstrapRequest, TkaBootstrapResponse,
85 TkaDisableRequest, TkaDisableResponse, TkaInitBeginRequest, TkaInitBeginResponse,
86 TkaInitFinishRequest, TkaInitFinishResponse, TkaSignInfo, TkaSubmitSignatureRequest,
87 TkaSubmitSignatureResponse, TkaSyncOfferRequest, TkaSyncOfferResponse, TkaSyncSendRequest,
88 TkaSyncSendResponse, UserId,
89};
90#[cfg(feature = "identity-federation")]
91pub use wif::{WifConfig, WifError, resolve_auth_key};
92
93#[cfg(feature = "async_tokio")]
97pub mod tls {
98 pub use tokio_rustls::{TlsAcceptor, rustls::sign::CertifiedKey, server::TlsStream};
99}
100
101#[cfg(feature = "async_tokio")]
102pub use crate::tokio::{
103 AsyncControlClient, FilterUpdate, IdTokenError, LogoutError, LogoutInternalErrorKind,
104 PeerUpdate, SetDnsError, SetDnsInternalErrorKind, StateUpdate, TkaSyncError,
105 TkaSyncInternalErrorKind, fetch_id_token, logout, set_dns, tka_bootstrap, tka_disable,
106 tka_init_begin, tka_init_finish, tka_submit_signature, tka_sync_offer, tka_sync_send,
107};
108
109#[derive(Debug, thiserror::Error, Clone, Eq, PartialEq)]
111pub enum Error {
112 #[error("machine was not authorized by control to join tailnet, authorize at {0}")]
114 MachineNotAuthorized(url::Url),
115
116 #[error("machine awaiting admin approval to join tailnet (no interactive auth URL)")]
124 NeedsMachineAuth,
125
126 #[error("invalid URL: {0}")]
128 InvalidUrl(url::Url),
129
130 #[error("control rejected registration: {0}")]
133 Registration(String),
134
135 #[error("control rate limited the request; retry after {0:?}")]
139 RateLimited(core::time::Duration),
140
141 #[error("a networking error occurred in {0}")]
147 NetworkError(Operation),
148
149 #[error("{0} error in {1}")]
154 Internal(InternalErrorKind, Operation),
155}
156
157impl Error {
158 fn io_error(err: std::io::Error, op: Operation) -> Self {
159 if crate::is_network_error(&err) {
160 Error::NetworkError(op)
161 } else {
162 Error::Internal(InternalErrorKind::Io, op)
163 }
164 }
165}
166
167#[non_exhaustive]
171#[derive(Debug, Clone, Copy, Eq, PartialEq)]
172pub enum InternalErrorKind {
173 Url,
175 Http,
177 SerDe,
179 Io,
181 MessageFormat,
183 Utf8,
185 NoiseHandshake,
187 Challenge,
189 MachineAuthorization,
192}
193
194impl fmt::Display for InternalErrorKind {
195 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
196 match self {
197 InternalErrorKind::Url => write!(f, "URL parsing error"),
198 InternalErrorKind::Http => write!(f, "unsuccessful HTTP request or upgrade"),
199 InternalErrorKind::SerDe => write!(f, "serialization/deserialization error"),
200 InternalErrorKind::Io => write!(f, "I/O error"),
201 InternalErrorKind::MessageFormat => write!(f, "message format error"),
202 InternalErrorKind::Utf8 => write!(f, "invalid UTF8"),
203 InternalErrorKind::NoiseHandshake => write!(f, "error in Noise handshake"),
204 InternalErrorKind::Challenge => write!(f, "error with Tailscale challenge packet"),
205 InternalErrorKind::MachineAuthorization => {
206 write!(f, "machine not authorized to register with Tailnet")
207 }
208 }
209 }
210}
211
212#[derive(Debug, Clone, Copy, Eq, PartialEq)]
214pub enum Operation {
215 MapRequest,
217 ConnectToControlServer,
219 Registration,
221}
222
223impl fmt::Display for Operation {
224 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
225 match self {
226 Operation::MapRequest => write!(f, "net map request"),
227 Operation::ConnectToControlServer => write!(f, "connection to control server"),
228 Operation::Registration => write!(f, "registration"),
229 }
230 }
231}
232
233impl From<ts_http_util::Error> for Error {
234 fn from(error: ts_http_util::Error) -> Self {
235 tracing::error!(%error, "http error");
236
237 if http_error_is_recoverable(error) {
238 Error::NetworkError(Operation::ConnectToControlServer)
239 } else {
240 Error::Internal(InternalErrorKind::Http, Operation::ConnectToControlServer)
241 }
242 }
243}
244
245fn is_network_error(err: &std::io::Error) -> bool {
247 use std::io::ErrorKind::*;
248 matches!(
249 err.kind(),
250 ConnectionRefused
251 | ConnectionReset
252 | HostUnreachable
253 | NetworkUnreachable
254 | ConnectionAborted
255 | NotConnected
256 | TimedOut
257 | AddrNotAvailable
258 | Interrupted
259 | NetworkDown
260 )
261}
262
263fn http_error_is_recoverable(error: ts_http_util::Error) -> bool {
265 match error {
266 ts_http_util::Error::Io => true,
267 ts_http_util::Error::InvalidInput
268 | ts_http_util::Error::Timeout
271 | ts_http_util::Error::InvalidResponse
272 | ts_http_util::Error::BodyTooLarge => false,
275 ts_http_util::Error::ConnectionClosed => false,
277 }
278}