From 455035ccb865a22702e7e7dd1e67e4ede2504750 Mon Sep 17 00:00:00 2001 From: Evan Pratten Date: Wed, 17 Nov 2021 13:36:13 -0500 Subject: [PATCH] Publish radio linking rfc --- content/rfc/_index.md | 10 ++ content/rfc/modern-repeater-linking.md | 150 +++++++++++++++++++++++++ sass/styles/layout.scss | 5 + templates/base.html | 3 +- templates/page.html | 5 +- templates/section.html | 13 +++ 6 files changed, 184 insertions(+), 2 deletions(-) create mode 100644 content/rfc/_index.md create mode 100644 content/rfc/modern-repeater-linking.md diff --git a/content/rfc/_index.md b/content/rfc/_index.md new file mode 100644 index 0000000..3df5817 --- /dev/null +++ b/content/rfc/_index.md @@ -0,0 +1,10 @@ +--- +title: Proposals +extra: + inject_rfcs: true +--- + +# Proposals & RFCs + +This section contains project proposals that I either lack the time or resources to complete. This is a bit of a public TODO list. + diff --git a/content/rfc/modern-repeater-linking.md b/content/rfc/modern-repeater-linking.md new file mode 100644 index 0000000..2a5c4d9 --- /dev/null +++ b/content/rfc/modern-repeater-linking.md @@ -0,0 +1,150 @@ +--- +title: "RFC: A fast system for repeater linking using modern web technology" +date: 2021-11-17 +extra: + is_rfc: true + uses_graphviz: true +--- + +The three common standards for amateur radio repeater linking are all built on old and cumbersome technology. In the world of repeater linking, the most predominant standards for interconnection are: + +- [Echolink](https://secure.echolink.org/) + - Closed-source + - Strict governance + - A strong lack of APIs + - Does not allow cross-system linking + - ex: cannot link Echolink conferences to Allstar nodes +- [AllStarLink](https://allstarlink.org/) + - Absolute overkill + - Most node configurations require special hardware to set up + - Built on the Asterisk PBX + - Uses phone networks to link repeaters +- [IRLP](http://www.irlp.net/) + - Requires specialized hardware + - Uses old connection standards + - Closed-source + +In my ideal world, there would be a small, lightweight, and configurable linking backend that developers could extend in any way they wished. This proposal is not the end-all-be-all solution, but a step in the direction of this vision. + +## Node discovery + +In a linking system like Echolink, *all* clients announce themselves to a central directory server. This server then decides if a client is "worthy" (verified license, not banned from the service, ...) then lists the client on [a somewhat parsable webpage](https://secure.echolink.org/logins.jsp). The topology for such a setup is as follows: + +
+flowchart LR + You(You) --- D1((Directory Server)) + D1 --- NA(Node A) & NB(Node B) & P1(Proxy) & ND(Node D) + P1 --- NC(Node C) +
+ +In most cases, this works just fine. However, there are two off the top of my head that raise an issue: + +- `Node A` and `Node B` are owned by a club that wants to keep the nodes private for member use only. +- `Node C` is behind a restrictive firewall, or does not want to expose a host to the public internet. + +The second of the two scenarios is solved already in the above graph using a proxy. Echolink proxies have become extremely common due to the fact very few people have access to a network where their host can be exposed publicly for other nodes to directly connect to when making a link. + +Echolink's solution to the first issue is fairly lacking in my opinion. Nodes only have the option to restrict connections based on a REGEX on their callsign. This means maintaining a list of allowed users, and updating *every* club-owned node when that changes. + +### Decentralizing directory services + +While I like Echolink a lot, I think some improvements could be made to both the node directory system, and how nodes link to each other. Ideally, I see a setup as follows: + +
+flowchart TD + D1((Directory A)) + D1 --- NA(Node A) & NB(Node B) & NC(Node C) + D2((Directory B)) + D2 --- NC & ND(Node D) + D3((Private Directory)) + D3 --- NE(Node E) & NF(Node F) + NE --> D2 +
+ +In this world, anyone could host a directory server, and either open it up to the public, or restrict it by means of their choosing (callsign rules, password, key-based access, ...). Nodes can then connect to one or many directories of their choosing. This could allow situations where a node could sit on the edge between two private intranets and have access to other nodes on *both* sides of the boundary without requiring either side to expose some part of itself to the public internet. + +### Routing + +I also envision a routing system where nodes attached to more than one directory service would periodically propagate information about their first-hop directories to other first-hop directories. An example of such interaction from the perspective of `Node C` would be: + +```text +Node C -> Directory A: You can find Directory B at
+Node C -> Directory B: You can find Directory A at
+``` + +`Node E` would have a different interaction. Since `Private Directory` is *private*, its announcement would *exclude* that directory: + +```text +Node E -> Private Directory: You can find Directory B at
+``` + +At this point, the directory servers should then be able to directly query eachother about their "routing tables". Over time, each directory server would build a list of which directory servers it can reach for information about nodes. + +The rable for `Private Directory` would look like this: + +| Server | Address | Hops | +|-------------|-------------|------| +| Directory B | `
` | 1 | +| Directory A | `
` | 2 | + +If `Node F` were to try connecting to `Node A`, the order of messages would be: + +```text +Node F -> Private Directory: Do you know Node A? + +# Private Directory queries the first table entry +Private Directory -> Directory B: Do you know Node A? +Directory B -> Private Directory: No + +# Private Directory queries the second table entry +Private Directory -> Directory A: Do you know Node A? +Directory A -> Private Directory: Node A is at
+ +# Node F is informed of the address of Node A +Private Directory -> Node F: Node A is at
+``` + +Of course, since `Node F` is *only* connected to a *private* directory service, if `Node A` were to try connecting to `Node F`, the order of messages would be: + +```text +Node A -> Directory A: Do you know Node F? + +# Directory A queries the first table entry +Directory A -> Directory B: Do you know Node F? +Directory B -> Directory A: No + +# Directory A was never informed that Private Directory exists, since it is private +# Thus, there is no route to Node F +Directory A -> Node A: No +``` + +Like every other piece of amateur radio backbone software, this routing system relies on trust. I am not an expert in identity verification, and I will not attempt to devise a system for stopping bad actors from poisoning the routing tables in this document. + +## Audio transport using modern tech + +So far in this document, node addresses have been used extensively, yet nodes are not expected to expose any ports to the internet. In the real-world, it has become very common to see peer-to-peer real-time media links in the form of video conferencing services. These services largely rely on WebRTC, a standardized, strong, and reliable protocol for peer-to-peer content streams. + +While WebRTC still struggles to deliver perfect quality audio and video simultaneously, it is more than capable of *just* streaming audio, with the added benefit that basically every browser supports it natively. Most programming languages also have one or many strong WebRTC libraries to work with. + +This means that a full implementation of this document would result in something similar to the following: + +- Directory server + - Handles incoming clients + - Handles routing tables + - Handles access rules + - Performs client lookups when requested + - Possibly provides telemetry information +- Client library + - Establishes connections with one or more directories + - Periodically propagates information about connected directories + - Can request to create or destroy a link + - When a link is created, it exposes a direct WebRTC connection to the remote node + - Hands off responsibility for the connection to whatever software wraps the library + +The wrapping software could be one of: + +- A website +- A smartphone app +- A headless application running on a Raspberry Pi that connects straight to a radio for a simplex node +- An application running on a repeater controller +- Anything else \ No newline at end of file diff --git a/sass/styles/layout.scss b/sass/styles/layout.scss index e386ced..4965157 100644 --- a/sass/styles/layout.scss +++ b/sass/styles/layout.scss @@ -132,3 +132,8 @@ ul { } } } + +.mermaid { + width: fit-content; + margin: auto; +} diff --git a/templates/base.html b/templates/base.html index 6a3c535..6be9243 100644 --- a/templates/base.html +++ b/templates/base.html @@ -78,7 +78,8 @@

