Compare commits

...

10 Commits
main ... back

Author SHA1 Message Date
moxitech
e8fe53008c upd ip get logic \ pisa popa 2024-11-07 15:33:41 +07:00
wow22831
2d94537192 ggaa 2024-11-07 15:10:32 +07:00
wow22831
0ccca0922c gg 2024-11-07 14:28:42 +07:00
wow22831
a8baa8f0ab legi 2024-11-07 00:06:15 +07:00
wow22831
dcdb5bfb13 connect 2024-10-29 02:35:09 +07:00
wow22831
741206df0d yra 2024-10-28 03:26:52 +07:00
wow22831
8c7b33c853 cock 2024-10-28 01:34:24 +07:00
wow22831
8a9b66b1fd zanovaw 2024-10-24 21:19:14 +07:00
wow22831
6027da6a92 zanova 2024-10-24 21:07:18 +07:00
wow22831
f146f46eb2 pervii 2024-10-24 21:00:29 +07:00
11 changed files with 359 additions and 43 deletions

View File

@ -3,13 +3,13 @@
build:
docker compose build
docker-compose build
dev: build
docker compose up -d --force-recreate
docker-compose up -d --force-recreate
devf: dev
docker compose logs -f
docker-compose logs -f
up:
docker compose up -d --force-recreate
@ -18,7 +18,7 @@ upf: up
docker compose logs -f
logs:
docker compose logs -f
docker-compose logs -f
stop:
@ -28,4 +28,4 @@ start:
docker compose start
drop:
docker-compose down --volumes
docker-compose down --volumes

33
package-lock.json generated
View File

