first commit
This commit is contained in:
commit
283264a261
46
CloudlogUDP.go
Normal file
46
CloudlogUDP.go
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"git.rouggy.com/CloudlogUDP/cloudlog"
|
||||||
|
"git.rouggy.com/CloudlogUDP/clublog"
|
||||||
|
"git.rouggy.com/CloudlogUDP/config"
|
||||||
|
"git.rouggy.com/CloudlogUDP/qrzcom"
|
||||||
|
"git.rouggy.com/CloudlogUDP/qso"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
fmt.Println("Loading config file...")
|
||||||
|
|
||||||
|
cfg := config.NewConfig()
|
||||||
|
|
||||||
|
fmt.Printf("Launching the server on port %v\n", cfg.UDP.Port)
|
||||||
|
udpServer, err := net.ListenPacket("udp", cfg.UDP.Port)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
defer udpServer.Close()
|
||||||
|
fmt.Printf("Server listening on port %v for incoming QSO\n", cfg.UDP.Port)
|
||||||
|
for {
|
||||||
|
buf := make([]byte, 4096)
|
||||||
|
_, _, err := udpServer.ReadFrom(buf)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
msg := string(buf[:])
|
||||||
|
msgSplit := strings.Split(msg, "<")
|
||||||
|
|
||||||
|
qso := qso.NewQSO()
|
||||||
|
qso.ExtractQSOData(msgSplit)
|
||||||
|
|
||||||
|
cloudlog.SendCloudlogMsg(qso, cfg)
|
||||||
|
qrzcom.SendQRZcomlogMsg(qso, cfg)
|
||||||
|
clublog.SendClublogMsg(qso, cfg)
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
51
cloudlog/cloudlog.go
Normal file
51
cloudlog/cloudlog.go
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
package cloudlog
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"git.rouggy.com/CloudlogUDP/config"
|
||||||
|
"git.rouggy.com/CloudlogUDP/qso"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CloudlogAPIString struct {
|
||||||
|
Key string `json:"key"`
|
||||||
|
StationProfileID string `json:"station_profile_id"`
|
||||||
|
Type string `json:"type"`
|
||||||
|
QSOData string `json:"string"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func SendCloudlogMsg(qso qso.QSO, config config.Config) {
|
||||||
|
QSODataJson := qso.Call + qso.Band + qso.Mode + qso.Freq + qso.QSODate + qso.TimeOn + qso.RSTR + qso.RSTS + qso.GridSquare
|
||||||
|
|
||||||
|
jsonStr := CloudlogAPIString{
|
||||||
|
Key: config.CloudLog.API,
|
||||||
|
StationProfileID: "1",
|
||||||
|
Type: "adif",
|
||||||
|
QSOData: QSODataJson,
|
||||||
|
}
|
||||||
|
|
||||||
|
b, err := json.Marshal(jsonStr)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("error:", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
req, err := http.NewRequest("POST", config.CloudLog.QSOUrl, bytes.NewBuffer(b))
|
||||||
|
req.Header.Set("X-Custom-Header", "myvalue")
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
|
||||||
|
client := &http.Client{}
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
// body, _ := io.ReadAll(resp.Body)
|
||||||
|
// fmt.Println("response Body:", string(body))
|
||||||
|
|
||||||
|
fmt.Printf("QSO Details:\nCall: %v\nBand: %v\nFreq: %v\n", qso.Call, qso.Band, qso.Freq)
|
||||||
|
fmt.Println("QSO Uploaded to Cloudlog")
|
||||||
|
}
|
47
clublog/clublog.go
Normal file
47
clublog/clublog.go
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
package clublog
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"git.rouggy.com/CloudlogUDP/config"
|
||||||
|
"git.rouggy.com/CloudlogUDP/qso"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ClublogAPI struct {
|
||||||
|
Email string `json:"email"`
|
||||||
|
Password string `json:"password"`
|
||||||
|
Callsign string `json:"callsign"`
|
||||||
|
ADIF string `json:"adif"`
|
||||||
|
APIKey string `json:"api"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func SendClublogMsg(qso qso.QSO, config config.Config) {
|
||||||
|
QSOData := qso.Call + qso.Band + qso.Mode + qso.Freq + qso.QSODate + qso.TimeOn + qso.RSTR + qso.RSTS + qso.GridSquare + "<eor>"
|
||||||
|
|
||||||
|
apiUrl := config.Clublog.URL
|
||||||
|
data := url.Values{}
|
||||||
|
data.Set("api", config.Clublog.API)
|
||||||
|
data.Set("email", config.Clublog.EMail)
|
||||||
|
data.Set("password", config.Clublog.Password)
|
||||||
|
data.Set("callsign", config.Clublog.Callsign)
|
||||||
|
data.Set("adif", QSOData)
|
||||||
|
|
||||||
|
u, _ := url.ParseRequestURI(apiUrl)
|
||||||
|
urlStr := u.String()
|
||||||
|
|
||||||
|
client := &http.Client{}
|
||||||
|
r, _ := http.NewRequest(http.MethodPost, urlStr, strings.NewReader(data.Encode())) // URL-encoded payload
|
||||||
|
r.Header.Add("Content-Type", "application/x-www-form-urlencoded")
|
||||||
|
|
||||||
|
resp, _ := client.Do(r)
|
||||||
|
|
||||||
|
if http.StatusOK == 200 {
|
||||||
|
fmt.Println("QSO Uploaded to Clublog")
|
||||||
|
} else {
|
||||||
|
fmt.Printf("Error uploading QSO to Clublog %v", resp.Status)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
49
config/config.go
Normal file
49
config/config.go
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
package config
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"gopkg.in/yaml.v2"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Config from YAML
|
||||||
|
type Config struct {
|
||||||
|
CloudLog struct {
|
||||||
|
Server string `yaml:"server"`
|
||||||
|
QSOUrl string `yaml:"qso_url"`
|
||||||
|
API string `yaml:"api"`
|
||||||
|
} `yaml:"cloudlog"`
|
||||||
|
UDP struct {
|
||||||
|
Port string `yaml:"port"`
|
||||||
|
} `yaml:"udp"`
|
||||||
|
|
||||||
|
QRZ struct {
|
||||||
|
API string `yaml:"api"`
|
||||||
|
Callsign string `yaml:"callsign"`
|
||||||
|
}
|
||||||
|
Clublog struct {
|
||||||
|
EMail string `yaml:"email"`
|
||||||
|
Password string `yaml:"password"`
|
||||||
|
Callsign string `yaml:"callsign"`
|
||||||
|
API string `yaml:"api"`
|
||||||
|
URL string `yaml:"url"`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewConfig() Config {
|
||||||
|
var c Config
|
||||||
|
ex, err := os.Executable()
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
|
configFile := filepath.Dir(ex) + "/config/config.yaml"
|
||||||
|
yamlFile, err := ioutil.ReadFile(configFile)
|
||||||
|
err = yaml.Unmarshal([]byte(yamlFile), &c)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("error: %v", err)
|
||||||
|
}
|
||||||
|
return c
|
||||||
|
}
|
20
config/config.yaml
Normal file
20
config/config.yaml
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
# Cloudlog Setup
|
||||||
|
# Generate a Read/Write API Key cloudlog/api/help
|
||||||
|
cloudlog:
|
||||||
|
server: "https://log.rouggy.com"
|
||||||
|
qso_url: "https://log.rouggy.com/index.php/api/qso/"
|
||||||
|
api: "cl65388812a62d1"
|
||||||
|
|
||||||
|
udp:
|
||||||
|
port: ":2240"
|
||||||
|
|
||||||
|
qrz:
|
||||||
|
api: "4AEC-1886-2E4E-C3AF"
|
||||||
|
callsign: "XV9Q"
|
||||||
|
|
||||||
|
clublog:
|
||||||
|
email: "legreg002@hotmail.com"
|
||||||
|
password: "89DGgg290379"
|
||||||
|
callsign: "XV9Q"
|
||||||
|
api: "5767f19333363a9ef432ee9cd4141fe76b8adf38"
|
||||||
|
url: "https://clublog.org/realtime.php"
|
5
go.mod
Normal file
5
go.mod
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
module git.rouggy.com/CloudlogUDP
|
||||||
|
|
||||||
|
go 1.21.4
|
||||||
|
|
||||||
|
require gopkg.in/yaml.v2 v2.4.0
|
4
go.sum
Normal file
4
go.sum
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
|
||||||
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
|
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
|
||||||
|
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
|
53
qrzcom/qrzcom.go
Normal file
53
qrzcom/qrzcom.go
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
package qrzcom
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
|
||||||
|
"git.rouggy.com/CloudlogUDP/config"
|
||||||
|
"git.rouggy.com/CloudlogUDP/qso"
|
||||||
|
)
|
||||||
|
|
||||||
|
type QRZComAPIString struct {
|
||||||
|
Key string `json:"KEY"`
|
||||||
|
Action string `json:"ACTION"`
|
||||||
|
Adif string `json:"ADIF"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func SendQRZcomlogMsg(qso qso.QSO, config config.Config) {
|
||||||
|
|
||||||
|
QSODataJson := qso.Call + qso.Band + qso.Mode + qso.Freq + qso.QSODate + qso.TimeOn + qso.RSTR + qso.RSTS + qso.StationCallsign + "<eor>"
|
||||||
|
|
||||||
|
jsonStr := QRZComAPIString{
|
||||||
|
Key: config.QRZ.API,
|
||||||
|
Action: "INSERT",
|
||||||
|
Adif: QSODataJson,
|
||||||
|
}
|
||||||
|
|
||||||
|
b, err := json.Marshal(jsonStr)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("error:", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
params := url.Values{}
|
||||||
|
params.Add("KEY", jsonStr.Key)
|
||||||
|
params.Add("ACTION", "INSERT")
|
||||||
|
params.Add("ADIF", jsonStr.Adif)
|
||||||
|
|
||||||
|
urlQRZ := fmt.Sprint("https://logbook.qrz.com/api?" + params.Encode())
|
||||||
|
|
||||||
|
req, err := http.NewRequest("POST", urlQRZ, bytes.NewBuffer(b))
|
||||||
|
|
||||||
|
client := &http.Client{}
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
fmt.Println("QSO Uploaded to QRZ.com")
|
||||||
|
|
||||||
|
}
|
56
qso/qso.go
Normal file
56
qso/qso.go
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
package qso
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type QSO struct {
|
||||||
|
Call string
|
||||||
|
Band string
|
||||||
|
Mode string
|
||||||
|
Freq string
|
||||||
|
QSODate string
|
||||||
|
TimeOn string
|
||||||
|
RSTR string
|
||||||
|
RSTS string
|
||||||
|
Country string
|
||||||
|
GridSquare string
|
||||||
|
StationCallsign string
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewQSO() QSO {
|
||||||
|
return *&QSO{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (q *QSO) ExtractQSOData(msg []string) {
|
||||||
|
for _, entry := range msg {
|
||||||
|
entry = strings.Trim(entry, " ")
|
||||||
|
|
||||||
|
switch {
|
||||||
|
case strings.Contains(entry, "FREQ:"):
|
||||||
|
q.Freq = "<" + entry
|
||||||
|
case strings.Contains(entry, "CALL:"):
|
||||||
|
q.Call = "<" + entry
|
||||||
|
case strings.Contains(entry, "BAND:"):
|
||||||
|
q.Band = "<" + entry
|
||||||
|
case strings.Contains(entry, "MODE:"):
|
||||||
|
q.Mode = "<" + entry
|
||||||
|
case strings.Contains(entry, "QSO_DATE:"):
|
||||||
|
if q.QSODate == "" {
|
||||||
|
q.QSODate = "<" + entry
|
||||||
|
}
|
||||||
|
case strings.Contains(entry, "TIME_ON:"):
|
||||||
|
q.TimeOn = "<" + entry
|
||||||
|
case strings.Contains(entry, "RST_RCVD:"):
|
||||||
|
q.RSTR = "<" + entry
|
||||||
|
case strings.Contains(entry, "RST_SENT:"):
|
||||||
|
q.RSTS = "<" + entry
|
||||||
|
case strings.Contains(entry, "COUNTRY:"):
|
||||||
|
q.Country = "<" + entry
|
||||||
|
case strings.Contains(entry, "GRIDSQUARE:"):
|
||||||
|
q.GridSquare = "<" + entry
|
||||||
|
case strings.Contains(entry, "STATION_CALLSIGN:"):
|
||||||
|
q.StationCallsign = "<" + entry
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user