Compare commits

...

4 Commits

Author SHA1 Message Date
kingmo888
71c6f290d1 更新说明 2024-09-25 12:46:58 +08:00
kingmo888
d04154aa7e 当客户端未登录时,前端所属用户显示[未登录] 2024-09-10 00:07:08 +08:00
kingmo888
175e1165db 解决反代时IP不正确的情况 2024-09-09 23:56:47 +08:00
kingmo888
7b8951d441 add ip_address for model resolves #137, resolves #15 2024-09-09 10:50:13 +08:00
7 changed files with 79 additions and 30 deletions

View File

@ -6,7 +6,7 @@
<p align="center"> <p align="center">
<i>一个 python 实现的 Rustdesk API 接口,支持 WebUI 管理</i> <i>一个 python 实现的 Rustdesk API 接口,支持 WebUI 管理</i>
<br/> <br/>
<img src ="https://img.shields.io/badge/Version-1.5.1-blueviolet.svg"/> <img src ="https://img.shields.io/badge/Version-1.5.2-blueviolet.svg"/>
<img src ="https://img.shields.io/badge/Python-3.7|3.8|3.9|3.10|3.11-blue.svg" /> <img src ="https://img.shields.io/badge/Python-3.7|3.8|3.9|3.10|3.11-blue.svg" />
<img src ="https://img.shields.io/badge/Django-3.2+|4.x-yelow.svg" /> <img src ="https://img.shields.io/badge/Django-3.2+|4.x-yelow.svg" />
<br/> <br/>
@ -15,6 +15,9 @@
</p> </p>
# 1.2.3版本与1.2.6+版本区别 # 1.2.3版本与1.2.6+版本区别
#### **请使用自定义key因不填写key、或使用服务端自动生成的key而引起的链接超时或建立链接时间过长的问题不在本项目解决范围内。**
> rustdesk官方在其新版服务端中已[强制要求key](https://rustdesk.com/docs/zh-cn/self-host/rustdesk-server-oss/install/#key)(rustdesk-server版本号大概>=1.1.10) > rustdesk官方在其新版服务端中已[强制要求key](https://rustdesk.com/docs/zh-cn/self-host/rustdesk-server-oss/install/#key)(rustdesk-server版本号大概>=1.1.10)
- rustdesk版本<=1.2.3, 服务端请配合使用rustdesk-server<=1.1.10 - rustdesk版本<=1.2.3, 服务端请配合使用rustdesk-server<=1.1.10

View File

@ -0,0 +1,18 @@
# Generated by Django 3.2 on 2024-09-09 10:37
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('api', '0007_alter_rustdesdevice_options_and_more'),
]
operations = [
migrations.AddField(
model_name='rustdesdevice',
name='ip_address',
field=models.CharField(blank=True, max_length=60, verbose_name='IP'),
),
]

View File

