首页 > 其他分享 >【halo】博客系统从部署到访问

【halo】博客系统从部署到访问

时间:2024-03-14 16:31:14浏览次数:24  
标签:__ domain record self 博客 content 访问 halo

九术AI

【halo】博客系统从部署到访问

部署

本文采用 docker-compose的方式进行部署,可以有效避免一些莫名其妙的问题

halo 支持多种数据库方式进行部署,这里采用的是PSQL方式进行部署,其他数据库部署方式可以参考文档

docker-compose.yaml

  • halo
    • ports: 端口映射。修改第一个端口值为期望访问的端口
    • image: 镜像名。每次更新版本可以通过修改该参数的值执行重新构建即可无痕升级
    • command
      • spring.r2dbc.password: 数据库密码。需要与下面的halodb中的数据库密码保持一致
      • halo.external-url: 外部访问地址。根据实际域名填写,也可以填写127.0.0.1地址
  • halodb
    • environment
      • POSTGRES_PASSWORD: 数据库密码。可以修改为自定义密码。
version: "3"

services:
  halo:
    image: halohub/halo:2.10.0
    container_name: halo
    restart: on-failure:3
    depends_on:
      halodb:
        condition: service_healthy
    networks:
      halo_network:
    volumes:
      - ./:/root/.halo2
    ports:
      - "8090:8090"
    healthcheck:
      test: ["CMD", "curl", "-f", "http://localhost:8090/actuator/health/readiness"]
      interval: 30s
      timeout: 5s
      retries: 5
      start_period: 30s          
    command:
      - --spring.r2dbc.url=r2dbc:pool:postgresql://halodb/halo
      - --spring.r2dbc.username=halo
      # PostgreSQL 的密码,请保证与下方 POSTGRES_PASSWORD 的变量值一致。
      - --spring.r2dbc.password=password
      - --spring.sql.init.platform=postgresql
      # 外部访问地址,请根据实际需要修改
      - --halo.external-url=https://blog.cnkj.site
      # 初始化的超级管理员用户名
      - --halo.security.initializer.superadminusername=admin
      # 初始化的超级管理员密码
      - --halo.security.initializer.superadminpassword=P@88w0rd
      # 启用缓存
      - --halo.cache.page.disabled=false
  halodb:
    image: postgres:latest
    container_name: halodb
    restart: on-failure:3
    networks:
      halo_network:
    volumes:
      - ./db:/var/lib/postgresql/data
    ports:
      - "15432:5432"
    healthcheck:
      test: [ "CMD", "pg_isready" ]
      interval: 10s
      timeout: 5s
      retries: 5
    environment:
      - POSTGRES_PASSWORD=password
      - POSTGRES_USER=halo
      - POSTGRES_DB=halo
      - PGUSER=halo

networks:
  halo_network:
    external: false

部署完成后就可以通过 http://127.0.0.1:8090 进行站点访问了

外部访问

这里主要讲解如果通过域名进行公网访问站点

首先你需要一个自己的域名

其次需要打电话给宽带运营商申请动态IP(专线固定IP不用申请)

解析动态IP到域名上

这里采用python脚本方式解析动态IP到域名上,以便于后续其他各个服务可以使用该域名作为基础域名跳转,而不需要每次都修改相关IP地址

详细部署流程可参考自动获取动态IPV4并解析到阿里云域名

  1. SSH登录服务器后台或相关可执行计划任务的服务
  2. 添加脚本到服务器上并设置周期执行时间
  3. 修改脚本中__execute方法块中域名相关变量

参考脚本内容

#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""
    @Time    : 2023/08/23
    @Author  : LXW
    @Site    : 
    @File    : ipv4.py
    @Software: PyCharm
    @Description:
        - 版本
        -- python==3.8
        -- alibabacloud_alidns20150109==3.0.7
        -- aliyun-python-sdk-domain==3.14.9
        - 依赖安装
        -- pip3 install alibabacloud_alidns20150109==3.0.7
        -- pip3 install aliyun-python-sdk-domain==3.14.9