Home | Blog | - Hobbies + Hobbies | + RFC


diff --git a/templates/page.html b/templates/page.html index fa6f24f..63728e5 100644 --- a/templates/page.html +++ b/templates/page.html @@ -67,7 +67,7 @@ MathJax.Hub.Config({tex2jax: {inlineMath: [['$','$'], ['\\(','\\)']], displayMat {# Comments only on blog posts #} - {% if page.description and page.date %} + {% if page.extra.is_rfc or page.description and page.date %}

{% endif %} +{% if page.extra.uses_graphviz %} + +{% endif %} {% endblock content %} \ No newline at end of file diff --git a/templates/section.html b/templates/section.html index a6b0549..25b29d4 100644 --- a/templates/section.html +++ b/templates/section.html @@ -49,6 +49,7 @@ {% block content %} {{section.content | safe}} +{# Blog posts #} {% if section.extra.inject_blog_posts %}
    {% for page in section.pages %} @@ -59,4 +60,16 @@ {% endfor %}
{% endif %} + +{# RFCs #} +{% if section.extra.inject_rfcs %} +
    + {% for page in section.pages %} + {% if not page.extra.hidden %} +
  • {{page.title}} ({{page.date}}) +
  • + {% endif %} + {% endfor %} +
+{% endif %} {% endblock content %} \ No newline at end of file