@ -3,6 +3,7 @@ from django.db import models
from django.contrib import admin from django.contrib import admin
from django.utils.translation import gettext as _ from django.utils.translation import gettext as _
class RustDeskToken(models.Model): class RustDeskToken(models.Model):
''' Token ''' Token
''' '''
@ -12,17 +13,19 @@ class RustDeskToken(models.Model):
uuid = models.CharField(verbose_name=_('uuid'), max_length=60) uuid = models.CharField(verbose_name=_('uuid'), max_length=60)
access_token = models.CharField(verbose_name=_('access_token'), max_length=60, blank=True) access_token = models.CharField(verbose_name=_('access_token'), max_length=60, blank=True)
create_time = models.DateTimeField(verbose_name=_('登录时间'), auto_now_add=True) create_time = models.DateTimeField(verbose_name=_('登录时间'), auto_now_add=True)
#expire_time = models.DateTimeField(verbose_name='过期时间') # expire_time = models.DateTimeField(verbose_name='过期时间')
class Meta: class Meta:
ordering = ('-username',) ordering = ('-username',)
verbose_name = "Token" verbose_name = "Token"
verbose_name_plural = _("Token列表") verbose_name_plural = _("Token列表")
class RustDeskTokenAdmin(admin.ModelAdmin): class RustDeskTokenAdmin(admin.ModelAdmin):
list_display = ('username', 'uid') list_display = ('username', 'uid')
search_fields = ('username', 'uid') search_fields = ('username', 'uid')
list_filter = ('create_time', ) #过滤器 list_filter = ('create_time', ) # 过滤器
class RustDeskTag(models.Model): class RustDeskTag(models.Model):
''' Tags ''' Tags
@ -30,17 +33,18 @@ class RustDeskTag(models.Model):
uid = models.CharField(verbose_name=_('所属用户ID'), max_length=16) uid = models.CharField(verbose_name=_('所属用户ID'), max_length=16)
tag_name = models.CharField(verbose_name=_('标签名称'), max_length=60) tag_name = models.CharField(verbose_name=_('标签名称'), max_length=60)
tag_color = models.CharField(verbose_name=_('标签颜色'), max_length=60, blank=True) tag_color = models.CharField(verbose_name=_('标签颜色'), max_length=60, blank=True)
class Meta: class Meta:
ordering = ('-uid',) ordering = ('-uid',)
verbose_name = "Tags" verbose_name = "Tags"
verbose_name_plural = _("Tags列表") verbose_name_plural = _("Tags列表")
class RustDeskTagAdmin(admin.ModelAdmin): class RustDeskTagAdmin(admin.ModelAdmin):
list_display = ('tag_name', 'uid', 'tag_color') list_display = ('tag_name', 'uid', 'tag_color')
search_fields = ('tag_name', 'uid') search_fields = ('tag_name', 'uid')
list_filter = ('uid', ) list_filter = ('uid', )
class RustDeskPeer(models.Model): class RustDeskPeer(models.Model):
''' Pees ''' Pees
@ -53,19 +57,19 @@ class RustDeskPeer(models.Model):
platform = models.CharField(verbose_name=_('平台'), max_length=30) platform = models.CharField(verbose_name=_('平台'), max_length=30)
tags = models.CharField(verbose_name=_('标签'), max_length=30) tags = models.CharField(verbose_name=_('标签'), max_length=30)
rhash = models.CharField(verbose_name=_('设备链接密码'), max_length=60) rhash = models.CharField(verbose_name=_('设备链接密码'), max_length=60)
class Meta: class Meta:
ordering = ('-username',) ordering = ('-username',)
verbose_name = "Peers" verbose_name = "Peers"
verbose_name_plural = _("Peers列表" ) verbose_name_plural = _("Peers列表")
class RustDeskPeerAdmin(admin.ModelAdmin): class RustDeskPeerAdmin(admin.ModelAdmin):
list_display = ('rid', 'uid', 'username', 'hostname', 'platform', 'alias', 'tags') list_display = ('rid', 'uid', 'username', 'hostname', 'platform', 'alias', 'tags')
search_fields = ('deviceid', 'alias') search_fields = ('deviceid', 'alias')
list_filter = ('rid', 'uid', ) list_filter = ('rid', 'uid', )
class RustDesDevice(models.Model): class RustDesDevice(models.Model):
rid = models.CharField(verbose_name=_('客户端ID'), max_length=60, blank=True) rid = models.CharField(verbose_name=_('客户端ID'), max_length=60, blank=True)
cpu = models.CharField(verbose_name='CPU', max_length=100) cpu = models.CharField(verbose_name='CPU', max_length=100)
@ -75,21 +79,24 @@ class RustDesDevice(models.Model):
uuid = models.CharField(verbose_name='uuid', max_length=100) uuid = models.CharField(verbose_name='uuid', max_length=100)
username = models.CharField(verbose_name=_('系统用户名'), max_length=100, blank=True) username = models.CharField(verbose_name=_('系统用户名'), max_length=100, blank=True)
version = models.CharField(verbose_name=_('客户端版本'), max_length=100) version = models.CharField(verbose_name=_('客户端版本'), max_length=100)
ip_address = models.CharField(verbose_name=_('IP'), max_length=60, blank=True)
create_time = models.DateTimeField(verbose_name=_('设备注册时间'), auto_now_add=True) create_time = models.DateTimeField(verbose_name=_('设备注册时间'), auto_now_add=True)
update_time = models.DateTimeField(verbose_name=('设备更新时间'), auto_now=True, blank=True) update_time = models.DateTimeField(verbose_name=('设备更新时间'), auto_now=True, blank=True)
class Meta: class Meta:
ordering = ('-rid',) ordering = ('-rid',)
verbose_name = _("设备") verbose_name = _("设备")
verbose_name_plural = _("设备列表" ) verbose_name_plural = _("设备列表")
class RustDesDeviceAdmin(admin.ModelAdmin): class RustDesDeviceAdmin(admin.ModelAdmin):
list_display = ('rid', 'hostname', 'memory', 'uuid', 'version', 'create_time', 'update_time') list_display = ('rid', 'hostname', 'memory', 'uuid', 'version', 'create_time', 'update_time')
search_fields = ('hostname', 'memory') search_fields = ('hostname', 'memory')
list_filter = ('rid', ) list_filter = ('rid', )
class ConnLog(models.Model): class ConnLog(models.Model):
id = models.IntegerField(verbose_name='ID',primary_key=True) id = models.IntegerField(verbose_name='ID', primary_key=True)
action = models.CharField(verbose_name='Action', max_length=20, null=True) action = models.CharField(verbose_name='Action', max_length=20, null=True)
conn_id = models.CharField(verbose_name='Connection ID', max_length=10, null=True) conn_id = models.CharField(verbose_name='Connection ID', max_length=10, null=True)
from_ip = models.CharField(verbose_name='From IP', max_length=30, null=True) from_ip = models.CharField(verbose_name='From IP', max_length=30, null=True)
@ -100,13 +107,15 @@ class ConnLog(models.Model):
session_id = models.CharField(verbose_name='Session ID', max_length=60, null=True) session_id = models.CharField(verbose_name='Session ID', max_length=60, null=True)
uuid = models.CharField(verbose_name='uuid', max_length=60, null=True) uuid = models.CharField(verbose_name='uuid', max_length=60, null=True)
class ConnLogAdmin(admin.ModelAdmin): class ConnLogAdmin(admin.ModelAdmin):
list_display = ('id', 'action', 'conn_id', 'from_ip', 'from_id', 'rid', 'conn_start', 'conn_end', 'session_id', 'uuid') list_display = ('id', 'action', 'conn_id', 'from_ip', 'from_id', 'rid', 'conn_start', 'conn_end', 'session_id', 'uuid')
search_fields = ('from_ip', 'rid') search_fields = ('from_ip', 'rid')
list_filter = ('id', 'from_ip', 'from_id', 'rid', 'conn_start', 'conn_end') list_filter = ('id', 'from_ip', 'from_id', 'rid', 'conn_start', 'conn_end')
class FileLog(models.Model): class FileLog(models.Model):
id = models.IntegerField(verbose_name='ID',primary_key=True) id = models.IntegerField(verbose_name='ID', primary_key=True)
file = models.CharField(verbose_name='Path', max_length=500) file = models.CharField(verbose_name='Path', max_length=500)
remote_id = models.CharField(verbose_name='Remote ID', max_length=20, default='0') remote_id = models.CharField(verbose_name='Remote ID', max_length=20, default='0')
user_id = models.CharField(verbose_name='User ID', max_length=20, default='0') user_id = models.CharField(verbose_name='User ID', max_length=20, default='0')
@ -115,11 +124,13 @@ class FileLog(models.Model):
direction = models.IntegerField(verbose_name='Direction', default=0) direction = models.IntegerField(verbose_name='Direction', default=0)
logged_at = models.DateTimeField(verbose_name='Logged At', null=True) logged_at = models.DateTimeField(verbose_name='Logged At', null=True)
class FileLogAdmin(admin.ModelAdmin): class FileLogAdmin(admin.ModelAdmin):
list_display = ('id', 'file', 'remote_id', 'user_id', 'user_ip', 'filesize', 'direction', 'logged_at') list_display = ('id', 'file', 'remote_id', 'user_id', 'user_ip', 'filesize', 'direction', 'logged_at')
search_fields = ('file', 'remote_id', 'user_id', 'user_ip') search_fields = ('file', 'remote_id', 'user_id', 'user_ip')
list_filter = ('id', 'file', 'remote_id', 'user_id', 'user_ip', 'filesize', 'direction', 'logged_at') list_filter = ('id', 'file', 'remote_id', 'user_id', 'user_ip', 'filesize', 'direction', 'logged_at')
class ShareLink(models.Model): class ShareLink(models.Model):
''' 分享链接 ''' 分享链接
''' '''
@ -129,16 +140,14 @@ class ShareLink(models.Model):
is_used = models.BooleanField(verbose_name=_('是否使用'), default=False) is_used = models.BooleanField(verbose_name=_('是否使用'), default=False)
is_expired = models.BooleanField(verbose_name=_('是否过期'), default=False) is_expired = models.BooleanField(verbose_name=_('是否过期'), default=False)
create_time = models.DateTimeField(verbose_name=_('生成时间'), auto_now_add=True) create_time = models.DateTimeField(verbose_name=_('生成时间'), auto_now_add=True)
class Meta: class Meta:
ordering = ('-create_time',) ordering = ('-create_time',)
verbose_name = _("分享链接") verbose_name = _("分享链接")
verbose_name_plural = _("链接列表" ) verbose_name_plural = _("链接列表")
class ShareLinkAdmin(admin.ModelAdmin): class ShareLinkAdmin(admin.ModelAdmin):
list_display = ('shash', 'uid', 'peers', 'is_used', 'is_expired', 'create_time') list_display = ('shash', 'uid', 'peers', 'is_used', 'is_expired', 'create_time')
search_fields = ('peers', ) search_fields = ('peers', )
list_filter = ('is_used', 'uid', 'is_expired' ) list_filter = ('is_used', 'uid', 'is_expired')

