2023-07-18 18:38:30 +00:00
|
|
|
<!doctype html>
|
2023-06-10 07:15:38 +00:00
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<meta charset="utf-8" />
|
|
|
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
|
|
<link
|
|
|
|
rel="stylesheet"
|
2024-11-28 11:01:17 +00:00
|
|
|
href="https://cdnjs.cloudflare.com/ajax/libs/github-markdown-css/5.8.1/github-markdown.min.css"
|
|
|
|
integrity="sha512-BrOPA520KmDMqieeM7XFe6a3u3Sb3F1JBaQnrIAmWg3EYrciJ+Qqe6ZcKCdfPv26rGcgTrJnZ/IdQEct8h3Zhw=="
|
2024-11-16 01:25:52 +00:00
|
|
|
crossorigin="anonymous"
|
|
|
|
referrerpolicy="no-referrer"
|
2023-06-10 07:15:38 +00:00
|
|
|
/>
|
|
|
|
<title>haunt98 posts</title>
|
|
|
|
</head>
|
|
|
|
<style>
|
|
|
|
.markdown-body {
|
|
|
|
box-sizing: border-box;
|
|
|
|
min-width: 200px;
|
|
|
|
max-width: 980px;
|
|
|
|
margin: 0 auto;
|
|
|
|
padding: 45px;
|
2024-03-01 10:11:20 +00:00
|
|
|
font-family:
|
|
|
|
Shantell Sans Normal,
|
|
|
|
Inter,
|
2025-01-14 12:01:08 +00:00
|
|
|
SF Pro,
|
2024-03-01 10:11:20 +00:00
|
|
|
sans-serif;
|
|
|
|
font-weight: 500;
|
|
|
|
}
|
|
|
|
|
|
|
|
.markdown-body pre {
|
|
|
|
font-family:
|
2025-01-14 12:01:08 +00:00
|
|
|
Iosevka Pacman,
|
2024-03-01 10:11:20 +00:00
|
|
|
Jetbrains Mono,
|
2025-01-14 12:01:08 +00:00
|
|
|
SF Mono,
|
2024-03-01 10:11:20 +00:00
|
|
|
monospace;
|
2023-06-10 07:15:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@media (max-width: 767px) {
|
|
|
|
.markdown-body {
|
|
|
|
padding: 15px;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</style>
|
|
|
|
<body class="markdown-body">
|
2023-08-09 07:22:58 +00:00
|
|
|
<h2>
|
|
|
|
<a href="index.html"><code>~</code></a>
|
|
|
|
</h2>
|
2024-02-29 12:29:26 +00:00
|
|
|
<div class="markdown-heading">
|
|
|
|
<h1 class="heading-element">Another day another incident #02</h1>
|
2023-11-19 15:59:21 +00:00
|
|
|
<a
|
|
|
|
id="user-content-another-day-another-incident-02"
|
2024-03-08 08:12:41 +00:00
|
|
|
class="anchor"
|
2024-02-29 12:29:26 +00:00
|
|
|
aria-label="Permalink: Another day another incident #02"
|
2023-11-19 15:59:21 +00:00
|
|
|
href="#another-day-another-incident-02"
|
2024-02-29 12:29:26 +00:00
|
|
|
><span aria-hidden="true" class="octicon octicon-link"></span
|
|
|
|
></a>
|
|
|
|
</div>
|
2023-06-10 07:15:38 +00:00
|
|
|
<p>Today's incident is all about Go context.</p>
|
|
|
|
<p>TLDR: context got canceled, but it shouldn't.</p>
|
2024-02-29 12:29:26 +00:00
|
|
|
<div class="markdown-heading">
|
|
|
|
<h2 class="heading-element">The problem</h2>
|
2023-11-19 15:59:21 +00:00
|
|
|
<a
|
|
|
|
id="user-content-the-problem"
|
2024-03-08 08:12:41 +00:00
|
|
|
class="anchor"
|
2024-02-29 12:29:26 +00:00
|
|
|
aria-label="Permalink: The problem"
|
2023-11-19 15:59:21 +00:00
|
|
|
href="#the-problem"
|
2024-02-29 12:29:26 +00:00
|
|
|
><span aria-hidden="true" class="octicon octicon-link"></span
|
|
|
|
></a>
|
|
|
|
</div>
|
2023-06-10 07:15:38 +00:00
|
|
|
<p>Imagine a chain of APIs:</p>
|
|
|
|
<ul>
|
|
|
|
<li>Calling API A</li>
|
|
|
|
<li>Calling API B</li>
|
|
|
|
</ul>
|
|
|
|
<p>
|
|
|
|
Normally, if API A fails, API B should not be called. But what if API A is
|
2023-06-10 17:03:19 +00:00
|
|
|
<strong>optional</strong>, whether it successes or fails, API B should be
|
|
|
|
called anyway.
|
2023-06-10 07:15:38 +00:00
|
|
|
</p>
|
|
|
|
<p>My buggy code is like this:</p>
|
|
|
|
<div class="highlight highlight-source-go">
|
2024-12-14 04:49:37 +00:00
|
|
|
<pre><span class="pl-k">if</span> <span class="pl-s1">err</span> <span class="pl-c1">:=</span> <span class="pl-s1">doA</span>(<span class="pl-s1">ctx</span>); <span class="pl-s1">err</span> <span class="pl-c1">!=</span> <span class="pl-c1">nil</span> {
|
|
|
|
<span class="pl-s1">log</span>.<span class="pl-c1">Error</span>(<span class="pl-s1">err</span>)
|
2023-06-10 07:15:38 +00:00
|
|
|
<span class="pl-c">// Skip error</span>
|
|
|
|
}
|
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-s1">doB</span>(<span class="pl-s1">ctx</span>)</pre>
|
2023-06-10 07:15:38 +00:00
|
|
|
</div>
|
|
|
|
<p>
|
|
|
|
The problem is <code>doA</code> taking too long, so <code>ctx</code> is
|
|
|
|
canceled, and the parent of <code>ctx</code> is canceled too. So when
|
|
|
|
<code>doB</code> is called with <code>ctx</code>, it will be canceled too
|
|
|
|
(not what we want but sadly that what we got).
|
|
|
|
</p>
|
|
|
|
<p>
|
|
|
|
Example buggy code (<a
|
|
|
|
href="https://go.dev/play/p/p4S27Su16VH"
|
|
|
|
rel="nofollow"
|
|
|
|
>The Go Playground</a
|
|
|
|
>):
|
|
|
|
</p>
|
|
|
|
<div class="highlight highlight-source-go">
|
2024-12-14 04:49:37 +00:00
|
|
|
<pre><span class="pl-k">func</span> <span class="pl-s1">main</span>() {
|
|
|
|
<span class="pl-s1">ctx</span>, <span class="pl-s1">cancel</span> <span class="pl-c1">:=</span> <span class="pl-s1">context</span>.<span class="pl-c1">WithTimeout</span>(<span class="pl-s1">context</span>.<span class="pl-c1">Background</span>(), <span class="pl-c1">2</span><span class="pl-c1">*</span><span class="pl-s1">time</span>.<span class="pl-c1">Second</span>)
|
|
|
|
<span class="pl-k">defer</span> <span class="pl-s1">cancel</span>()
|
2023-06-10 07:15:38 +00:00
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-s1">doA</span>(<span class="pl-s1">ctx</span>)
|
|
|
|
<span class="pl-s1">doB</span>(<span class="pl-s1">ctx</span>)
|
2023-06-10 07:15:38 +00:00
|
|
|
}
|
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">func</span> <span class="pl-s1">doA</span>(<span class="pl-s1">ctx</span> context.<span class="pl-smi">Context</span>) {
|
|
|
|
<span class="pl-s1">ctx</span>, <span class="pl-s1">ctxCancel</span> <span class="pl-c1">:=</span> <span class="pl-s1">context</span>.<span class="pl-c1">WithTimeout</span>(<span class="pl-s1">ctx</span>, <span class="pl-c1">1</span><span class="pl-c1">*</span><span class="pl-s1">time</span>.<span class="pl-c1">Second</span>)
|
|
|
|
<span class="pl-k">defer</span> <span class="pl-s1">ctxCancel</span>()
|
2023-06-10 07:15:38 +00:00
|
|
|
|
|
|
|
<span class="pl-k">select</span> {
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">case</span> <span class="pl-c1"><-</span><span class="pl-s1">time</span>.<span class="pl-c1">After</span>(<span class="pl-c1">2</span> <span class="pl-c1">*</span> <span class="pl-s1">time</span>.<span class="pl-c1">Second</span>):
|
|
|
|
<span class="pl-s1">fmt</span>.<span class="pl-c1">Println</span>(<span class="pl-s">"doA"</span>)
|
|
|
|
<span class="pl-k">case</span> <span class="pl-c1"><-</span><span class="pl-s1">ctx</span>.<span class="pl-c1">Done</span>():
|
|
|
|
<span class="pl-s1">fmt</span>.<span class="pl-c1">Println</span>(<span class="pl-s">"doA"</span>, <span class="pl-s1">ctx</span>.<span class="pl-c1">Err</span>())
|
2023-06-10 07:15:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">func</span> <span class="pl-s1">doB</span>(<span class="pl-s1">ctx</span> context.<span class="pl-smi">Context</span>) {
|
|
|
|
<span class="pl-s1">ctx</span>, <span class="pl-s1">ctxCancel</span> <span class="pl-c1">:=</span> <span class="pl-s1">context</span>.<span class="pl-c1">WithTimeout</span>(<span class="pl-s1">ctx</span>, <span class="pl-c1">3</span><span class="pl-c1">*</span><span class="pl-s1">time</span>.<span class="pl-c1">Second</span>)
|
|
|
|
<span class="pl-k">defer</span> <span class="pl-s1">ctxCancel</span>()
|
2023-06-10 07:15:38 +00:00
|
|
|
|
|
|
|
<span class="pl-k">select</span> {
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">case</span> <span class="pl-c1"><-</span><span class="pl-s1">time</span>.<span class="pl-c1">After</span>(<span class="pl-c1">2</span> <span class="pl-c1">*</span> <span class="pl-s1">time</span>.<span class="pl-c1">Second</span>):
|
|
|
|
<span class="pl-s1">fmt</span>.<span class="pl-c1">Println</span>(<span class="pl-s">"doB"</span>)
|
|
|
|
<span class="pl-k">case</span> <span class="pl-c1"><-</span><span class="pl-s1">ctx</span>.<span class="pl-c1">Done</span>():
|
|
|
|
<span class="pl-s1">fmt</span>.<span class="pl-c1">Println</span>(<span class="pl-s">"doB"</span>, <span class="pl-s1">ctx</span>.<span class="pl-c1">Err</span>())
|
2023-06-10 07:15:38 +00:00
|
|
|
}
|
|
|
|
}</pre>
|
|
|
|
</div>
|
|
|
|
<p>The output is:</p>
|
|
|
|
<div class="highlight highlight-text-adblock">
|
|
|
|
<pre>
|
|
|
|
doA context deadline exceeded
|
|
|
|
doB context deadline exceeded</pre
|
|
|
|
>
|
|
|
|
</div>
|
|
|
|
<p>As you see both <code>doA</code> and <code>doB</code> are canceled.</p>
|
2024-02-29 12:29:26 +00:00
|
|
|
<div class="markdown-heading">
|
|
|
|
<h2 class="heading-element">The (temporary) solution</h2>
|
2023-11-19 15:59:21 +00:00
|
|
|
<a
|
|
|
|
id="user-content-the-temporary-solution"
|
2024-03-08 08:12:41 +00:00
|
|
|
class="anchor"
|
2024-02-29 12:29:26 +00:00
|
|
|
aria-label="Permalink: The (temporary) solution"
|
2023-11-19 15:59:21 +00:00
|
|
|
href="#the-temporary-solution"
|
2024-02-29 12:29:26 +00:00
|
|
|
><span aria-hidden="true" class="octicon octicon-link"></span
|
|
|
|
></a>
|
|
|
|
</div>
|
2023-06-10 07:15:38 +00:00
|
|
|
<p>
|
|
|
|
Quick Google search leads me to
|
|
|
|
<a href="https://github.com/golang/go/issues/40221"
|
|
|
|
>context: add WithoutCancel #40221</a
|
|
|
|
>
|
|
|
|
and I quote:
|
|
|
|
</p>
|
|
|
|
<blockquote>
|
|
|
|
<p>
|
|
|
|
This is useful in multiple frequently recurring and important scenarios:
|
|
|
|
</p>
|
|
|
|
<ul>
|
|
|
|
<li>
|
|
|
|
Handling of rollback/cleanup operations in the context of an event
|
|
|
|
(e.g., HTTP request) that has to continue regardless of whether the
|
|
|
|
triggering event is canceled (e.g., due to timeout or the client going
|
|
|
|
away)
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
Handling of long-running operations triggered by an event (e.g., HTTP
|
|
|
|
request) that terminates before the termination of the long-running
|
|
|
|
operation
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
</blockquote>
|
|
|
|
<p>
|
|
|
|
So beside waiting to upgrade to Go <code>1.21</code> to use
|
|
|
|
<code>context.WithoutCancel</code>, you can use this
|
|
|
|
<a href="https://pkg.go.dev/context@master#WithoutCancel" rel="nofollow"
|
|
|
|
>workaround code</a
|
|
|
|
>:
|
|
|
|
</p>
|
|
|
|
<div class="highlight highlight-source-go">
|
2024-12-14 04:49:37 +00:00
|
|
|
<pre><span class="pl-k">func</span> <span class="pl-s1">DisconnectContext</span>(<span class="pl-s1">parent</span> context.<span class="pl-smi">Context</span>) context.<span class="pl-smi">Context</span> {
|
2023-06-10 07:15:38 +00:00
|
|
|
<span class="pl-k">if</span> <span class="pl-s1">parent</span> <span class="pl-c1">==</span> <span class="pl-c1">nil</span> {
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">return</span> <span class="pl-s1">context</span>.<span class="pl-c1">Background</span>()
|
2023-06-10 07:15:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
<span class="pl-k">return</span> <span class="pl-smi">disconnectedContext</span>{
|
2024-03-23 16:45:02 +00:00
|
|
|
<span class="pl-s1">parent</span>: <span class="pl-s1">parent</span>,
|
2023-06-10 07:15:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
<span class="pl-k">type</span> <span class="pl-smi">disconnectedContext</span> <span class="pl-k">struct</span> {
|
|
|
|
<span class="pl-c1">parent</span> context.<span class="pl-smi">Context</span>
|
|
|
|
}
|
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">func</span> (<span class="pl-s1">ctx</span> <span class="pl-smi">disconnectedContext</span>) <span class="pl-c1">Deadline</span>() (<span class="pl-s1">deadline</span> time.<span class="pl-smi">Time</span>, <span class="pl-s1">ok</span> <span class="pl-smi">bool</span>) {
|
2023-06-10 07:15:38 +00:00
|
|
|
<span class="pl-k">return</span>
|
|
|
|
}
|
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">func</span> (<span class="pl-s1">ctx</span> <span class="pl-smi">disconnectedContext</span>) <span class="pl-c1">Done</span>() <span class="pl-c1"><-</span><span class="pl-k">chan</span> <span class="pl-k">struct</span>{} {
|
2023-06-10 07:15:38 +00:00
|
|
|
<span class="pl-k">return</span> <span class="pl-c1">nil</span>
|
|
|
|
}
|
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">func</span> (<span class="pl-s1">ctx</span> <span class="pl-smi">disconnectedContext</span>) <span class="pl-c1">Err</span>() <span class="pl-smi">error</span> {
|
2023-06-10 07:15:38 +00:00
|
|
|
<span class="pl-k">return</span> <span class="pl-c1">nil</span>
|
|
|
|
}
|
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">func</span> (<span class="pl-s1">ctx</span> <span class="pl-smi">disconnectedContext</span>) <span class="pl-c1">Value</span>(<span class="pl-s1">key</span> <span class="pl-smi">any</span>) <span class="pl-smi">any</span> {
|
|
|
|
<span class="pl-k">return</span> <span class="pl-s1">ctx</span>.<span class="pl-c1">parent</span>.<span class="pl-c1">Value</span>(<span class="pl-s1">key</span>)
|
2023-06-10 07:15:38 +00:00
|
|
|
}</pre>
|
|
|
|
</div>
|
|
|
|
<p>
|
|
|
|
So the buggy code becomes (<a
|
|
|
|
href="https://go.dev/play/p/oIU-WxEJ_F3"
|
|
|
|
rel="nofollow"
|
|
|
|
>The Go Playground</a
|
|
|
|
>):
|
|
|
|
</p>
|
|
|
|
<div class="highlight highlight-source-go">
|
2024-12-14 04:49:37 +00:00
|
|
|
<pre><span class="pl-k">func</span> <span class="pl-s1">main</span>() {
|
|
|
|
<span class="pl-s1">ctx</span>, <span class="pl-s1">cancel</span> <span class="pl-c1">:=</span> <span class="pl-s1">context</span>.<span class="pl-c1">WithTimeout</span>(<span class="pl-s1">context</span>.<span class="pl-c1">Background</span>(), <span class="pl-c1">2</span><span class="pl-c1">*</span><span class="pl-s1">time</span>.<span class="pl-c1">Second</span>)
|
|
|
|
<span class="pl-k">defer</span> <span class="pl-s1">cancel</span>()
|
|
|
|
<span class="pl-s1">doA</span>(<span class="pl-s1">ctx</span>)
|
|
|
|
<span class="pl-s1">doB</span>(<span class="pl-s1">ctx</span>)
|
2023-06-10 07:15:38 +00:00
|
|
|
}
|
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">func</span> <span class="pl-s1">doA</span>(<span class="pl-s1">ctx</span> context.<span class="pl-smi">Context</span>) {
|
|
|
|
<span class="pl-s1">ctx</span>, <span class="pl-s1">ctxCancel</span> <span class="pl-c1">:=</span> <span class="pl-s1">context</span>.<span class="pl-c1">WithTimeout</span>(<span class="pl-s1">ctx</span>, <span class="pl-c1">1</span><span class="pl-c1">*</span><span class="pl-s1">time</span>.<span class="pl-c1">Second</span>)
|
|
|
|
<span class="pl-k">defer</span> <span class="pl-s1">ctxCancel</span>()
|
2023-06-10 07:15:38 +00:00
|
|
|
|
|
|
|
<span class="pl-k">select</span> {
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">case</span> <span class="pl-c1"><-</span><span class="pl-s1">time</span>.<span class="pl-c1">After</span>(<span class="pl-c1">2</span> <span class="pl-c1">*</span> <span class="pl-s1">time</span>.<span class="pl-c1">Second</span>):
|
|
|
|
<span class="pl-s1">fmt</span>.<span class="pl-c1">Println</span>(<span class="pl-s">"doA"</span>)
|
|
|
|
<span class="pl-k">case</span> <span class="pl-c1"><-</span><span class="pl-s1">ctx</span>.<span class="pl-c1">Done</span>():
|
|
|
|
<span class="pl-s1">fmt</span>.<span class="pl-c1">Println</span>(<span class="pl-s">"doA"</span>, <span class="pl-s1">ctx</span>.<span class="pl-c1">Err</span>())
|
2023-06-10 07:15:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">func</span> <span class="pl-s1">doB</span>(<span class="pl-s1">ctx</span> context.<span class="pl-smi">Context</span>) {
|
|
|
|
<span class="pl-s1">ctx</span>, <span class="pl-s1">ctxCancel</span> <span class="pl-c1">:=</span> <span class="pl-s1">context</span>.<span class="pl-c1">WithTimeout</span>(<span class="pl-s1">DisconnectContext</span>(<span class="pl-s1">ctx</span>), <span class="pl-c1">3</span><span class="pl-c1">*</span><span class="pl-s1">time</span>.<span class="pl-c1">Second</span>)
|
|
|
|
<span class="pl-k">defer</span> <span class="pl-s1">ctxCancel</span>()
|
2023-06-10 07:15:38 +00:00
|
|
|
|
|
|
|
<span class="pl-k">select</span> {
|
2024-12-14 04:49:37 +00:00
|
|
|
<span class="pl-k">case</span> <span class="pl-c1"><-</span><span class="pl-s1">time</span>.<span class="pl-c1">After</span>(<span class="pl-c1">2</span> <span class="pl-c1">*</span> <span class="pl-s1">time</span>.<span class="pl-c1">Second</span>):
|
|
|
|
<span class="pl-s1">fmt</span>.<span class="pl-c1">Println</span>(<span class="pl-s">"doB"</span>)
|
|
|
|
<span class="pl-k">case</span> <span class="pl-c1"><-</span><span class="pl-s1">ctx</span>.<span class="pl-c1">Done</span>():
|
|
|
|
<span class="pl-s1">fmt</span>.<span class="pl-c1">Println</span>(<span class="pl-s">"doB"</span>, <span class="pl-s1">ctx</span>.<span class="pl-c1">Err</span>())
|
2023-06-10 07:15:38 +00:00
|
|
|
}
|
|
|
|
}</pre>
|
|
|
|
</div>
|
|
|
|
<p>The output is:</p>
|
|
|
|
<div class="highlight highlight-text-adblock">
|
|
|
|
<pre>
|
|
|
|
doA context deadline exceeded
|
|
|
|
doB</pre
|
|
|
|
>
|
|
|
|
</div>
|
|
|
|
<p>
|
|
|
|
As you see only <code>doA</code> is canceled, <code>doB</code> is done
|
|
|
|
perfectly. And that what we want in this case.
|
|
|
|
</p>
|
2024-02-29 12:29:26 +00:00
|
|
|
<div class="markdown-heading">
|
|
|
|
<h2 class="heading-element">Thanks</h2>
|
2023-11-19 15:59:21 +00:00
|
|
|
<a
|
|
|
|
id="user-content-thanks"
|
2024-03-08 08:12:41 +00:00
|
|
|
class="anchor"
|
2024-02-29 12:29:26 +00:00
|
|
|
aria-label="Permalink: Thanks"
|
2023-11-19 15:59:21 +00:00
|
|
|
href="#thanks"
|
2024-02-29 12:29:26 +00:00
|
|
|
><span aria-hidden="true" class="octicon octicon-link"></span
|
|
|
|
></a>
|
|
|
|
</div>
|
2023-06-10 13:54:19 +00:00
|
|
|
<ul>
|
|
|
|
<li>
|
|
|
|
<a
|
|
|
|
href="https://www.sohamkamani.com/golang/context-cancellation-and-values/"
|
|
|
|
rel="nofollow"
|
|
|
|
>Using Context in Golang - Cancellation, Timeouts and Values (With
|
|
|
|
Examples)</a
|
|
|
|
>
|
|
|
|
</li>
|
|
|
|
<li>
|
|
|
|
<a
|
|
|
|
href="https://uptrace.dev/blog/golang-context-timeout.html"
|
|
|
|
rel="nofollow"
|
|
|
|
>Go Context timeouts can be harmful</a
|
|
|
|
>
|
|
|
|
</li>
|
|
|
|
</ul>
|
2023-06-10 07:15:38 +00:00
|
|
|
|
|
|
|
<div>
|
|
|
|
Feel free to ask me via
|
|
|
|
<a href="mailto:hauvipapro+posts@gmail.com">email</a> or
|
2023-08-20 17:29:13 +00:00
|
|
|
<a rel="me" href="https://hachyderm.io/@haunguyen">Mastodon</a>.
|
|
|
|
<br />Source code is available on
|
2023-06-10 07:15:38 +00:00
|
|
|
<a href="https://github.com/haunt98/posts-go">GitHub</a>
|
2023-08-20 17:29:13 +00:00
|
|
|
<a href="https://codeberg.org/yoshie/posts-go">Codeberg</a>
|
|
|
|
<a href="https://git.sr.ht/~youngyoshie/posts-go">sourcehut</a>
|
|
|
|
<a href="https://gitea.treehouse.systems/yoshie/posts-go">Treehouse</a>
|
|
|
|
<a href="https://gitlab.com/youngyoshie/posts-go">GitLab</a>
|
2023-06-10 07:15:38 +00:00
|
|
|
</div>
|
|
|
|
</body>
|
|
|
|
</html>
|