@ -14,14 +14,16 @@
"@testing-library/jest-dom": "^5.11.9",
"@testing-library/react": "^13.3.0",
"@testing-library/user-event": "^12.6.3",
"@types/ip": "^1.1.3",
"@types/jest": "^26.0.20",
"@types/node": "^12.19.15",
"@types/node": "^12.20.55",
"@types/react": "^18.0.17",
"@types/react-dom": "^18.0.6",
"@types/react-router": "^5.1.11",
"@types/react-router-dom": "^5.1.7",
"history": "^4.9.0",
"ionicons": "^7.0.0",
"ip": "^2.0.1",
"os": "^0.1.2",
"os-browserify": "^0.3.0",
"peerjs": "^1.5.4",
@ -47,6 +49,7 @@
"workbox-streams": "^5.1.4"
},
"devDependencies": {
"@types/uuid": "^10.0.0",
"@typescript-eslint/eslint-plugin": "^5.59.2",
"@typescript-eslint/parser": "^5.59.2",
"eslint": "^8.35.0",
@ -3767,6 +3770,15 @@
"@types/node": "*"
}
},
"node_modules/@types/ip": {
"version": "1.1.3",
"resolved": "https://registry.npmjs.org/@types/ip/-/ip-1.1.3.tgz",
"integrity": "sha512-64waoJgkXFTYnCYDUWgSATJ/dXEBanVkaP5d4Sbk7P6U7cTTMhxVyROTckc6JKdwCrgnAjZMn0k3177aQxtDEA==",
"license": "MIT",
"dependencies": {
"@types/node": "*"
}
},
"node_modules/@types/istanbul-lib-coverage": {
"version": "2.0.6",
"resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.6.tgz",
@ -3852,7 +3864,8 @@
"node_modules/@types/node": {
"version": "12.20.55",
"resolved": "https://registry.npmjs.org/@types/node/-/node-12.20.55.tgz",
"integrity": "sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ=="
"integrity": "sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==",
"license": "MIT"
},
"node_modules/@types/node-forge": {
"version": "1.3.11",
@ -3999,6 +4012,13 @@
"resolved": "https://registry.npmjs.org/@types/trusted-types/-/trusted-types-2.0.7.tgz",
"integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw=="
},
"node_modules/@types/uuid": {
"version": "10.0.0",
"resolved": "https://registry.npmjs.org/@types/uuid/-/uuid-10.0.0.tgz",
"integrity": "sha512-7gqG38EyHgyP1S+7+xomFtL+ZNHcKv6DwNaCZmJmo1vgMugyF3TCnXVg4t1uk89mLNwnLtnY3TpOpCOyp1/xHQ==",
"dev": true,
"license": "MIT"
},
"node_modules/@types/ws": {
"version": "8.5.12",
"resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.12.tgz",
@ -8919,6 +8939,12 @@
"@stencil/core": "^4.0.3"
}
},
"node_modules/ip": {
"version": "2.0.1",
"resolved": "https://registry.npmjs.org/ip/-/ip-2.0.1.tgz",
"integrity": "sha512-lJUL9imLTNi1ZfXT+DU6rBBdbiKGBuay9B6xGSPVjUeQwaH1RIGqef8RZkUtHioLmSNpPR5M4HVKJGm1j8FWVQ==",
"license": "MIT"
},
"node_modules/ipaddr.js": {
"version": "2.2.0",
"resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-2.2.0.tgz",
@ -11467,7 +11493,8 @@
"node_modules/os": {
"version": "0.1.2",
"resolved": "https://registry.npmjs.org/os/-/os-0.1.2.tgz",
"integrity": "sha512-ZoXJkvAnljwvc56MbvhtKVWmSkzV712k42Is2mA0+0KTSRakq5XXuXpjZjgAt9ctzl51ojhQWakQQpmOvXWfjQ=="
"integrity": "sha512-ZoXJkvAnljwvc56MbvhtKVWmSkzV712k42Is2mA0+0KTSRakq5XXuXpjZjgAt9ctzl51ojhQWakQQpmOvXWfjQ==",
"license": "MIT"
},
"node_modules/os-browserify": {
"version": "0.3.0",

View File

@ -9,14 +9,16 @@
"@testing-library/jest-dom": "^5.11.9",
"@testing-library/react": "^13.3.0",
"@testing-library/user-event": "^12.6.3",
"@types/ip": "^1.1.3",
"@types/jest": "^26.0.20",
"@types/node": "^12.19.15",
"@types/node": "^12.20.55",
"@types/react": "^18.0.17",
"@types/react-dom": "^18.0.6",
"@types/react-router": "^5.1.11",
"@types/react-router-dom": "^5.1.7",
"history": "^4.9.0",
"ionicons": "^7.0.0",
"ip": "^2.0.1",
"os": "^0.1.2",
"os-browserify": "^0.3.0",
"peerjs": "^1.5.4",
@ -42,6 +44,7 @@
"workbox-streams": "^5.1.4"
},
"devDependencies": {
"@types/uuid": "^10.0.0",
"@typescript-eslint/eslint-plugin": "^5.59.2",
"@typescript-eslint/parser": "^5.59.2",
"eslint": "^8.35.0",

View File

@ -2,26 +2,42 @@
display: flex;
flex-direction: column;
align-items: center;
justify-content: space-around; /* Распределение элементов с равным отступом */
height: 100vh; /* Высота на весь экран для центрирования */
justify-content: space-around;
height: 100vh;
text-align: center;
}
.peer-id-container {
margin-bottom: 10px;
display: flex;
align-items: center;
}
.peer-id-text {
font-size: 16px;
color: #fff;
margin-right: 10px;
}
.copy-button {
background: none;
border: none;
color: #007aff;
cursor: pointer;
padding: 0;
}
.copy-notification {
position: fixed;
bottom: 20px;
left: 50%;
transform: translateX(-50%);
background-color: #4caf50;
color: white;
padding: 10px 20px;
border-radius: 5px;
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
font-size: 16px;
z-index: 1000;
}
.input-container {
@ -30,7 +46,7 @@
justify-content: center;
margin-bottom: 10px;
width: 100%;
max-width: 400px; /* Ограничение ширины для компактности */
max-width: 400px;
}
.call-info-container {
@ -43,13 +59,13 @@
.call-info-text {
font-size: 18px;
margin-bottom: 5px; /* Небольшой отступ снизу для Peer ID */
margin-bottom: 5px;
color: #ffffff;
}
.call-duration-text {
font-size: 18px;
margin-bottom: 15px; /* Отступ снизу для времени */
margin-bottom: 15px;
color: #ffffff;
}
@ -66,7 +82,7 @@
}
.custom-hang-up-button:hover svg {
transform: scale(1.1); /* Увеличение кнопки при наведении */
transform: scale(1.1);
}
.call-button {
@ -76,7 +92,14 @@
cursor: pointer;
font-size: 24px;
margin-left: 10px;
padding: 10px; /* Увеличение кнопки для удобства */
padding: 15px;
border-radius: 50%;
box-shadow: 0 4px 6px rgba(0, 0, 0, 0.1);
transition: background 0.3s ease;
}
.call-button:hover {
background: #005bb5;
}
.panel-optimizer {
@ -89,11 +112,17 @@
display: flex;
align-items: center;
justify-content: center;
height: 100vh; /* Центрирование всего контента по высоте экрана */
height: 100vh;
}
.drone-gif {
width: 300px; /* Задайте нужный размер */
width: 300px;
height: auto;
margin-bottom: 20px;
}
.screen-off {
background-color: #000;
color: #000;
opacity: 0;
}

View File

@ -1,11 +1,12 @@
"use strict";
import React, { useState, useEffect, useCallback } from 'react';
import './ExploreContainer.css';
import InputContainer from './Input';
import {
IonButton, IonContent, IonFooter, IonToolbar, IonText, IonModal, IonIcon
} from '@ionic/react';
import { call as callIcon, close as hangUpIcon } from 'ionicons/icons';
import { IonButton, IonContent, IonText, IonModal, IonIcon, IonLoading } from '@ionic/react';
import { call as callIcon, close as hangUpIcon, shirt } from 'ionicons/icons';
import Peer, { MediaConnection } from 'peerjs';
import './ExploreContainer.css';
import { randomBytes } from 'crypto';
const ExploreContainer: React.FC = () => {
const [callString, setCallString] = useState("");
@ -14,23 +15,61 @@ const ExploreContainer: React.FC = () => {
const [localStream, setLocalStream] = useState<MediaStream | null>(null);
const [connectionInfo, setConnectionInfo] = useState<string | null>(null);
const [isLoading, setIsLoading] = useState(false);
const [isConnecting, setIsConnecting] = useState(false);
const [isCallActive, setIsCallActive] = useState(false);
const [callDuration, setCallDuration] = useState<number>(0);
const [callInterval, setCallInterval] = useState<NodeJS.Timeout | null>(null);
const [currentCallId, setCurrentCallId] = useState<string | null>(null);
const [showCopyNotification, setShowCopyNotification] = useState(false);
const [ipv5, setIpv5] = useState<string | null>('')
const getLocalIP = async () => {
return new Promise((resolve, reject) => {
const peerConnection = new RTCPeerConnection({ });
// Обрабатываем события ICE-кандидатов, которые будут содержать IP-адрес
peerConnection.onicecandidate = (event) => {
if (event.candidate) {
const ip = event.candidate.candidate.split(" ")[4];
console.log(event.candidate)
resolve(ip);
peerConnection.onicecandidate = null; // Завершаем обработку
}
};
// Создаём пустой канал данных для инициирования связи
peerConnection.createDataChannel("");
peerConnection.createOffer()
.then((offer) => peerConnection.setLocalDescription(offer))
.catch((err) => reject(err));
});
};
useEffect(() => {
// Использование функции
getLocalIP().then((ip) => {
console.log("Local IP Address in function getLocalIp:", ip);
// Debug this func!!
setIpv5(ip as string) // ip is eq of "peer-string".local !!??
}).catch((err) => {
setIpv5(null)
console.error("Error getting IP address:", err);
});
const initializePeer = () => {
const newPeer = new Peer({ debug: 3 });
console.log(ipv5)
// ERROR CAST SET AS RANDOMIZE : "error_cast%s", where %s - random string or int
const newPeer = new Peer( ipv5 ? ipv5 : "error_cast",{ debug: 3 });
setPeer(newPeer);
newPeer.on('open', (id) => {
console.log(`Peer успешно открыт с ID: ${id}`);
console.log(`Peer successfully opened with ID: ${id}`);
setConnectionInfo(id);
});
newPeer.on('call', (call: MediaConnection) => handleIncomingCall(call));
newPeer.on('error', (err) => console.error('Ошибка Peer:', err));
newPeer.on('error', (err) => console.error('Peer error:', err));
return () => {
newPeer.destroy();
@ -42,43 +81,50 @@ const ExploreContainer: React.FC = () => {
}, []);
const handleIncomingCall = useCallback((call: MediaConnection) => {
console.log('Получен входящий звонок');
console.log('Incoming call received');
setCurrentCallId(call.peer);
setIsConnecting(true); // Show connecting screen
navigator.mediaDevices.getUserMedia({ audio: true }).then((stream) => {
setLocalStream(stream);
call.answer(stream);
call.on('stream', (remoteStream: MediaStream) => {
console.log('Получен удалённый поток');
console.log('Remote stream received');
playAudio(remoteStream);
setIsConnecting(false); // Hide connecting screen once audio is received
startCallTimer();
setIsCallActive(true);
});
}).catch((err) => {
console.error('Не удалось получить локальный поток для ответа', err);
console.error('Failed to get local stream for answer', err);
setIsConnecting(false); // Hide connecting screen in case of error
});
}, []);
const makeCall = useCallback(() => {
if (peer && callString) {
setIsLoading(true);
console.log(`Попытка звонка на: ${callString}`);
setIsConnecting(true); // Show connecting screen
console.log(`Attempting to call: ${callString}`);
navigator.mediaDevices.getUserMedia({ audio: true }).then((stream) => {
setLocalStream(stream);
const call = peer.call(callString, stream);
setCurrentCallId(callString);
call.on('stream', (remoteStream: MediaStream) => {
console.log('Получен удалённый поток во время вызова');
console.log('Remote stream received during call');
playAudio(remoteStream);
setIsLoading(false);
setIsConnecting(false); // Hide connecting screen once audio is received
startCallTimer();
setIsCallActive(true);
});
}).catch((err) => {
console.error('Не удалось получить локальный поток', err);
console.error('Failed to get local stream', err);
setIsLoading(false);
setIsConnecting(false); // Hide connecting screen on error
});
} else {
console.warn('Звонок невозможен: отсутствует Peer или callString');
console.warn('Call not possible: Peer or callString is missing');
}
}, [peer, callString]);
@ -106,9 +152,9 @@ const ExploreContainer: React.FC = () => {
const audioElement = document.createElement('audio');
audioElement.srcObject = stream;
audioElement.play().then(() => {
console.log('Аудио начато воспроизведение');
console.log('Audio playback started');
}).catch((error) => {
console.error('Ошибка воспроизведения аудио', error);
console.error('Audio playback error', error);
});
}, []);
@ -116,10 +162,14 @@ const ExploreContainer: React.FC = () => {
if (connectionInfo) {
navigator.clipboard.writeText(connectionInfo)
.then(() => {
console.log('Peer ID скопирован в буфер обмена');
console.log('Peer ID copied to clipboard');
setShowCopyNotification(true);
setTimeout(() => {
setShowCopyNotification(false);
}, 2000);
})
.catch((err) => {
console.error('Не удалось скопировать Peer ID', err);
console.error('Failed to copy Peer ID', err);
});
}
}, [connectionInfo]);
@ -127,11 +177,27 @@ const ExploreContainer: React.FC = () => {
return (
<>
<IonContent fullscreen={true} className="ion-padding">
{/* Connecting Indicator */}
<IonLoading
isOpen={isConnecting}
message={'Connecting...'}
/>
<div id="container">
<div className="peer-id-container">
<IonText className="peer-id-text">Ваш Peer ID: {connectionInfo}</IonText>
<button className="copy-button" onClick={copyPeerId}>Копировать</button>
<IonText className="peer-id-text">Your Peer ID: {connectionInfo}</IonText>
<button className="copy-button" onClick={copyPeerId}>
<svg width="51" height="51" fill="none" stroke="#34554a" strokeLinecap="round" strokeLinejoin="round" strokeWidth="2.5" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
<path d="M19.078 6H8.672A2.672 2.672 0 0 0 6 8.672v10.406a2.672 2.672 0 0 0 2.672 2.672h10.406a2.672 2.672 0 0 0 2.672-2.672V8.672A2.672 2.672 0 0 0 19.078 6Z"></path>
<path d="M17.977 6 18 4.875a2.633 2.633 0 0 0-2.625-2.625H5.25a3.009 3.009 0 0 0-3 3v10.125A2.633 2.633 0 0 0 4.875 18H6"></path>
</svg>
</button>
</div>
{showCopyNotification && (
<div className="copy-notification">
Peer ID copied
</div>
)}
<div className="input-container">
<InputContainer
callString={callString}
@ -144,11 +210,11 @@ const ExploreContainer: React.FC = () => {
</div>
</div>
{/* Модальное окно для звонка */}
{/* Call Modal */}
<IonModal isOpen={isCallActive}>
<IonContent className="ion-padding">
<div className="call-info-container">
{/* Добавляем гифку */}
{/* Add GIF */}
<img src="/monkey-monkey-with-drone.gif" alt="Monkey with Drone" className="drone-gif" />
<IonText className="call-info-text">Peer ID: {currentCallId}</IonText>
<IonText className="call-duration-text">{Math.floor(callDuration / 60)}:{callDuration % 60}</IonText>
@ -165,4 +231,4 @@ const ExploreContainer: React.FC = () => {
);
};
export default ExploreContainer;
export default ExploreContainer;

View File

@ -7,6 +7,14 @@
// code you'd like.
// You can also remove this file if you'd prefer not to use a
// service worker, and the Workbox build step will be skipped.
/** Гигачады
* Люднев Виталий
Баранова Софья
Белков Семен
Корнеев Владимир
Иноземцев Василий
Болганюк Константин
*/
import { clientsClaim } from 'workbox-core';
import { ExpirationPlugin } from 'workbox-expiration';

9
turn/Dockerfile Normal file
View File

@ -0,0 +1,9 @@
FROM golang:1.22.5-alpine3.20
LABEL author="moxitech"
WORKDIR /var/moxitech_goturn
COPY app /var/moxitech_goturn
RUN go get ./...;
RUN go build -o main cmd/main.go
EXPOSE 3434
CMD "./main"

71
turn/app/cmd/main.go Normal file
View File

@ -0,0 +1,71 @@
package main
import (
"flag"
"log"
"net"
"os"
"os/signal"
"regexp"
"strconv"
"syscall"
"github.com/pion/turn/v2"
)
func main() {
publicIP := flag.String("public-ip", "127.0.0.1", "") //"154.194.52.123"
port := flag.Int("port", 8081, "")
users := flag.String("users", "moxitech=moxitech", "") // user=pass,user=pass
realm := flag.String("realm", "callcoocall", "")
flag.Parse()
if len(*publicIP) == 0 {
log.Fatalf("public-ip is required")
}
if len(*users) == 0 {
log.Fatalf("'users' is required")
}
udpListener, err := net.ListenPacket("udp4", "0.0.0.0:"+strconv.Itoa(*port))
if err != nil {
log.Panicf("Ошибка создания TURN сервера: %s", err)
}
usersMap := map[string][]byte{}
for _, kv := range regexp.MustCompile(`(\w+)=(\w+)`).FindAllStringSubmatch(*users, -1) {
usersMap[kv[1]] = turn.GenerateAuthKey(kv[1], *realm, kv[2])
}
s, err := turn.NewServer(turn.ServerConfig{
Realm: *realm,
AuthHandler: func(username string, realm string, srcAddr net.Addr) ([]byte, bool) {
if key, ok := usersMap[username]; ok {
return key, true
}
return nil, false
},
PacketConnConfigs: []turn.PacketConnConfig{
{
PacketConn: udpListener,
RelayAddressGenerator: &turn.RelayAddressGeneratorPortRange{
RelayAddress: net.ParseIP(*publicIP),
Address: "0.0.0.0",
MinPort: 50000,
MaxPort: 55000,
},
},
},
})
if err != nil {
log.Panic(err)
}
sigs := make(chan os.Signal, 1)
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
<-sigs
if err = s.Close(); err != nil {
log.Panic(err)
}
}

14
turn/app/go.mod Normal file
View File

@ -0,0 +1,14 @@
module moxitech/turn
go 1.22.5
require (
github.com/pion/dtls/v2 v2.2.7 // indirect
github.com/pion/logging v0.2.2 // indirect
github.com/pion/randutil v0.1.0 // indirect
github.com/pion/stun v0.6.1 // indirect
github.com/pion/transport/v2 v2.2.1 // indirect
github.com/pion/turn/v2 v2.1.6 // indirect
golang.org/x/crypto v0.8.0 // indirect
golang.org/x/sys v0.9.0 // indirect
)

63
turn/app/go.sum Normal file
View File

@ -0,0 +1,63 @@
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/pion/dtls/v2 v2.2.7 h1:cSUBsETxepsCSFSxC3mc/aDo14qQLMSL+O6IjG28yV8=
github.com/pion/dtls/v2 v2.2.7/go.mod h1:8WiMkebSHFD0T+dIU+UeBaoV7kDhOW5oDCzZ7WZ/F9s=
github.com/pion/logging v0.2.2 h1:M9+AIj/+pxNsDfAT64+MAVgJO0rsyLnoJKCqf//DoeY=
github.com/pion/logging v0.2.2/go.mod h1:k0/tDVsRCX2Mb2ZEmTqNa7CWsQPc+YYCB7Q+5pahoms=
github.com/pion/randutil v0.1.0 h1:CFG1UdESneORglEsnimhUjf33Rwjubwj6xfiOXBa3mA=
github.com/pion/randutil v0.1.0/go.mod h1:XcJrSMMbbMRhASFVOlj/5hQial/Y8oH/HVo7TBZq+j8=
github.com/pion/stun v0.6.1 h1:8lp6YejULeHBF8NmV8e2787BogQhduZugh5PdhDyyN4=
github.com/pion/stun v0.6.1/go.mod h1:/hO7APkX4hZKu/D0f2lHzNyvdkTGtIy3NDmLR7kSz/8=
github.com/pion/transport/v2 v2.2.1 h1:7qYnCBlpgSJNYMbLCKuSY9KbQdBFoETvPNETv0y4N7c=
github.com/pion/transport/v2 v2.2.1/go.mod h1:cXXWavvCnFF6McHTft3DWS9iic2Mftcz1Aq29pGcU5g=
github.com/pion/turn/v2 v2.1.6 h1:Xr2niVsiPTB0FPtt+yAWKFUkU1eotQbGgpTIld4x1Gc=
github.com/pion/turn/v2 v2.1.6/go.mod h1:huEpByKKHix2/b9kmTAM3YoX6MKP+/D//0ClgUYR2fY=
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg=
github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU=
github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo=
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
golang.org/x/crypto v0.8.0 h1:pd9TJtTueMTVQXzk8E2XESSMQDj/U7OUu0PqJqPXQjQ=
golang.org/x/crypto v0.8.0/go.mod h1:mRqEX+O9/h5TFCrQhkgjo2yKi0yYA+9ecGkdQoHrywE=
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
golang.org/x/net v0.9.0/go.mod h1:d48xBJpPfHeWQsugry2m+kC02ZBRGRgulfHnEXEuWns=
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.7.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.9.0 h1:KS/R3tvhPqvJvwcKfnBHJwwthS11LRhmM5D59eEXa0s=
golang.org/x/sys v0.9.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
golang.org/x/term v0.7.0/go.mod h1:P32HKFT3hSsZrRxla30E9HqToFYAQPCMs/zFMBUFqPY=
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=

26
turn/docker-compose.yml Normal file
View File

@ -0,0 +1,26 @@
version: '3.9'
networks:
moxitech-network:
driver: bridge
services:
turn_server:
container_name: turn
build:
context: .
dockerfile: Dockerfile
networks:
moxitech-network:
restart: always