View File

@ -23,6 +23,7 @@
<th>{{ "系统" | translate }}</th> <th>{{ "系统" | translate }}</th>
<th>{{ "CPU" | translate }}</th> <th>{{ "CPU" | translate }}</th>
<th>{{ "内存" | translate }}</th> <th>{{ "内存" | translate }}</th>
<th>{{ "IP" | translate }}</th>
<th>{{ "注册时间" | translate }}</th> <th>{{ "注册时间" | translate }}</th>
<th>{{ "更新时间" | translate }}</th> <th>{{ "更新时间" | translate }}</th>
<th>{{ "状态" | translate }}</th> <th>{{ "状态" | translate }}</th>
@ -41,6 +42,7 @@
<td>{{one.os}}</td> <td>{{one.os}}</td>
<td>{{one.cpu}}</td> <td>{{one.cpu}}</td>
<td>{{one.memory}}</td> <td>{{one.memory}}</td>
<td>{{one.ip_address}}</td>
<td>{{one.create_time}}</td> <td>{{one.create_time}}</td>
<td>{{one.update_time}}</td> <td>{{one.update_time}}</td>
<td>{{one.status}} </td> <td>{{one.status}} </td>
@ -67,10 +69,10 @@
<button class="layui-btn" ><a href="?page={{ page_obj.paginator.num_pages }}">{{ "尾页" | translate }} &raquo;</a></button> <button class="layui-btn" ><a href="?page={{ page_obj.paginator.num_pages }}">{{ "尾页" | translate }} &raquo;</a></button>
{% endif %} {% endif %}
</span> </span>
</div> </div>
{% endif %} {% endif %}
{% if u.is_admin and show_all %} {% if u.is_admin and show_all %}
<div class="layui-col-md15"> <div class="layui-col-md15">
<div class="layui-card"> <div class="layui-card">
@ -89,6 +91,7 @@
<th>{{ "系统" | translate }}</th> <th>{{ "系统" | translate }}</th>
<th>{{ "CPU" | translate }}</th> <th>{{ "CPU" | translate }}</th>
<th>{{ "内存" | translate }}</th> <th>{{ "内存" | translate }}</th>
<th>{{ "IP" | translate }}</th>
<th>{{ "注册日期" | translate }}</th> <th>{{ "注册日期" | translate }}</th>
<th>{{ "更新时间" | translate }}</th> <th>{{ "更新时间" | translate }}</th>
<th>{{ "状态" | translate }}</th> <th>{{ "状态" | translate }}</th>
@ -106,6 +109,7 @@
<td>{{one.os}} </td> <td>{{one.os}} </td>
<td>{{one.cpu}} </td> <td>{{one.cpu}} </td>
<td>{{one.memory}} </td> <td>{{one.memory}} </td>
<td>{{one.ip_address}}</td>
<td>{{one.create_time}} </td> <td>{{one.create_time}} </td>
<td>{{one.update_time}} </td> <td>{{one.update_time}} </td>
<td>{{one.status}} </td> <td>{{one.status}} </td>
@ -132,11 +136,11 @@
<button class="layui-btn" ><a href="?show_type=admin&page={{ page_obj.paginator.num_pages }}">{{ "尾页" | translate }} &raquo;</a></button> <button class="layui-btn" ><a href="?show_type=admin&page={{ page_obj.paginator.num_pages }}">{{ "尾页" | translate }} &raquo;</a></button>
{% endif %} {% endif %}
</span> </span>
</div> </div>
{% endif %} {% endif %}
</div> </div>
</div> </div>