"""

import json

import requests
from alibabacloud_alidns20150109 import models as alidns_20150109_models
from alibabacloud_alidns20150109.client import Client as Alidns20150109Client
from alibabacloud_tea_openapi import models as open_api_models
from alibabacloud_tea_util import models as util_models
from alibabacloud_tea_util.client import Client as UtilClient


class RemoteIp:
    def __init__(self, url, username, passwd):
        self.url = url
        self.username = username
        self.passwd = passwd

    def __get_token(self):
        _url = self.url + "/emby/Users/authenticatebyname?X-Emby-Client=Emby%20Web&X-Emby-Device-Name=Chrome%20macOS&X-Emby-Device-Id=fa6b7733-23fc-4cb7-bae3-d7f0536b485e&X-Emby-Client-Version=4.7.13.0&X-Emby-Language=zh-cn"
        content = requests.post(_url, False, {"Username": self.username, "Pw": self.passwd}).text
        return json.loads(content)["AccessToken"]

    def get_remote_addr(self):
        token = self.__get_token()
        full_url = self.url + "/emby/System/Info?X-Emby-Token=" + token
        content = requests.get(full_url, False).text
        addr = list(json.loads(content)["RemoteAddresses"])
        remote_add = str(addr[0]).split(":")[1].replace("/", "")
        return remote_add

def request_params(request, value):
    request.value = value

class AliyunOperation:
    def __init__(self, access_key_id, access_secret):
        """
        初始化
        :param access_key_id: 阿里云AccessKey的ID
        :param access_secret: 阿里云AccessKey的密码
        """
        config = open_api_models.Config(
            # 必填,您的 AccessKey ID,
            access_key_id=access_key_id,
            # 必填,您的 AccessKey Secret,
            access_key_secret=access_secret
        )
        # Endpoint 请参考 https://api.aliyun.com/product/Alidns
        config.endpoint = f'alidns.cn-hangzhou.aliyuncs.com'
        self.client = Alidns20150109Client(config)

    def describe_subdomain_records(self, subdomain, domain_name):
        """
        子域名解析记录查询 https://api.aliyun.com/api/Alidns/2015-01-09/DescribeSubDomainRecords
        这个函数会返回一大堆东西,但是目前我们需要使用的就两个,一个是解析记录的数量,一个就是RecordId,RecordId修改域名解析时需要用到
        :param domain_name:
        :param subdomain:
        :return:
        """
        describe_sub_domain_records_request = alidns_20150109_models.DescribeSubDomainRecordsRequest(
            sub_domain=subdomain,
            domain_name=domain_name
        )
        runtime = util_models.RuntimeOptions()
        try:
            response = self.client.describe_sub_domain_records_with_options(describe_sub_domain_records_request,
                                                                            runtime)
            if 200 == response.status_code:
                return response.body
        except Exception as error:
            # 如有需要,请打印 error
            UtilClient.assert_as_string(error.message)

    def add_record(self, domain_name, rr, domain_type, value):
        """
        新增域名解析记录 https://api.aliyun.com/api/Alidns/2015-01-09/AddDomainRecord
        如果域名解析记录不存在时,就需要新增域名解析记录,新增域名解析记录比较简单,直接填写参数即可
        :param domain_name: 域名名称 baidu.com
        :param rr: 主机记录。如果要解析@.exmaple.com,主机记录要填写”@”,而不是空。
        :param domain_type: 解析记录类型。https://help.aliyun.com/document_detail/29805.html?spm=api-workbench.api_explorer.0.0.6b1a134bcODqcI
        :param value: 记录值。String
        :return:
        """
        add_domain_record_request = alidns_20150109_models.AddDomainRecordRequest(
            domain_name=domain_name,
            rr=rr,
            type=domain_type,
            value=value
        )
        runtime = util_models.RuntimeOptions()
        try:
            response = self.client.add_domain_record_with_options(add_domain_record_request, runtime)
            if 200 == response.status_code:
                return response.body
        except Exception as error:
            # 如有需要,请打印 error
            UtilClient.assert_as_string(error.message)

    def update_record(self, domain_type, value, rr, record_id):
        """
        更新域名解析记录
        如果域名解析记录已存在,则不能使用新增,而是更新域名解析记录。更新的时候需要用到RecordId,这个一般查询的时候就会有返回,直接使用即可
        :param domain_type: 解析记录类型。https://help.aliyun.com/document_detail/29805.html?spm=api-workbench.api_explorer.0.0.6b1a134bcODqcI
        :param value: 记录值。String
        :param rr: 主机记录。如果要解析@.exmaple.com,主机记录要填写”@”,而不是空。
        :param record_id: 解析记录的ID
        :return:
        """
        update_domain_record_request = alidns_20150109_models.UpdateDomainRecordRequest(
            record_id=record_id,
            rr=rr,
            type=domain_type,
            value=value
        )
        runtime = util_models.RuntimeOptions()
        try:
            # 复制代码运行请自行打印 API 的返回值
            self.client.update_domain_record_with_options(update_domain_record_request, runtime)
        except Exception as error:
            # 如有需要,请打印 error
            UtilClient.assert_as_string(error.message)

def __execute():
    # 获取动态 IPV4 地址(如果是IPV6也可以用相同方式获取)
    __remote_addr = __get_from_urls()
    if "None" == __remote_addr:
        __remote_addr = RemoteIp(url=__url, username=__username, passwd=__passwd).get_remote_addr()
    print("current ipv4: " + __remote_addr)
    __aliyun_client = AliyunOperation(access_key_id=__accessKeyId, access_secret=__accessSecret)
    # 查询 example.com 主机下 test.example.com 域名的解析列表
    content = __aliyun_client.describe_subdomain_records("redirect.your.domain", "your.domain")
    print("current desc domain content: " + str(content))
    if 0 == content.total_count:
        # 新增一条 test.example.com 的IPV4域名解析记录(如果是IPV6地址,修改参数 domain_type 为 AAAA)
        __aliyun_client.add_record("your.domain", "redirect", "A", __remote_addr)
    elif 1 == content.total_count:
        # 如果 test.example.com 域名解析已经存在,则进行更新操作
        source = content.domain_records.record[0]
        record_id = source.record_id
        if __remote_addr != source.value:
            __aliyun_client.update_record("A", __remote_addr, "redirect", record_id)
    # else:
    #     # 不推荐使用该方式进行域名解析更新,容易错误更新
    #     sources = content.domain_records.record
    #     for source in sources:
    #         # 因为是更新IP,所以只考虑 A 或者 AAAA
    #         domain_type = source.type
    #         if "A" == domain_type or "AAAA" == domain_type:
    #             record_id = source.record_id
    #             rr = source.rr
    #             __aliyun_client.update_record(domain_type, __remote_addr, rr, record_id)

def __get_from_urls():
    """
    通过url获取本地IP地址
        "https://myip4.ipip.net",
        "https://ddns.oray.com/checkip",
        "https://ip.3322.net",
        "https://4.ipw.cn"
    :return: ip
    """
    ipw = "https://4.ipw.cn"
    response = requests.get(ipw)
    if 200 == response.status_code:
        content = response.text
        return content.strip()
    ip = "https://ip.3322.net"
    response = requests.get(ip)
    if 200 == response.status_code:
        content = response.text
        return content.strip()
    myip4 = "https://myip4.ipip.net"
    response = requests.get(myip4)
    if 200 == response.status_code:
        content = response.text.split(":")[1]
        content = content.replace("来自于", "")
        return content.strip()
    ddns = "https://ddns.oray.com/checkip"
    response = requests.get(ddns)
    if 200 == response.status_code:
        content = response.text.split(":")[1]
        return content.strip()
    return "None"

if __name__ == '__main__':
    # emby相关参数,可以不填
    __url = "http://127.0.0.1:8096"
    __username = "emby_user"
    __passwd = "emby_passwd"
    # 域名解析权限,必填
    __accessKeyId = "__accessKeyId"
    __accessSecret = "__accessSecret"
    __execute()

设置公网访问

这里以阿里云CDN方式进行公网访问举例,利用了CDN的缓存、去除端口访问的限制、https证书访问配置等优点

假设在上文中解析的动态IP地址到的域名为 redirect.your.domain,那么站点的访问地址应该为 http://redirect.your.domain:8090

  1. 登录并通过官网购买CDN流量包
  2. 进入CDN控制台
  3. 在下面的域名管理添加域名
  4. 填写你要访问的域名,例如访问 http://blog.your.domain ,并根据出现的验证提示选择其中方式进行验证
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
  5. 在下面的源站信息中点击新增源站信息
  6. 新增源站信息页面选择源站信息方式为源站域名
  7. 填写相关源站信息,其中域名为上文中我们通过动态IP解析到的域名(redirect.your.domain),端口为我们的站点HTTP访问端口(8090)
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
  8. 确认完成后等待完成验证。在左侧域名管理中就可以看到新增的域名已经完成状态了
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
  9. 点击域名右侧操作中的管理对域名进行配置,大部分配置使用默认参数即可,以下仅说明几个重要配置

CDN配置管理

  • IPv6开关: 控制客户端是IPV6地址可以正常访问站点
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
  • 回源配置: 控制站点实际回源信息。因为我们采用的域名回源,所以需要设置最下方的301/302限制,否则实际站点返回可能出现端口
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
  • 缓存配置: 用于配置各个节点的缓存和有效期,通过该配置可以有效提高高频访问数据的访问效率
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
  • HTTPS配置: 该配置可以使我们的站点访问始终请求https证书地址
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
  • 性能优化: 性能优化设置
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
  • 流量限制: 限制峰值流量大小,非常重要。合理的设置可以有效避免你的CDN流量被恶意消耗导致产生高额费用
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

标签:__,domain,record,self,博客,content,访问,halo
From: https://blog.csdn.net/carolcoral/article/details/136615131

相关文章

  • 如何在Nginx服务器上配置访问静态文件目录并提供文件下载功能
    引言在搭建网站的过程中,我们经常需要让访客通过URL直接访问或下载存储在服务器特定目录下的静态文件。本文将详细介绍如何在Nginx服务器环境中配置一个名为"download"的文件目录,以便用户能够通过浏览器访问并下载其中的手册和其他文档。1.目录结构与权限设置首先,请确保您的Ngi......
  • 【计算机网络】数据链路层——动态划分信道之轮询访问介质控制
    之前的信道划分の介质访问控制信道划分介质访问控制(MAC,MultipleAccessControl)协议:基于多路复用技术划分资源。网络负载重:共享信道效率高,且公平网络负载轻:共享信道效率低随机访问MAC协议:用户根据意愿随机发送信息,发送信息时可独占信道带宽。网络负载重:产生冲突开销网络......
  • Windows Server 各版本搭建终端服务器实现远程访问(03~19)
    一、WindowsServer2003左下角开始➡管理工具➡管理您的服务器,点击添加或删除角色点击下一步 勾选自定义,点击下一步蒂埃涅吉终端服务器,点击下一步 点击确定重新登录后点击确定点击开始➡管理工具➡计算机管理,展开本地用户和组,点击组可以发现有个组关门用来远程......
  • 我的博客(计划篇)
    2024年实习加毕设,忙得不可开交,有空余时间的话会来折腾博客。美化借用cute-cnblogs样式,先有个漂亮的界面(当然好多东西都没改,后面有时间慢慢改)有时间的话,在此基础上自定义修改一下,顺便学点前端知识。这里留下几个计划要修改的地方::鼠标样式,点击文字效果或者爆炸效果:音乐......
  • 01-列表操作-使用slice()命名切片,增强程序可读及可维护性,兼使用indices()方法,防止出现
    程序中的切片,使用原始的索引访问时,如果数量过多,时间久了,就会导致难以阅读和维护。但使用slice()函数,创建【命名切片】后,赋予了切片与现实相近的名称,让程序更容易理解。同时,slice类中的indices方法,返回start,stop,step,3个值组成的元组。并且indices()对3个值进行自动调整,确......
  • 【3月13日-云服务器推荐】京东云新上1年福利机型 阿里云活动页大改 选购指南 最新价格
     本文纯原创,侵权必究 《最新对比表》已更新在文章头部—腾讯云文档,文章具有时效性,请以腾讯文档为准!【腾讯文档实时更新】云服务器1分钟教会你如何选择教程https://docs.qq.com/document/DV0RCS0lGeHdMTFFV?tab=000003视频解读最新活动:【云服务器推荐】价格对比!阿里云......
  • 在Linux中,访问一个站点5xx了,如何定位问题?
    当访问一个托管在Linux服务器上的站点时,如果收到5xx系列的HTTP状态码(如500InternalServerError、502BadGateway、503ServiceUnavailable等),这意味着服务器在处理请求时遇到了问题。定位这类问题通常需要遵循以下步骤:查看服务器错误日志对于Web服务器,如Apache或Nginx,首......
  • Python爬虫实战系列1:博客园cnblogs热门新闻采集
    实战案例:博客园热门新闻采集一、分析页面打开博客园网址https://www.cnblogs.com/,点击【新闻】再点击【本周】本次采集,我们以页面新闻标题为案例来采集。这里可以看到标题“李彦宏:以后不会存在“程序员”这种职业了”。1.1、分析请求F12打开开发者模式,然后点击Network后点......
  • zabbix直接ip访问web前端
    1、修改配置文件, sudovim/etc/apache2/sites-available/000-default.conf把之前的DocumentRoot/var/www/html这行注释掉,另起一行输入 DocumentRoot/usr/share/zabbix 2、重启Apache、zabbix-server服务sudosystemctlrestartapache2.servicezabbix-server.serv......
  • 利用Nginx正向代理实现局域网电脑访问外网
    引言在网络环境中,有时候我们需要让局域网内的电脑访问外网,但是由于网络策略或其他原因,直接访问外网是不可行的。这时候,可以借助Nginx来搭建一个正向代理服务器,实现局域网内电脑通过Nginx转发访问外网的需求。在工作中我遇到了一个类似的情况:在公司网络中,由于管理要求,局域网......