EUserv免费的德国VPS近期更改了续期步骤,无法自动续期,Github Action续期可能导致Github账号被封。EUserv的免费德鸡性能疲弱,经常满负载,甚至掉线,但好在用WARP+和Cloudflare Spectrum加持下看个4K油管还是可以的,所以有必要给这个只有IPv6的免费VPS续期。

由于本来就是玩具鸡,难道要开台付费的VPS给它自动续期? 手动续期也可行,只是怕忘记了。 Cloudflare Work和腾讯云函数可以给免费给这德鸡自动续期,

本文详细介绍如何用腾讯云函数自动续期。没有接触腾讯云函数SCF的同学也不要担心困难,按下面步骤来,小白亦可搞定。



使用腾讯云函数 SCF自动续期EUserv免费IPv6 VPS脚本见 https://github.com/o0oo0ooo0/EUserv_extend

目的:自动获取账号内所有的VPS项目,并检测是否需要续期,需要续期会自动续期。


1. 打开腾讯云函数SFC

打开 https://console.cloud.tencent.com/scf/  登录腾讯云,初次使用云函数SCF要授权。

2. 新建BeautifulSoup层

新建层 BeautifulSoup ,将 BeautifulSoup.zip 导入 ,添加运行环境 Python 3.6 。注意层的位置选择香港等海外地区,因为我们要续费的VPS在德国。

建好层后如下,后面用的上。

3. 新建腾讯云函数EUserv_extend

运行环境选择 Python 3.6,创建方式选择 空白函数,内存选择 64M,执行超时时间建议为 300 s(网站在国外访问比较慢,建议部署在非大陆区域,例如HK、SG等,需要和上面创建层的位置是一样的。),将修改后的 main.py 粘贴进去。

函数服务–新建–自定义创建,按下图配置

复制https://github.com/o0oo0ooo0/EUserv_extend库中的main.py文件内容。

如下:


# -*- coding: utf8 -*-
import re
import json
import time
import requests
from bs4 import BeautifulSoup

# 强烈建议部署在非大陆区域,例如HK、SG等
# 常量命名使用全部大写的方式,可以使用下划线。
USERNAME = ''  # 这里填用户名,邮箱也可
PASSWORD = ''  # 这里填密码

# Server酱 http://sc.ftqq.com/?c=code
SCKEY = ''  # 这里填Server酱的key,无需推送可不填 示例: SCU646xxxxxxxxdacd6a5dc3f6

# 酷推 https://cp.xuthus.cc
COOL_PUSH_SKEY = ''
# 通知类型 CoolPush_MODE的可选项有(默认send):send[QQ私聊]、group[QQ群聊]、wx[个微]、ww[企微]
COOL_PUSH_MODE = 'send'

# PushPlus https://pushplus.hxtrip.com/message
PUSH_PLUS_TOKEN = ''

# Telegram Bot Push https://core.telegram.org/bots/api#authorizing-your-bot
TG_BOT_TOKEN = ''  # 通过 @BotFather 申请获得,示例:1077xxx4424:AAFjv0FcqxxxxxxgEMGfi22B4yh15R5uw
TG_USER_ID = ''  # 用户、群组或频道 ID,示例:129xxx206
TG_API_HOST = 'api.telegram.org'  # 自建 API 反代地址,供网络环境无法访问时使用,网络正常则保持默认

# wecomchan https://github.com/easychen/wecomchan
WECOMCHAN_DOMAIN = ''  # http(s)://example.com/
WECOMCHAN_SEND_KEY = ''
WECOMCHAN_TO_USER = '@all'  # 默认全部推送, 对个别人推送可用 User1|User2
# 变量命名使用全部小写的方式,可以使用下划线。
desp = ''  # 不用动


# 函数命名使用全部小写的方式,可以使用下划线。
def print_(info):
    print(info)
    global desp
    desp = desp + info + '\n\n'