View File

@ -14,6 +14,15 @@ from .views_front import *
from django.utils.translation import gettext as _ from django.utils.translation import gettext as _
def get_client_ip(request):
x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')
if x_forwarded_for:
ip = x_forwarded_for.split(',')[0]
else:
ip = request.META.get('REMOTE_ADDR')
return ip
def login(request): def login(request):
result = {} result = {}
if request.method == 'GET': if request.method == 'GET':
@ -226,7 +235,7 @@ def sysinfo(request):
if request.method == 'GET': if request.method == 'GET':
result['error'] = _('错误的提交方式!') result['error'] = _('错误的提交方式!')
return JsonResponse(result) return JsonResponse(result)
client_ip = get_client_ip(request)
postdata = json.loads(request.body) postdata = json.loads(request.body)
device = RustDesDevice.objects.filter(Q(rid=postdata['id']) & Q(uuid=postdata['uuid'])).first() device = RustDesDevice.objects.filter(Q(rid=postdata['id']) & Q(uuid=postdata['uuid'])).first()
if not device: if not device:
@ -239,6 +248,7 @@ def sysinfo(request):
username=postdata.get('username', '-'), username=postdata.get('username', '-'),
uuid=postdata['uuid'], uuid=postdata['uuid'],
version=postdata['version'], version=postdata['version'],
ip_address=client_ip
) )
device.save() device.save()
else: else:
@ -254,6 +264,8 @@ def heartbeat(request):
postdata = json.loads(request.body) postdata = json.loads(request.body)
device = RustDesDevice.objects.filter(Q(rid=postdata['id']) & Q(uuid=postdata['uuid'])).first() device = RustDesDevice.objects.filter(Q(rid=postdata['id']) & Q(uuid=postdata['uuid'])).first()
if device: if device:
client_ip = get_client_ip(request)
device.ip_address = client_ip
device.save() device.save()
# token保活 # token保活
create_time = datetime.datetime.now() + datetime.timedelta(seconds=EFFECTIVE_SECONDS) create_time = datetime.datetime.now() + datetime.timedelta(seconds=EFFECTIVE_SECONDS)

