Skip to content

Commit

Permalink
Merge pull request #108 from zediious/gameservers-keep-players
Browse files Browse the repository at this point in the history
Retain joined players in the database
  • Loading branch information
zediious authored Nov 6, 2023
2 parents fb0c585 + 5d820e6 commit 4be3639
Show file tree
Hide file tree
Showing 6 changed files with 187 additions and 14 deletions.
41 changes: 37 additions & 4 deletions raptorWeb/gameservers/admin.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@

from tinymce.widgets import TinyMCE

from raptorWeb.gameservers.models import Server, Player, ServerStatistic
from raptorWeb.gameservers.models import Server, Player, ServerStatistic, PlayerCountHistoric

class ServerAdminForm(ModelForm):
class Meta:
Expand Down Expand Up @@ -70,20 +70,52 @@ class PlayerAdmin(admin.ModelAdmin):
('Player Information', {
'fields': (
'server',
'name')
'name',
'online',
'last_online')
}),
)

readonly_fields: tuple[str] = (
'server',
'name'
'name',
'online',
'last_online'
)

search_fields: list[str] = [
'name',
]

list_display: list[str] = ['name', 'server']
list_display: list[str] = ['name', 'server', 'online', 'last_online']

def has_add_permission(self, request, obj=None):
return False

def has_change_permission(self, request, obj=None):
return False

class PlayerCountHistoricAdmin(admin.ModelAdmin):
"""
Object defining behavior and display of
PlayerCountHistoric in the Django admin interface.
"""
fieldsets: tuple[tuple[str, dict[str, tuple[str]]]] = (
('Player Information', {
'fields': (
'server',
'player_count',
'checked_time')
}),
)

readonly_fields: tuple[str] = (
'server',
'player_count',
'checked_time'
)

list_display: list[str] = ['server', 'player_count', 'checked_time']

def has_add_permission(self, request, obj=None):
return False
Expand Down Expand Up @@ -117,4 +149,5 @@ def has_change_permission(self, request, obj=None):

admin.site.register(Server, ServerAdmin)
admin.site.register(Player, PlayerAdmin)
admin.site.register(PlayerCountHistoric, PlayerCountHistoricAdmin)
admin.site.register(ServerStatistic, ServerStatisticAdmin)
18 changes: 18 additions & 0 deletions raptorWeb/gameservers/migrations/0004_player_online.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
# Generated by Django 4.2.7 on 2023-11-06 18:16

from django.db import migrations, models


class Migration(migrations.Migration):

dependencies = [
('gameservers', '0003_server_archived'),
]

operations = [
migrations.AddField(
model_name='player',
name='online',
field=models.BooleanField(default=False),
),
]
19 changes: 19 additions & 0 deletions raptorWeb/gameservers/migrations/0005_player_last_online.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
# Generated by Django 4.2.7 on 2023-11-06 18:35

import datetime
from django.db import migrations, models


class Migration(migrations.Migration):

dependencies = [
('gameservers', '0004_player_online'),
]

operations = [
migrations.AddField(
model_name='player',
name='last_online',
field=models.DateTimeField(auto_now_add=True, verbose_name='Last Online'),
),
]
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
# Generated by Django 4.2.7 on 2023-11-06 19:59

from django.db import migrations, models
import django.db.models.deletion


class Migration(migrations.Migration):

dependencies = [
('gameservers', '0005_player_last_online'),
]

operations = [
migrations.AlterField(
model_name='player',
name='server',
field=models.ForeignKey(default=0, on_delete=django.db.models.deletion.PROTECT, to='gameservers.server'),
),
migrations.CreateModel(
name='PlayerCountHistoric',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('player_count', models.IntegerField(verbose_name='Players Online')),
('checked_time', models.DateTimeField(auto_now_add=True, verbose_name='Time of Query')),
('server', models.ForeignKey(default=0, on_delete=django.db.models.deletion.CASCADE, to='gameservers.server')),
],
options={
'verbose_name': 'Historic Player Count',
'verbose_name_plural': 'Historic Player Counts',
},
),
]
88 changes: 78 additions & 10 deletions raptorWeb/gameservers/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
from typing import Optional