def login(username, password) -> (str, requests.session):
    headers = {
        "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) "
                      "Chrome/83.0.4103.116 Safari/537.36",
        "origin": "https://www.euserv.com",
    }
    url = "https://support.euserv.com/index.iphp"
    session = requests.Session()

    sess = session.get(url, headers=headers)
    sess_id = re.findall("PHPSESSID=(\\w{10,100});", str(sess.headers))[0]
    # 访问png
    png_url = "https://support.euserv.com/pic/logo_small.png"
    session.get(png_url, headers=headers)

    login_data = {
        "email": username,
        "password": password,
        "form_selected_language": "en",
        "Submit": "Login",
        "subaction": "login",
        "sess_id": sess_id
    }
    f = session.post(url, headers=headers, data=login_data)
    f.raise_for_status()

    if f.text.find('Hello') == -1:
        return '-1', session
    return sess_id, session


def get_servers(sess_id: str, session: requests.session) -> {}:
    d = {}
    url = "https://support.euserv.com/index.iphp?sess_id=" + sess_id
    headers = {
        "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) "
                      "Chrome/83.0.4103.116 Safari/537.36",
        "origin": "https://www.euserv.com"
    }
    f = session.get(url=url, headers=headers)
    f.raise_for_status()
    soup = BeautifulSoup(f.text, 'html.parser')
    for tr in soup.select('#kc2_order_customer_orders_tab_content_1 .kc2_order_table.kc2_content_table tr'):
        server_id = tr.select('.td-z1-sp1-kc')
        if not len(server_id) == 1:
            continue
        flag = True if tr.select('.td-z1-sp2-kc .kc2_order_action_container')[
                           0].get_text().find('Contract extension possible from') == -1 else False
        d[server_id[0].get_text()] = flag
    return d


def renew(sess_id: str, session: requests.session, password: str, order_id: str) -> bool:
    url = "https://support.euserv.com/index.iphp"
    headers = {
        "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) "
                      "Chrome/83.0.4103.116 Safari/537.36",
        "Host": "support.euserv.com",
        "origin": "https://support.euserv.com",
        "Referer": "https://support.euserv.com/index.iphp"
    }
    data = {
        "Submit": "Extend contract",
        "sess_id": sess_id,
        "ord_no": order_id,
        "subaction": "choose_order",
        "choose_order_subaction": "show_contract_details"
    }
    session.post(url, headers=headers, data=data)
    data = {
        "sess_id": sess_id,
        "subaction": "kc2_security_password_get_token",
        "prefix": "kc2_customer_contract_details_extend_contract_",
        "password": password
    }
    f = session.post(url, headers=headers, data=data)
    f.raise_for_status()
    if not json.loads(f.text)["rs"] == "success":
        return False
    token = json.loads(f.text)["token"]["value"]
    data = {
        "sess_id": sess_id,
        "ord_id": order_id,
        "subaction": "kc2_customer_contract_details_extend_contract_term",
        "token": token
    }
    session.post(url, headers=headers, data=data)
    time.sleep(5)
    return True


def check(sess_id: str, session: requests.session):
    print("Checking.......")
    d = get_servers(sess_id, session)
    flag = True
    for key, val in d.items():
        if val:
            flag = False
            print_("ServerID: %s Renew Failed!" % key)
    if flag:
        print_("ALL Work Done! Enjoy")


# Server酱 http://sc.ftqq.com/?c=code
def server_chan():
    data = (
        ('text', 'EUserv续费日志'),
        ('desp', desp)
    )
    response = requests.post('https://sc.ftqq.com/' + SCKEY + '.send', data=data)
    if response.status_code != 200:
        print('Server酱 推送失败')
    else:
        print('Server酱 推送成功')


# 酷推 https://cp.xuthus.cc/
def coolpush():
    c = 'EUserv续费日志\n\n' + desp
    data = json.dumps({'c': c})
    url = 'https://push.xuthus.cc/' + COOL_PUSH_MODE + '/' + COOL_PUSH_SKEY
    response = requests.post(url, data=data)
    if response.status_code != 200:
        print('酷推 推送失败')
    else:
        print('酷推 推送成功')


# PushPlus https://pushplus.hxtrip.com/message
def push_plus():
    data = (
        ('token', PUSH_PLUS_TOKEN),
        ('title', 'EUserv续费日志'),
        ('content', desp)
    )
    url = 'https://pushplus.hxtrip.com/send'
    response = requests.post(url, data=data)
    if response.status_code != 200:
        print('PushPlus 推送失败')
    else:
        print('PushPlus 推送成功')