View File

@ -229,6 +229,9 @@ def get_all_info():
if device: if device:
devices[peer.rid]['rust_user'] = user.username devices[peer.rid]['rust_user'] = user.username
for rid in devices.keys():
if not devices[rid].get('rust_user', ''):
devices[rid]['rust_user'] = _('未登录')
for k, v in devices.items(): for k, v in devices.items():
devices[k]['status'] = _('在线') if (now - datetime.datetime.strptime(v['update_time'], '%Y-%m-%d %H:%M')).seconds <= 120 else _('离线') devices[k]['status'] = _('在线') if (now - datetime.datetime.strptime(v['update_time'], '%Y-%m-%d %H:%M')).seconds <= 120 else _('离线')
return [v for k, v in devices.items()] return [v for k, v in devices.items()]
@ -295,7 +298,7 @@ def share(request):
sharelinks = ShareLink.objects.filter(Q(uid=request.user.id) & Q(is_used=False) & Q(is_expired=False)) sharelinks = ShareLink.objects.filter(Q(uid=request.user.id) & Q(is_used=False) & Q(is_expired=False))
# 省资源:处理已过期请求,不主动定时任务轮询请求,在任意地方请求时,检查是否过期,过期则保存。 # 省资源:处理已过期请求,不主动定时任务轮询请求,在任意地方请求时,检查是否过期,过期则保存。
now = datetime.datetime.now() # now = datetime.datetime.now()
for sl in sharelinks: for sl in sharelinks:
check_sharelink_expired(sl) check_sharelink_expired(sl)
sharelinks = ShareLink.objects.filter(Q(uid=request.user.id) & Q(is_used=False) & Q(is_expired=False)) sharelinks = ShareLink.objects.filter(Q(uid=request.user.id) & Q(is_used=False) & Q(is_expired=False))
@ -327,7 +330,7 @@ def share(request):
# 自己的peers若重叠需要跳过 # 自己的peers若重叠需要跳过
peers_self_ids = [x.rid for x in RustDeskPeer.objects.filter(Q(uid=request.user.id))] peers_self_ids = [x.rid for x in RustDeskPeer.objects.filter(Q(uid=request.user.id))]
peers_share = RustDeskPeer.objects.filter(Q(rid__in=peers) & Q(uid=sharelink.uid)) peers_share = RustDeskPeer.objects.filter(Q(rid__in=peers) & Q(uid=sharelink.uid))
peers_share_ids = [x.rid for x in peers_share] # peers_share_ids = [x.rid for x in peers_share]
for peer in peers_share: for peer in peers_share:
if peer.rid in peers_self_ids: if peer.rid in peers_self_ids:

Binary file not shown.