mirror of
https://github.com/skidoodle/ncore-stats.git
synced 2025-02-15 05:09:14 +01:00
Init
This commit is contained in:
commit
614504e933
9 changed files with 564 additions and 0 deletions
3
.env.example
Normal file
3
.env.example
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
NICK=
|
||||||
|
PASS=
|
||||||
|
PROFILE_1=https://ncore.pro/profile.php?id=1577943
|
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
.env
|
11
Dockerfile
Normal file
11
Dockerfile
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
FROM golang:alpine as builder
|
||||||
|
WORKDIR /build
|
||||||
|
COPY . .
|
||||||
|
RUN go build -o trackncore .
|
||||||
|
|
||||||
|
FROM alpine:latest
|
||||||
|
WORKDIR /app/
|
||||||
|
COPY --from=builder /build/trackncore .
|
||||||
|
COPY --from=builder /build/index.html .
|
||||||
|
EXPOSE 3000
|
||||||
|
CMD ["./trackncore"]
|
12
docker-compose.yaml
Normal file
12
docker-compose.yaml
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
services:
|
||||||
|
trackncore:
|
||||||
|
image: ghcr.io/skidoodle/trackncore:main
|
||||||
|
container_name: trackncore
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- "3000:3000"
|
||||||
|
volumes:
|
||||||
|
- data:/app
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
data:
|
15
go.mod
Normal file
15
go.mod
Normal file
|
@ -0,0 +1,15 @@
|
||||||
|
module trackncore
|
||||||
|
|
||||||
|
go 1.23
|
||||||
|
|
||||||
|
toolchain go1.23.2
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/PuerkitoBio/goquery v1.10.0
|
||||||
|
github.com/joho/godotenv v1.5.1
|
||||||
|
)
|
||||||
|
|
||||||
|
require (
|
||||||
|
github.com/andybalholm/cascadia v1.3.2 // indirect
|
||||||
|
golang.org/x/net v0.30.0 // indirect
|
||||||
|
)
|
42
go.sum
Normal file
42
go.sum
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
github.com/PuerkitoBio/goquery v1.10.0 h1:6fiXdLuUvYs2OJSvNRqlNPoBm6YABE226xrbavY5Wv4=
|
||||||
|
github.com/PuerkitoBio/goquery v1.10.0/go.mod h1:TjZZl68Q3eGHNBA8CWaxAN7rOU1EbDz3CWuolcO5Yu4=
|
||||||
|
github.com/andybalholm/cascadia v1.3.2 h1:3Xi6Dw5lHF15JtdcmAHD3i1+T8plmv7BQ/nsViSLyss=
|
||||||
|
github.com/andybalholm/cascadia v1.3.2/go.mod h1:7gtRlve5FxPPgIgX36uWBX58OdBsSS6lUvCFb+h7KvU=
|
||||||
|
github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0=
|
||||||
|
github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4=
|
||||||
|
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/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/net v0.30.0 h1:AcW1SDZMkb8IpzCdQUaIq2sP4sZ4zw+55h6ynffypl4=
|
||||||
|
golang.org/x/net v0.30.0/go.mod h1:2wGyMJ5iFasEhkwi13ChkO/t1ECNC4X4eBKkVFyYFlU=
|
||||||
|
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/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=
|
225
index.html
Normal file
225
index.html
Normal file
|
@ -0,0 +1,225 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>nCore Profile Stats</title>
|
||||||
|
<script src="//cdn.tailwindcss.com"></script>
|
||||||
|
<script src="//unpkg.com/boxicons"></script>
|
||||||
|
<style>
|
||||||
|
body {
|
||||||
|
background-color: #121212;
|
||||||
|
color: #efefef;
|
||||||
|
font-family: 'Roboto', sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
#historyModal {
|
||||||
|
transition: opacity 0.3s ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
#historyModal.hidden {
|
||||||
|
opacity: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#historyModal.visible {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
#profiles {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 1rem;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 768px) {
|
||||||
|
#profiles {
|
||||||
|
grid-template-columns: repeat(2, minmax(0, 1fr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 1024px) {
|
||||||
|
#profiles {
|
||||||
|
grid-template-columns: repeat(3, minmax(0, 1fr));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.profile-card {
|
||||||
|
flex: 0 1 calc(100% - 2rem);
|
||||||
|
max-width: 400px;
|
||||||
|
background-color: #1f1f1f;
|
||||||
|
color: #e5e5e5;
|
||||||
|
}
|
||||||
|
|
||||||
|
.profile-card button {
|
||||||
|
background-color: #3b3b3b;
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.profile-card button:hover {
|
||||||
|
background-color: #5b5b5b;
|
||||||
|
}
|
||||||
|
|
||||||
|
#historyModal .bg-gray-800 {
|
||||||
|
background-color: #2c2c2c;
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
|
#historyModal button {
|
||||||
|
background-color: #444444;
|
||||||
|
}
|
||||||
|
|
||||||
|
#historyModal button:hover {
|
||||||
|
background-color: #666666;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-button {
|
||||||
|
background-color: #444444;
|
||||||
|
color: #ffffff;
|
||||||
|
padding: 0.75rem 1.5rem;
|
||||||
|
border-radius: 0.375rem;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-button.active {
|
||||||
|
background-color: #ff5722;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-content {
|
||||||
|
display: none;
|
||||||
|
max-height: 300px;
|
||||||
|
overflow-y: auto;
|
||||||
|
padding: 1rem;
|
||||||
|
background-color: #333333;
|
||||||
|
border-radius: 0.375rem;
|
||||||
|
margin-top: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-content.active {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.button-container {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 640px) {
|
||||||
|
#historyModal .bg-gray-800 {
|
||||||
|
padding: 1.5rem;
|
||||||
|
width: 90%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body class="p-8 flex flex-col items-center">
|
||||||
|
<h1 class="text-4xl mb-10 font-semibold text-gray-100 text-center">nCore Profile Stats</h1>
|
||||||
|
|
||||||
|
<div id="profiles" class="w-full max-w-7xl"></div>
|
||||||
|
|
||||||
|
<div id="historyModal" class="fixed inset-0 hidden bg-black bg-opacity-50 flex items-center justify-center z-50" aria-hidden="true" aria-modal="true" tabindex="-1">
|
||||||
|
<div class="bg-gray-800 rounded-lg p-8 w-11/12 max-w-3xl text-white shadow-lg relative">
|
||||||
|
<h2 class="text-2xl mb-6 font-semibold" id="modal-profile-name">Historical Data for <span></span></h2>
|
||||||
|
|
||||||
|
<div class="button-container mb-4">
|
||||||
|
<button class="tab-button active" onclick="openTab(event, 'rankTab')">Rank</button>
|
||||||
|
<button class="tab-button" onclick="openTab(event, 'uploadTab')">Upload</button>
|
||||||
|
<button class="tab-button" onclick="openTab(event, 'downloadTab')">Download</button>
|
||||||
|
<button class="tab-button" onclick="openTab(event, 'pointsTab')">Points</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="rankTab" class="tab-content active">
|
||||||
|
<p id="rank-data"></p>
|
||||||
|
</div>
|
||||||
|
<div id="uploadTab" class="tab-content">
|
||||||
|
<p id="upload-data"></p>
|
||||||
|
</div>
|
||||||
|
<div id="downloadTab" class="tab-content">
|
||||||
|
<p id="download-data"></p>
|
||||||
|
</div>
|
||||||
|
<div id="pointsTab" class="tab-content">
|
||||||
|
<p id="points-data"></p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button class="mt-4 px-6 py-2 bg-red-600 hover:bg-red-700 text-white rounded" onclick="closeModal()">Close</button>
|
||||||
|
<button class="absolute top-2 right-2 text-white" onclick="closeModal()"><i class='bx bx-x text-2xl'></i></button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
let allProfileData = [];
|
||||||
|
|
||||||
|
async function fetchProfiles() {
|
||||||
|
const response = await fetch('/data');
|
||||||
|
allProfileData = await response.json();
|
||||||
|
|
||||||
|
const profilesDiv = document.getElementById('profiles');
|
||||||
|
profilesDiv.innerHTML = '';
|
||||||
|
|
||||||
|
const profileGroups = groupByOwner(allProfileData);
|
||||||
|
|
||||||
|
for (const [owner, records] of Object.entries(profileGroups)) {
|
||||||
|
const latestRecord = records[records.length - 1];
|
||||||
|
const profileCard = document.createElement('div');
|
||||||
|
profileCard.classList.add('profile-card', 'rounded-lg', 'shadow-lg', 'p-6', 'w-full', 'flex', 'flex-col', 'justify-between');
|
||||||
|
|
||||||
|
profileCard.innerHTML = `
|
||||||
|
<h2 class="text-2xl font-semibold mb-4"><i class='bx bxs-user text-2xl mr-2'></i>${owner}</h2>
|
||||||
|
<div class="space-y-2">
|
||||||
|
<p><i class='bx bx-trophy mr-2'></i> Rank: ${latestRecord.rank}</p>
|
||||||
|
<p><i class='bx bx-cloud-upload mr-2'></i> Upload: ${latestRecord.upload}</p>
|
||||||
|
<p><i class='bx bx-upload mr-2'></i> Current Upload: ${latestRecord.current_upload}</p>
|
||||||
|
<p><i class='bx bx-download mr-2'></i> Current Download: ${latestRecord.current_download}</p>
|
||||||
|
<p><i class='bx bx-coin mr-2'></i> Points: ${latestRecord.points}</p>
|
||||||
|
</div>
|
||||||
|
<button class="mt-6 px-4 py-2 text-white rounded" onclick="showHistory('${owner}', event)">View History</button>
|
||||||
|
`;
|
||||||
|
|
||||||
|
profilesDiv.appendChild(profileCard);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function groupByOwner(data) {
|
||||||
|
return data.reduce((acc, record) => {
|
||||||
|
acc[record.owner] = acc[record.owner] || [];
|
||||||
|
acc[record.owner].push(record);
|
||||||
|
return acc;
|
||||||
|
}, {});
|
||||||
|
}
|
||||||
|
|
||||||
|
function showHistory(owner, event) {
|
||||||
|
event.stopPropagation();
|
||||||
|
const historyModal = document.getElementById('historyModal');
|
||||||
|
document.getElementById('modal-profile-name').innerText = owner;
|
||||||
|
|
||||||
|
const profileHistory = allProfileData.filter(record => record.owner === owner);
|
||||||
|
|
||||||
|
document.getElementById('rank-data').innerHTML = profileHistory.map(record => `<p>${new Date(record.timestamp).toLocaleDateString()} - Rank: ${record.rank}</p>`).join('');
|
||||||
|
document.getElementById('upload-data').innerHTML = profileHistory.map(record => `<p>${new Date(record.timestamp).toLocaleDateString()} - Upload: ${record.upload}</p>`).join('');
|
||||||
|
document.getElementById('download-data').innerHTML = profileHistory.map(record => `<p>${new Date(record.timestamp).toLocaleDateString()} - Download: ${record.current_download}</p>`).join('');
|
||||||
|
document.getElementById('points-data').innerHTML = profileHistory.map(record => `<p>${new Date(record.timestamp).toLocaleDateString()} - Points: ${record.points}</p>`).join('');
|
||||||
|
|
||||||
|
historyModal.classList.remove('hidden');
|
||||||
|
setTimeout(() => historyModal.classList.add('visible'), 10);
|
||||||
|
}
|
||||||
|
|
||||||
|
function openTab(evt, tabName) {
|
||||||
|
document.querySelectorAll('.tab-button').forEach(button => button.classList.remove('active'));
|
||||||
|
document.querySelectorAll('.tab-content').forEach(content => content.classList.remove('active'));
|
||||||
|
|
||||||
|
evt.target.classList.add('active');
|
||||||
|
document.getElementById(tabName).classList.add('active');
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeModal() {
|
||||||
|
const historyModal = document.getElementById('historyModal');
|
||||||
|
historyModal.classList.remove('visible');
|
||||||
|
historyModal.classList.add('hidden');
|
||||||
|
}
|
||||||
|
|
||||||
|
fetchProfiles();
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
180
main.go
Normal file
180
main.go
Normal file
|
@ -0,0 +1,180 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/PuerkitoBio/goquery"
|
||||||
|
"github.com/joho/godotenv"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProfileData struct {
|
||||||
|
Owner string `json:"owner"`
|
||||||
|
Timestamp time.Time `json:"timestamp"`
|
||||||
|
Rank string `json:"rank"`
|
||||||
|
Upload string `json:"upload"`
|
||||||
|
CurrentUpload string `json:"current_upload"`
|
||||||
|
CurrentDownload string `json:"current_download"`
|
||||||
|
Points string `json:"points"`
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
profiles = map[string]string{}
|
||||||
|
jsonFile = "data.json"
|
||||||
|
nick string
|
||||||
|
pass string
|
||||||
|
client *http.Client
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
if err := godotenv.Load(); err != nil {
|
||||||
|
log.Fatal("Error loading .env file")
|
||||||
|
}
|
||||||
|
|
||||||
|
nick = os.Getenv("NICK")
|
||||||
|
pass = os.Getenv("PASS")
|
||||||
|
|
||||||
|
for i := 1; i <= 10; i++ {
|
||||||
|
profileKey := fmt.Sprintf("PROFILE_%d", i)
|
||||||
|
if profileURL := os.Getenv(profileKey); profileURL != "" {
|
||||||
|
profiles[profileURL] = profileURL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
client = &http.Client{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func fetchProfile(url string) (*ProfileData, error) {
|
||||||
|
req, err := http.NewRequest("GET", url, nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
req.Header.Set("Cookie", fmt.Sprintf("nick=%s; pass=%s", nick, pass))
|
||||||
|
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode != http.StatusOK {
|
||||||
|
return nil, fmt.Errorf("failed to fetch profile: %s", url)
|
||||||
|
}
|
||||||
|
|
||||||
|
doc, err := goquery.NewDocumentFromReader(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
owner := strings.TrimSpace(doc.Find(".fobox_fej").Text())
|
||||||
|
owner = strings.Replace(owner, " profilja", "", 1)
|
||||||
|
|
||||||
|
profile := &ProfileData{
|
||||||
|
Owner: owner,
|
||||||
|
Timestamp: time.Now(),
|
||||||
|
}
|
||||||
|
|
||||||
|
doc.Find(".userbox_tartalom_mini .profil_jobb_elso2").Each(func(i int, s *goquery.Selection) {
|
||||||
|
label := s.Text()
|
||||||
|
value := s.Next().Text()
|
||||||
|
|
||||||
|
switch label {
|
||||||
|
case "Helyezés:":
|
||||||
|
profile.Rank = value
|
||||||
|
case "Feltöltés:":
|
||||||
|
profile.Upload = value
|
||||||
|
case "Aktuális feltöltés:":
|
||||||
|
profile.CurrentUpload = value
|
||||||
|
case "Aktuális letöltés:":
|
||||||
|
profile.CurrentDownload = value
|
||||||
|
case "Pontok száma:":
|
||||||
|
profile.Points = value
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
return profile, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func readExistingProfiles() ([]ProfileData, error) {
|
||||||
|
if _, err := os.Stat(jsonFile); os.IsNotExist(err) {
|
||||||
|
return []ProfileData{}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Open(jsonFile)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
var profiles []ProfileData
|
||||||
|
byteValue, _ := io.ReadAll(file)
|
||||||
|
err = json.Unmarshal(byteValue, &profiles)
|
||||||
|
return profiles, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func logToJSON(profile *ProfileData) error {
|
||||||
|
existingProfiles, err := readExistingProfiles()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
existingProfiles = append(existingProfiles, *profile)
|
||||||
|
|
||||||
|
file, err := os.OpenFile(jsonFile, os.O_TRUNC|os.O_CREATE|os.O_WRONLY, 0644)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
enc := json.NewEncoder(file)
|
||||||
|
enc.SetIndent("", " ")
|
||||||
|
return enc.Encode(existingProfiles)
|
||||||
|
}
|
||||||
|
|
||||||
|
func dataHandler(w http.ResponseWriter, r *http.Request) {
|
||||||
|
profiles, err := readExistingProfiles()
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, "Could not read data", http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
w.Header().Set("Content-Type", "application/json")
|
||||||
|
json.NewEncoder(w).Encode(profiles)
|
||||||
|
}
|
||||||
|
|
||||||
|
func serveHTML(w http.ResponseWriter, r *http.Request) {
|
||||||
|
http.ServeFile(w, r, "index.html")
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
ticker := time.NewTicker(24 * time.Hour)
|
||||||
|
defer ticker.Stop()
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
for _, url := range profiles {
|
||||||
|
profile, err := fetchProfile(url)
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := logToJSON(profile); err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
<-ticker.C
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
http.HandleFunc("/data", dataHandler)
|
||||||
|
http.HandleFunc("/", serveHTML)
|
||||||
|
log.Fatal(http.ListenAndServe(":3000", nil))
|
||||||
|
}
|
75
readme.md
Normal file
75
readme.md
Normal file
|
@ -0,0 +1,75 @@
|
||||||
|
# nCore Profile Tracker
|
||||||
|
|
||||||
|
A simple Go project to scrape and track profile statistics (rank, upload, download, points) on nCore, the largest Hungarian BitTorrent tracker. The stats are displayed on a basic web interface and saved as JSON for historical tracking.
|
||||||
|
|
||||||
|
## Features
|
||||||
|
|
||||||
|
- Scrapes and logs profile stats from nCore.
|
||||||
|
- Serves a simple HTML dashboard to display the latest data.
|
||||||
|
- Provides a JSON API to fetch historical profile data.
|
||||||
|
- Automatically updates data every 24 hours.
|
||||||
|
|
||||||
|
## Setup
|
||||||
|
|
||||||
|
1. Clone the repo:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone https://github.com/skidoodle/ncore-profile-tracker.git
|
||||||
|
cd ncore-profile-tracker
|
||||||
|
```
|
||||||
|
|
||||||
|
2. Create a `.env` file with your nCore credentials and profile URLs:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
NICK=your_nick
|
||||||
|
PASS=your_password
|
||||||
|
PROFILE_1=https://ncore.pro/profile.php?id=1577943
|
||||||
|
```
|
||||||
|
|
||||||
|
### How to obtain `NICK` and `PASS`
|
||||||
|
|
||||||
|
- Open the developer tools in your browser (F12), go to the "Network" tab.
|
||||||
|
- Log in using "lower security" mode.
|
||||||
|
- Find the `login.php` request in the network activity.
|
||||||
|
- In the response headers, locate the `Set-Cookie` header, which will contain `nick=` and `pass=` values.
|
||||||
|
- Copy those values and add them to your `.env` file.
|
||||||
|
|
||||||
|
## Running with Docker Compose
|
||||||
|
|
||||||
|
To deploy the project using Docker Compose:
|
||||||
|
|
||||||
|
1. Create the following `docker-compose.yml` file:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
version: "3"
|
||||||
|
services:
|
||||||
|
trackncore:
|
||||||
|
image: ghcr.io/skidoodle/trackncore:main
|
||||||
|
container_name: trackncore
|
||||||
|
restart: unless-stopped
|
||||||
|
ports:
|
||||||
|
- "3000:3000"
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
volumes:
|
||||||
|
- data:/app
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
data:
|
||||||
|
```
|
||||||
|
|
||||||
|
2. Run the Docker Compose setup:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker-compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
3. Open `:3000` to view your stats.
|
||||||
|
|
||||||
|
### Updating
|
||||||
|
|
||||||
|
To pull the latest image and restart the service:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker-compose pull
|
||||||
|
docker-compose up -d
|
Loading…
Add table
Add a link
Reference in a new issue