[{"data":1,"prerenderedAt":-1},["ShallowReactive",2],{"academy-blogs-en-1-1-all-ep116-realtime-voice-video-webrtc-websocket-go-all--*":3,"academy-blog-translations-kgoilbfmq2b2kt1":89},{"data":4,"page":77,"perPage":77,"totalItems":77,"totalPages":77},[5],{"alt":6,"collectionId":7,"collectionName":8,"content":9,"cover_image":10,"cover_image_path":11,"created":12,"created_by":13,"expand":14,"id":83,"keywords":84,"locale":59,"published_at":85,"scheduled_at":13,"school_blog":81,"short_description":86,"slug":87,"status":79,"title":6,"updated":88,"updated_by":13,"views":82},"EP.116 Real-time Voice\u002FVideo Communication with WebRTC + WebSocket Signaling in Go","sclblg987654321","school_blog_translations","\u003Cp>Text-based chat over WebSocket is just the beginning. Modern communication systems must support real-time voice and video features just like your favorite meeting apps.\u003C\u002Fp>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>And the global standard for real-time media transmission is WebRTC. But here’s the catch: WebRTC cannot initiate the connection by itself. It needs a mechanism called Signaling, and in this tutorial, we’ll use WebSocket as our signaling server.\u003C\u002Fp>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>This article will walk you through everything you need to know: From PeerConnection → ICE Candidates → Signaling → actual audio\u002Fvideo streaming all with real, production-grade Go code.\u003C\u002Fp>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>🔊 What is WebRTC?\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>WebRTC is a framework for real-time peer-to-peer communication directly between browsers.\u003C\u002Fp>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch3>✨ Key features:\u003C\u002Fh3>\u003Cul>\u003Cli>Low-latency audio\u002Fvideo (less than 100ms)\u003C\u002Fli>\u003Cli>True peer-to-peer connection (no central server for media)\u003C\u002Fli>\u003Cli>Secure by default (encrypted media streams)\u003C\u002Fli>\u003C\u002Ful>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>🤔 What about WebSocket?\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cfigure class=\"table\">\u003Ctable>\u003Cthead>\u003Ctr>\u003Cth>Feature\u003C\u002Fth>\u003Cth>WebSocket\u003C\u002Fth>\u003Cth>WebRTC\u003C\u002Fth>\u003C\u002Ftr>\u003C\u002Fthead>\u003Ctbody>\u003Ctr>\u003Ctd>Real-time messaging\u003C\u002Ftd>\u003Ctd>✅\u003C\u002Ftd>\u003Ctd>✅\u003C\u002Ftd>\u003C\u002Ftr>\u003Ctr>\u003Ctd>File transfer\u003C\u002Ftd>\u003Ctd>✅\u003C\u002Ftd>\u003Ctd>✅\u003C\u002Ftd>\u003C\u002Ftr>\u003Ctr>\u003Ctd>Audio\u002FVideo streaming\u003C\u002Ftd>\u003Ctd>❌ Not ideal\u003C\u002Ftd>\u003Ctd>✅\u003C\u002Ftd>\u003C\u002Ftr>\u003Ctr>\u003Ctd>Peer-to-peer connection\u003C\u002Ftd>\u003Ctd>❌\u003C\u002Ftd>\u003Ctd>✅\u003C\u002Ftd>\u003C\u002Ftr>\u003Ctr>\u003Ctd>Sub-50ms latency\u003C\u002Ftd>\u003Ctd>❌\u003C\u002Ftd>\u003Ctd>✅\u003C\u002Ftd>\u003C\u002Ftr>\u003C\u002Ftbody>\u003C\u002Ftable>\u003C\u002Ffigure>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>WebRTC is purpose-built for real-time media transmission.\u003C\u002Fp>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>🛰 Components of WebRTC\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cul>\u003Cli>RTCPeerConnection\u003Cbr>Handles encrypted media connection between peers.\u003C\u002Fli>\u003Cli>ICE Candidate\u003Cbr>Determines possible network paths (LAN, Public IP, TURN server, etc.)\u003C\u002Fli>\u003Cli>SDP (Session Description Protocol)\u003Cbr>Describes media details (codec, bitrate, etc.)\u003C\u002Fli>\u003Cli>Signaling Server\u003Cbr>Responsible for exchanging messages like offer, answer, and ICE candidates between clients.\u003C\u002Fli>\u003C\u002Ful>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>⚠️ WebRTC does not include signaling you need to implement it yourself.\u003C\u002Fp>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>🔄 Using WebSocket as Signaling Server\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>Connection flow:\u003C\u002Fp>\u003Col>\u003Cli>👤 Client A creates an \u003Ccode inline=\"\">offer\u003C\u002Fcode> and sends it via WebSocket\u003C\u002Fli>\u003Cli>🔁 Server forwards the offer to Client B\u003C\u002Fli>\u003Cli>👤 Client B replies with an \u003Ccode inline=\"\">answer\u003C\u002Fcode>\u003C\u002Fli>\u003Cli>🔄 Both clients exchange ICE candidates\u003C\u002Fli>\u003Cli>✅ Once handshake is complete → media streaming begins!\u003C\u002Fli>\u003C\u002Fol>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>🧪 Go Example: WebSocket Signaling Server\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cpre>\u003Ccode class=\"language-plaintext language-go\">type SignalMessage struct {\n    Type string `json:\"type\"` \u002F\u002F offer, answer, ice\n    From string `json:\"from\"`\n    To   string `json:\"to\"`\n    Data json.RawMessage `json:\"data\"`\n}\n\nvar clients = make(map[string]*websocket.Conn)\n\nfunc signalingHandler(c *fiber.Ctx) error {\n    conn, err := upgrader.Upgrade(c.Context(), nil)\n    if err != nil { return err }\n\n    clientID := c.Query(\"id\")\n    clients[clientID] = conn\n\n    for {\n        _, msg, err := conn.ReadMessage()\n        if err != nil {\n            delete(clients, clientID)\n            return nil\n        }\n\n        var signal SignalMessage\n        json.Unmarshal(msg, &amp;signal)\n\n        if target, ok := clients[signal.To]; ok {\n            target.WriteMessage(websocket.TextMessage, msg)\n        }\n    }\n}\n\u003C\u002Fcode>\u003C\u002Fpre>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>🎧 Client-side Code (JavaScript)\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch3>1. Create PeerConnection\u003C\u002Fh3>\u003Cpre>\u003Ccode class=\"language-plaintext language-js\">const pc = new RTCPeerConnection({\n  iceServers: [{ urls: \"stun:stun.l.google.com:19302\" }]\n});\n\npc.onicecandidate = event =&gt; {\n  if (event.candidate) {\n    socket.send(JSON.stringify({\n      type: \"ice\",\n      to: remoteUser,\n      data: event.candidate\n    }));\n  }\n};\n\u003C\u002Fcode>\u003C\u002Fpre>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch3>2. Start Call and Send Offer\u003C\u002Fh3>\u003Cpre>\u003Ccode class=\"language-plaintext language-js\">async function startCall() {\n  const stream = await navigator.mediaDevices.getUserMedia({ video: true, audio: true });\n  stream.getTracks().forEach(track =&gt; pc.addTrack(track, stream));\n\n  const offer = await pc.createOffer();\n  await pc.setLocalDescription(offer);\n\n  socket.send(JSON.stringify({\n    type: \"offer\",\n    to: remoteUser,\n    data: offer\n  }));\n}\n\u003C\u002Fcode>\u003C\u002Fpre>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>🔐 Using TURN Server (When P2P Fails)\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>In some scenarios, P2P connection might fail due to:\u003C\u002Fp>\u003Cul>\u003Cli>NAT restrictions\u003C\u002Fli>\u003Cli>Firewalls blocking ports\u003C\u002Fli>\u003Cli>Mobile networks\u003C\u002Fli>\u003C\u002Ful>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>In such cases, you’ll need a TURN server to relay the media Examples:\u003C\u002Fp>\u003Cul>\u003Cli>Coturn\u003C\u002Fli>\u003Cli>Twilio TURN Service\u003C\u002Fli>\u003Cli>Google Cloud WebRTC Infra\u003C\u002Fli>\u003C\u002Ful>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>⚙️ Production-grade Best Practices\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cfigure class=\"table\">\u003Ctable>\u003Cthead>\u003Ctr>\u003Cth>Security Concern\u003C\u002Fth>\u003Cth>Recommended Strategy\u003C\u002Fth>\u003C\u002Ftr>\u003C\u002Fthead>\u003Ctbody>\u003Ctr>\u003Ctd>Token Reuse Prevention\u003C\u002Ftd>\u003Ctd>Use JWT with Device Binding\u003C\u002Ftd>\u003C\u002Ftr>\u003Ctr>\u003Ctd>Stranger Calls\u003C\u002Ftd>\u003Ctd>Use Room-based Permission\u003C\u002Ftd>\u003C\u002Ftr>\u003Ctr>\u003Ctd>DDOS \u002F Abuse\u003C\u002Ftd>\u003Ctd>Implement Rate Limits\u003C\u002Ftd>\u003C\u002Ftr>\u003Ctr>\u003Ctd>Media Relay\u003C\u002Ftd>\u003Ctd>Use Secure TURN (with auth)\u003C\u002Ftd>\u003C\u002Ftr>\u003C\u002Ftbody>\u003C\u002Ftable>\u003C\u002Ffigure>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>🚀 Challenge: Build Your First Video Call\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>✅ Implement a working prototype using:\u003C\u002Fp>\u003Cul>\u003Cli>WebSocket → Signaling\u003C\u002Fli>\u003Cli>WebRTC → P2P connection\u003C\u002Fli>\u003Cli>ICE Candidates exchange\u003C\u002Fli>\u003Cli>Media streaming\u003C\u002Fli>\u003C\u002Ful>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>Start with 1-on-1 calls → Expand to group call later!\u003C\u002Fp>\u003Cp>&nbsp;\u003C\u002Fp>\u003Chr>\u003Cp>&nbsp;\u003C\u002Fp>\u003Ch2>🔮 Coming Up Next: WebSocket for IoT Device Networks\u003C\u002Fh2>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>In the next episode, we’ll explore building a real-time IoT network with WebSocket:\u003C\u002Fp>\u003Cul>\u003Cli>Manage thousands of devices (sensors, robots, smart home)\u003C\u002Fli>\u003Cli>Handle device auth &amp; secure communication\u003C\u002Fli>\u003Cli>Optimize bandwidth for constrained networks\u003C\u002Fli>\u003C\u002Ful>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cp>If you’ve made it this far you’re ready for real-time engineering at scale! 🎯\u003C\u002Fp>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cdiv class=\"raw-html-embed\">\u003Cdiv style=\"margin:0 0 6px 0; font-weight:700;\">Read more:\u003C\u002Fdiv>\n\u003Cul style=\"list-style:none; padding:0; margin:0; line-height:1.4;\">\n  \u003Cli style=\"margin:0;\">\u003Ca href=\"\u002Fen\u002Fblogs\u002Fcategories\u002FGolang\" title=\"Golang The Series\">Golang The Series\u003C\u002Fa>\u003C\u002Fli>\n  \u003Cli style=\"margin:0;\">\u003Ca href=\"\u002Fen\u002Fblogs\u002Fcategories\u002FJS2GO\" title=\"JS2GO\">JS2GO\u003C\u002Fa>\u003C\u002Fli>\n  \u003Cli style=\"margin:0;\">\u003Ca href=\"\u002Fen\u002Fblogs\u002Fcategories\u002FTailwind%20CSS\" title=\"Tailwind CSS\">Tailwind CSS\u003C\u002Fa>\u003C\u002Fli>\n\u003C\u002Ful>\u003C\u002Fdiv>\u003Cp>&nbsp;\u003C\u002Fp>\u003Cdiv class=\"raw-html-embed\">\n  \u003Cp style=\"margin:0 0 6px 0;\">\u003Cstrong>Follow Us:\u003C\u002Fstrong>\u003C\u002Fp>\n  \u003Cul style=\"list-style:none; padding:0; margin:0; line-height: 0.4;\">\n    \u003Cli style=\"display:flex; align-items:center; gap:6px; margin:0;\">\n      \n      \u003Csvg width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" fill=\"#1877F2\" aria-hidden=\"true\">\n        \u003Cpath d=\"M22 12.07C22 6.48 17.52 2 11.93 2S2 6.48 2 12.07c0 5 3.66 9.14 8.44 9.93v-7.02H7.9v-2.91h2.54V9.41c0-2.5 1.49-3.88 3.77-3.88 1.09 0 2.24.2 2.24.2v2.46h-1.26c-1.24 0-1.63.77-1.63 1.56v1.87h2.78l-.44 2.91h-2.34V22c4.78-.79 8.44-4.93 8.44-9.93Z\">\u003C\u002Fpath>\n      \u003C\u002Fsvg>\n      \u003Ca href=\"https:\u002F\u002Fwww.facebook.com\u002Fsuperdev.academy.th\" target=\"_blank\" rel=\"nofollow noopener\" title=\"Follow Superdev Academy on Facebook\">Facebook: Superdev Academy\u003C\u002Fa>\n    \u003C\u002Fli>\n\n    \u003Cli style=\"display:flex; align-items:center; gap:6px; margin:0;\">\n      \n      \u003Csvg width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" fill=\"#FF0000\" aria-hidden=\"true\">\n        \u003Cpath d=\"M23.5 6.2a3 3 0 0 0-2.1-2.1C19.5 3.5 12 3.5 12 3.5s-7.5 0-9.4.6A3 3 0 0 0 .5 6.2 31.5 31.5 0 0 0 0 12a31.5 31.5 0 0 0 .5 5.8 3 3 0 0 0 2.1 2.1c1.9.6 9.4.6 9.4.6s7.5 0 9.4-.6a3 3 0 0 0 2.1-2.1A31.5 31.5 0 0 0 24 12a31.5 31.5 0 0 0-.5-5.8ZM9.75 15.02V8.98L15.5 12l-5.75 3.02Z\">\u003C\u002Fpath>\n      \u003C\u002Fsvg>\n      \u003Ca href=\"https:\u002F\u002Fwww.youtube.com\u002F@SuperdevAcademy\" target=\"_blank\" rel=\"nofollow noopener\" title=\"Watch on YouTube\">YouTube: Superdev Academy\u003C\u002Fa>\n    \u003C\u002Fli>\n\n    \u003Cli style=\"display:flex; align-items:center; gap:6px; margin:0;\">\n      \n      \u003Csvg width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" fill=\"#E4405F\" aria-hidden=\"true\">\n        \u003Cpath d=\"M7 2h10a5 5 0 0 1 5 5v10a5 5 0 0 1-5 5H7a5 5 0 0 1-5-5V7a5 5 0 0 1 5-5Zm10 2H7a3 3 0 0 0-3 3v10a3 3 0 0 0 3 3h10a3 3 0 0 0 3-3V7a3 3 0 0 0-3-3Zm-5 3.5A5.5 5.5 0 1 1 6.5 13 5.5 5.5 0 0 1 12 7.5Zm0 2A3.5 3.5 0 1 0 15.5 13 3.5 3.5 0 0 0 12 9.5Zm5.75-2.75a1.25 1.25 0 1 1-1.25 1.25 1.25 1.25 0 0 1 1.25-1.25Z\">\u003C\u002Fpath>\n      \u003C\u002Fsvg>\n      \u003Ca href=\"https:\u002F\u002Fwww.instagram.com\u002Fsuperdevacademy\u002F?hl=en target=\" _blank\"=\"\" rel=\"nofollow noopener\" title=\"See behind-the-scenes on Instagram\">Instagram: Superdev Academy\u003C\u002Fa>\n    \u003C\u002Fli>\n\n    \u003Cli style=\"display:flex; align-items:center; gap:6px; margin:0;\">\n      \n      \u003Csvg width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" fill=\"#000000\" aria-hidden=\"true\">\n        \u003Cpath d=\"M21 8.12a6.86 6.86 0 0 1-4.8-2V16a6 6 0 1 1-6-6 5.9 5.9 0 0 1 1.63.23V8.05a9.08 9.08 0 0 1-1.63-.15V4.5a6.86 6.86 0 0 0 4.8 2.05V6.5a6.86 6.86 0 0 0 4.8 1.62ZM9.2 12.5A3.5 3.5 0 1 0 12.7 16V9.94a6 6 0 0 1-1.63-.27v3.95a3.5 3.5 0 0 1-1.87 3.17 3.5 3.5 0 0 1-4.78-3.23 3.5 3.5 0 0 1 4.78-3.06Z\">\u003C\u002Fpath>\n      \u003C\u002Fsvg>\n      \u003Ca href=\"https:\u002F\u002Fwww.tiktok.com\u002F@superdevacademy\" target=\"_blank\" rel=\"nofollow noopener\" title=\"Watch short tips on TikTok\">TikTok: @superdevacademy\u003C\u002Fa>\n    \u003C\u002Fli>\n\n    \u003Cli style=\"display:flex; align-items:center; gap:6px; margin:0;\">\n      \n      \u003Csvg width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" fill=\"#111827\" aria-hidden=\"true\">\n        \u003Cpath d=\"M12 2a10 10 0 1 0 10 10A10.01 10.01 0 0 0 12 2Zm6.93 6h-3.26a15.6 15.6 0 0 0-1.39-3.62A8.03 8.03 0 0 1 18.93 8ZM12 4c.73.93 1.7 2.74 2.2 4H9.8C10.3 6.74 11.27 4.93 12 4ZM8.72 4.38A15.6 15.6 0 0 0 7.32 8H4.07a8.03 8.03 0 0 1 4.65-3.62ZM4.07 16h3.25a15.6 15.6 0 0 0 1.4 3.62A8.03 8.03 0 0 1 4.07 16ZM12 20c-.73-.93-1.7-2.74-2.2-4h4.4C13.7 17.26 12.73 19.07 12 20Zm3.28-.38A15.6 15.6 0 0 0 16.68 16h3.25a8.03 8.03 0 0 1-4.65 3.62ZM20 14h-3.54a13.8 13.8 0 0 1-.26-4H20a7.98 7.98 0 0 1 0 4Zm-12.2 0H4a7.98 7.98 0 0 1 0-4h3.54a13.8 13.8 0 0 1-.26 4Zm2 .5h4.4a17.8 17.8 0 0 1-.72-4.5c0-1.58.25-3.1.72-4.5H9.8a17.8 17.8 0 0 1 .72 4.5c0 1.58-.25 3.1-.72 4.5Z\">\u003C\u002Fpath>\n      \u003C\u002Fsvg>\n      \u003Ca href=\"https:\u002F\u002Fwww.superdevacademy.com\u002F\" target=\"_blank\" rel=\"noopener\" title=\"Visit the official website of Superdev Academy\">Official Website: Superdev Academy.com\u003C\u002Fa>\n    \u003C\u002Fli>\n  \u003C\u002Ful>\n\u003C\u002Fdiv>\u003Cp>&nbsp;\u003C\u002Fp>","cover_image_ep_q518w3qnpf.etSignalinginGo.webp","https:\u002F\u002Ftwsme-r2.tumwebsme.com\u002Fsclblg987654321\u002Fxkmzgoufvuwfzdv\u002Fcover_image_ep_q518w3qnpf.etSignalinginGo.webp","2026-03-04 08:45:06.328Z","",{"keywords":15,"locale":53,"school_blog":63},[16,23,28,33,38,43,48],{"collectionId":17,"collectionName":18,"created":19,"created_by":13,"id":20,"name":21,"updated":22,"updated_by":13},"sclkey987654321","school_keywords","2026-03-04 08:20:14.253Z","ah6lvy4x8qe08l5","Golang","2026-04-10 16:07:26.172Z",{"collectionId":17,"collectionName":18,"created":24,"created_by":13,"id":25,"name":26,"updated":27,"updated_by":13},"2026-03-04 08:20:11.547Z","ey3puyme01a9bsw","Go","2026-04-10 16:07:25.893Z",{"collectionId":17,"collectionName":18,"created":29,"created_by":13,"id":30,"name":31,"updated":32,"updated_by":13},"2026-03-04 08:45:04.918Z","2oi71a8zf6vf68l","Video Call","2026-04-10 16:12:53.751Z",{"collectionId":17,"collectionName":18,"created":34,"created_by":13,"id":35,"name":36,"updated":37,"updated_by":13},"2026-03-04 08:45:05.158Z","836if8oqwpocqe3","Voice Chat","2026-04-10 16:12:53.881Z",{"collectionId":17,"collectionName":18,"created":39,"created_by":13,"id":40,"name":41,"updated":42,"updated_by":13},"2026-03-04 08:34:00.920Z","ecac9y661or1xka","WebSocket","2026-04-10 16:08:05.227Z",{"collectionId":17,"collectionName":18,"created":44,"created_by":13,"id":45,"name":46,"updated":47,"updated_by":13},"2026-03-04 08:45:05.632Z","cof4vbc2ts3emn3","WebSocket Signaling","2026-04-10 16:12:54.016Z",{"collectionId":17,"collectionName":18,"created":49,"created_by":13,"id":50,"name":51,"updated":52,"updated_by":13},"2026-03-04 08:45:05.836Z","37t2tvjs3vvty3x","WebRTC","2026-04-10 16:12:54.088Z",{"code":54,"collectionId":55,"collectionName":56,"created":57,"flag":58,"id":59,"is_default":60,"label":61,"updated":62},"en","pbc_1989393366","locales","2026-01-22 11:00:02.726Z","twemoji:flag-united-states","qv9c1llfov2d88z",false,"English","2026-04-10 15:42:46.825Z",{"category":64,"collectionId":65,"collectionName":66,"expand":67,"id":81,"views":82},"wqxt7ag2gn7xcmk","pbc_2105096300","school_blogs",{"category":68},{"blogIds":69,"collectionId":70,"collectionName":71,"created":72,"created_by":13,"id":64,"image":73,"image_alt":13,"image_path":74,"label":75,"name":76,"priority":77,"publish_at":78,"scheduled_at":13,"status":79,"updated":80,"updated_by":13},[],"sclcatblg987654321","school_category_blogs","2026-03-04 08:33:53.210Z","59ty92ns80w_15oc1implw.png","https:\u002F\u002Ftwsme-r2.tumwebsme.com\u002Fsclcatblg987654321\u002Fwqxt7ag2gn7xcmk\u002F59ty92ns80w_15oc1implw.png",{"en":76,"th":76},"Golang The Series",1,"2026-03-16 04:39:38.440Z","published","2026-04-25 02:32:15.470Z","kgoilbfmq2b2kt1",216,"xkmzgoufvuwfzdv",[20,25,30,35,40,45,50],"2025-12-15 01:58:24.825Z","Learn how to build real-time video calls and voice chat using WebRTC and WebSocket Signaling in Go, including production-ready architecture, code examples, and best practices.","ep116-realtime-voice-video-webrtc-websocket-go","2026-04-25 02:47:49.171Z",{"en":87}]