# Telegram Bot Push https://core.telegram.org/bots/api#authorizing-your-bot
def telegram():
    data = (
        ('chat_id', TG_USER_ID),
        ('text', 'EUserv续费日志\n\n' + desp)
    )
    response = requests.post('https://' + TG_API_HOST + '/bot' + TG_BOT_TOKEN + '/sendMessage', data=data)
    if response.status_code != 200:
        print('Telegram Bot 推送失败')
    else:
        print('Telegram Bot 推送成功')


# wecomchan https://github.com/easychen/wecomchan
def wecomchan():
    response = requests.get(WECOMCHAN_DOMAIN + 'wecomchan?sendkey=' + WECOMCHAN_SEND_KEY + '&msg_type=text' + '&to_user=' +
                            WECOMCHAN_TO_USER + '&msg=' + 'EUserv续费日志\n\n' + desp)
    if response.status_code != 200:
        print('wecomchan 推送失败')
    else:
        print('wecomchan 推送成功')


def main_handler(event, context):
    if not USERNAME or not PASSWORD:
        print_("你没有添加任何账户")
        exit(1)
    user_list = USERNAME.strip().split()
    passwd_list = PASSWORD.strip().split()
    if len(user_list) != len(passwd_list):
        print_("The number of usernames and passwords do not match!")
        exit(1)
    for i in range(len(user_list)):
        print('*' * 30)
        print_("正在续费第 %d 个账号" % (i + 1))
        sessid, s = login(user_list[i], passwd_list[i])
        if sessid == '-1':
            print_("第 %d 个账号登陆失败,请检查登录信息" % (i + 1))
            continue
        servers = get_servers(sessid, s)
        print_("检测到第 {} 个账号有 {} 台VPS,正在尝试续期".format(i + 1, len(servers)))
        for k, v in servers.items():
            if v:
                if not renew(sessid, s, passwd_list[i], k):
                    print_("ServerID: %s Renew Error!" % k)
                else:
                    print_("ServerID: %s has been successfully renewed!" % k)
            else:
                print_("ServerID: %s does not need to be renewed" % k)
        time.sleep(15)
        check(sessid, s)
        time.sleep(5)

    # 五个通知渠道至少选取一个
    SCKEY and server_chan()
    COOL_PUSH_MODE and COOL_PUSH_SKEY and coolpush()
    PUSH_PLUS_TOKEN and push_plus()
    TG_BOT_TOKEN and TG_USER_ID and TG_API_HOST and telegram()
    WECOMCHAN_DOMAIN and WECOMCHAN_SEND_KEY and WECOMCHAN_TO_USER and wecomchan()

    print('*' * 30)


if __name__ == '__main__':  # 方便我本地调试
    main_handler(None, None)

然后粘贴到刚才打开的腾讯云函数服务编辑框中,修改其中的用户名,密码。

USERNAME: 你的EUserv账户邮箱或Customer ID

USERNAME = ‘[email protected]
USERNAME = ‘[email protected] [email protected]’ # 多个账号写法

PASSWORD: 账户的密码

PASSWORD = ‘password’
PASSWORD = ‘password1 password2’ # 多个账号写法

设置通知提醒(可选操作)

我已Server酱的提醒为例,打开https://sct.ftqq.com后微信扫码获取sentkey, 然后填写在上面函数代码Server酱的key处。如果您不需要通知也可以不设置提醒。这步不用设置。

环境配置

设置64M内存就够用了,300秒延时。

其他默认就可,然后点完成。完成云函数配置。

4. 绑定层 BeautifulSoup

在 EUserv_extend⇨函数管理⇨层管理 里绑定层 BeautifulSoup。

5. 测试并创建时间触发

点击下图的测试,或会提示config等不存在,可以忽略。

测试没错误,就点EUserv_extend⇨触发管理⇨创建触发器触发周期⇨自定义触发周期。填入0 0 8 */7 * * *

每 7 天的 8 点执行,可修改成你想要的时间。

6. 查看运行日志

日志提示有2台VPS,说明以上成功设置了腾讯云函数SCF续期脚本。