from django.db import models
from django.utils.timezone import localtime
from django.utils.timezone import localtime, now
from django.conf import settings

from mcstatus import JavaServer
Expand Down Expand Up @@ -42,6 +42,9 @@ def _update_announcement_count(server: Server) -> None:
server=server
).count()

all_players = Player.objects.all()
online_players = []

if do_query == True:
try:
serverJSON: QueryResponse = JavaServer(
Expand All @@ -53,15 +56,33 @@ def _update_announcement_count(server: Server) -> None:
server.server_state = True
_update_announcement_count(server)
server.save()
[Player.objects.create(
name=player,
server=server).save() for player in serverJSON.players.names]

for player in serverJSON.players.names:
checked_player = all_players.filter(name=player).first()
# If a Player exists, update their information
if checked_player is not None:
online_players.append(checked_player.name)
checked_player.server = server
checked_player.online = True
checked_player.last_online = now()
checked_player.save()
# If not, create a new Player
else:
online_players.append(player)
new_player = Player.objects.create(
name=player,
server=server,
online=True
)
new_player.save()

except TimeoutError:
_set_offline_server(server)

else:
_set_offline_server(server)

return online_players

def poll_servers(self, servers: list['Server'], statistic_model: 'ServerStatistic') -> None:
if statistic_model.time_last_polled == None:
Expand All @@ -72,8 +93,8 @@ def poll_servers(self, servers: list['Server'], statistic_model: 'ServerStatisti

if minutes_since_poll > 1 or self._has_run == False:
statistic_model.total_player_count = 0
Player.objects.all().delete()


all_online_players = []
for server in servers:
if (server.server_address == "Default"
or server.in_maintenance == True
Expand All @@ -83,8 +104,19 @@ def poll_servers(self, servers: list['Server'], statistic_model: 'ServerStatisti
do_query = False)

else:
self._query_and_update_server(server)
online_players = self._query_and_update_server(server)
all_online_players.extend(online_players)
PlayerCountHistoric.objects.create(
server=server,
player_count=server.player_count
)
statistic_model.total_player_count += server.player_count

# Mark players who were not queried as online, but are marked as online in the database, as offline.
newly_offline_players = Player.objects.filter(online=True).exclude(name__in=all_online_players)
for player in newly_offline_players:
player.online = False
player.save()

self._has_run = True
statistic_model.time_last_polled = localtime()
Expand Down Expand Up @@ -338,16 +370,23 @@ class Meta:

class Player(models.Model):
"""
Playesr that are currently logged in to a Server
Players that have joined a server at some point.
"""
server = models.ForeignKey(
Server,
default=0,
on_delete=models.CASCADE)
default=0,
on_delete=models.PROTECT)

name = models.CharField(
max_length=50,
unique=True)

online = models.BooleanField(
default=False)

last_online = models.DateTimeField(
verbose_name="Last Online",
auto_now_add=True)

def __str__(self):
return self.name
Expand All @@ -361,3 +400,32 @@ def get_server(self):
class Meta:
verbose_name = "Player"
verbose_name_plural = "Players"


class PlayerCountHistoric(models.Model):
"""
The total count of players on a server at a
specific point in time. These are created each
time servers are queried, for each server.
"""
server = models.ForeignKey(
Server,
default=0,
on_delete=models.CASCADE)

player_count = models.IntegerField(
verbose_name="Players Online")

checked_time = models.DateTimeField(
verbose_name="Time of Query",
auto_now_add=True)

def get_player_count(self):
return self.player_count

def get_server(self):
return self.server

class Meta:
verbose_name = "Historic Player Count"
verbose_name_plural = "Historic Player Counts"
3 changes: 3 additions & 0 deletions raptorWeb/gameservers/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,9 @@ class Player_List(ListView):
the Server Manager.
"""
model: Player = Player

def get_queryset(self) -> Player.objects:
return Player.objects.filter(online=True)

def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
Expand Down

0 comments on commit 4be3639

Please sign in to comment.