ASCWH https://www.ascwh.com/ zh-CN Mon, 22 Aug 2022 16:16:18 +0800 Mon, 22 Aug 2022 16:16:18 +0800 记录下 go 基础 字符串 与 int 、int64 互转 https://ascwh.com/403.html https://ascwh.com/403.html Mon, 22 Aug 2022 16:16:18 +0800 ASCWH package main import ( "fmt" "reflect" "strconv" ) func main() { var testStr string = "123456" // 字符串 转 int intNum, _ := strconv.Atoi(testStr) fmt.Println("intNum =", intNum, ", intNum 类型是", reflect.TypeOf(intNum)) // 字符串 转 int64 int64Num, _ := strconv.ParseInt(testStr, 10, 64) fmt.Println("int64Num=", int64Num, ", int64Num 类型是", reflect.TypeOf(int64Num)) // =================================================== // int 转 字符串 var intNum2 int = 123456 var str1 string = strconv.Itoa(intNum2) fmt.Println("str1 =", str1, ", str1 类型是", reflect.TypeOf(str1)) // int64 转 字符串 var int64Num2 int64 = 123456 var str2 string = strconv.FormatInt(int64Num2, 10) fmt.Println("str2 =", str2, ", str2 类型是", reflect.TypeOf(str2)) fmt.Println("----------------- 类型转换 ---------------------") var num3 int = 30 var num4 int64 num4 = int64(num3) fmt.Printf("num3 的类型是 %T ,num3= %v \n", num3, num3) fmt.Printf("num4 的类型是 %T ,num4= %v \n", num4, num4) var num5 int64 = 20 var num6 int num6 = int(num5) fmt.Printf("num5 的类型是 %T ,num5= %v \n", num5, num5) fmt.Printf("num6 的类型是 %T ,num6= %v \n", num6, num6) }

运行结果:

intNum = 123456 , intNum 类型是 int
int64Num= 123456 , int64Num 类型是 int64
str1 = 123456 , str1 类型是 string      
str2 = 123456 , str2 类型是 string
----------------- 类型转换 ---------------------        
num3 的类型是 int ,num3= 30                             
num4 的类型是 int64 ,num4= 30                           
num5 的类型是 int64 ,num5= 20                           
num6 的类型是 int ,num6= 20  

字符串 与 int 互转

// 字符串 转 int
intNum, _ := strconv.Atoi(testStr)

// int 转 字符串
var str1 string = strconv.Itoa(intNum2)

字符串 与 int64 互转

// 字符串 转 int64
// 参数1:数字类型的字符串
// 参数2:数字字符串的进制, 比如 2进制、8进制、10进制、16进制
// 参数3:bitSize,意思是string转int64时的数值范围的限制,如int8 int16 int32 int64 不同数字类型取值范围是不一样的
int64Num, _ := strconv.ParseInt(testStr, 10, 64)


// int64 转 字符串
// 参数1:int64类型的数字,如果是int、int8 int16 int32,则要使用int64(数字)转换
// 参数2:表示int64数字要转换什么进制的字符串 , 比如 2进制、8进制、10进制、16进制。取值必须在2到36之间。
var str2 string = strconv.FormatInt(int64Num2, 10) 

示例:
字符串 e 属于16进制,是10进制中的14 。

var testStr3 string = "e"
int64Num, _ := strconv.ParseInt(testStr3, 16, 8) 
fmt.Println("int64Num=", int64Num, ", int64Num 类型是", reflect.TypeOf(int64Num))

运行结果:

int64Num= 14 , int64Num 类型是 int64  

int 与 int64 互转最简单
var intNum int = int(int64位数字)
var int64Num int64 = int64(int数字)

]]>
0 https://ascwh.com/403.html#comments https://ascwh.com/feed/
GoLang实时热加载之Air实现 https://ascwh.com/401.html https://ascwh.com/401.html Fri, 29 Jul 2022 22:40:00 +0800 ASCWH 1.Air介绍
怎样才能在基于gin框架开发时实现实时加载功能呢?像这种烦恼肯定不会只是你一个人的烦恼,所以我报着肯定有现成轮子的心态开始了全网大搜索。果不其然就在Github上找到了一个工具:Air。它支持以下特性:

彩色日志输出
自定义构建或二进制命令
支持忽略子目录
启动后支持监听新目录
更好的构建过程

安装Air
​ 在这里就介绍大众使用的两个版本MacOS和Windows;(建议使用 第一种)

1.这里有一个适用所有的方法:

//1.16可以同时使用这两种方法
go get -u github.com/cosmtrek/air                        //1.16及其以前
go install github.com/cosmtrek/air@latest       //1.16及其以后

注意:在这里大家要注意一下,go从1.16之后不再建议使用go get来获取依赖,但是1.16还是阔以使用go get来获取,1.17之后使用go install来获取相关的依赖

如果出现权限限制使用。sudo+go get -u github.com/cosmtrek/air(1.16版本之前)来强制获取,

2.MacOS

curl -fLo air https://git.io/darwin_air

3.Windows

curl -fLo air.exe https://git.io/windows_air

.使用Air:
为了敲命令时更简单更方便,你应该把alias air='~/.air'(有问题)加到你的.bashrc或.zshrc中。 //这里有个天坑。在后面的问题4种我会详细讲解
首先进入你的项目目录:

cd /path/to/your_project //或者在相关项目下直接操作
最简单的用法就是直接执行下面的命令:

# 首先在当前目录下查找 `.air.conf`配置文件,如果找不到就使用默认的
air -c .air.conf     //这里执行的是自己创建的.air.conf文件,.air,conf中配置运行过程中的相关操作,下面有讲

正常操作流程

#1.通过终端进入相关的项目
#2.直接在编辑器中进入这个项目或者直接在这个项目下
# 1. 在当前目录创建一个新的配置文件.air.conf
touch .air.conf     //也阔以手动在项目根目录下创建

# 2. 复制 `air.conf.example` 中的内容到这个文件,然后根据你的需要去修改它(下面有)

# 3. 使用你的配置运行 air, 如果文件名是 `.air.conf`,只需要执行 `air`。
air    

air_example.conf示例(重要)

# [Air](https://github.com/cosmtrek/air) TOML 格式的配置文件

# 工作目录
# 使用 . 或绝对路径,请注意 `tmp_dir` 目录必须在 `root` 目录下
root = "."
tmp_dir = "tmp"

[build]
# 只需要写你平常编译使用的shell命令。你也可以使用 `make`
# Windows平台示例: cmd = "go build -o tmp\main.exe ."
cmd = "go build -o ./tmp/main ."
# 由`cmd`命令得到的二进制文件名
# Windows平台示例:bin = "tmp\main.exe"
bin = "tmp/main"
# 自定义执行程序的命令,可以添加额外的编译标识例如添加 GIN_MODE=release
# Windows平台示例:full_bin = "tmp\main.exe"
full_bin = "APP_ENV=dev APP_USER=air ./tmp/main"
# 监听以下文件扩展名的文件.
include_ext = ["go", "tpl", "tmpl", "html"]
# 忽略这些文件扩展名或目录
exclude_dir = ["assets", "tmp", "vendor", "frontend/node_modules"]
# 监听以下指定目录的文件
include_dir = []
# 排除以下文件
exclude_file = []
# 如果文件更改过于频繁,则没有必要在每次更改时都触发构建。可以设置触发构建的延迟时间
delay = 1000 # ms
# 发生构建错误时,停止运行旧的二进制文件。
stop_on_error = true
# air的日志文件名,该日志文件放置在你的`tmp_dir`中
log = "air_errors.log"

[log]
# 显示日志时间
time = true

[color]
# 自定义每个部分显示的颜色。如果找不到颜色,使用原始的应用程序日志。
main = "magenta"
watcher = "cyan"
build = "yellow"
runner = "green"

[misc]
# 退出时删除tmp目录
clean_on_exit = true
]]>
1 https://ascwh.com/401.html#comments https://ascwh.com/feed/
supervisor unix:///tmp/supervisor.sock no such file 问题 https://ascwh.com/400.html https://ascwh.com/400.html Thu, 21 Jul 2022 15:29:34 +0800 ASCWH 1、打开配置文件

vim /etc/supervisord.conf
这里把所有的 /tmp 路径改掉

/tmp/supervisor.sock 改成 /var/run/supervisor.sock
/tmp/supervisord.log 改成/var/log/supervisor.log
/tmp/supervisord.pid 改成 /var/run/supervisor.pid
要不容易被 linux 自动清掉

2、修改权限

sudo chmod 777 /run
sudo chmod 777 /var/log
如果没改,启动报错 IOError: [Errno 13] Permission denied: '/var/log/supervisord.log'

3、创建 supervisor.sock

sudo touch /var/run/supervisor.sock
sudo chmod 777 /var/run/supervisor.sock

4、启动supervisord,启动之前的实例或杀死进程

]]>
0 https://ascwh.com/400.html#comments https://ascwh.com/feed/
CentOS7 安装 、配置 Supervisor 基础操作 https://ascwh.com/399.html https://ascwh.com/399.html Sat, 21 May 2022 00:50:00 +0800 ASCWH Supervisor是什么

Supervisor是比较常用的进程管理工具,支持 Linux/MacOS平台,可以用来控制一组Linux/Unix进程(启动、重启、kill)等,使用Supervisor管理的进程,可以做到以守护进程的方式运行,服务异常关闭后可以自动重启。

Supervisor是一个 Client/Server模式的系统,允许用户在类unix操作系统上监视和控制多个进程,或者可以说是多个程序。

Supervisor 有下面几个组件:

supervisord: Supervisor的服务端程序,使用前,需要先启动该组件;
supervisorctl: Supervisor的客户端程序,用来实际控制子进程(自定义的服务、程序);
Supervisor 通过配置文件,还可以启动Web控制台,通过Web页面来管理子进程;

Supervisor 与系统自带 init 进程管理比较

方便:有些编译运行的程序,在安装完成后,需要为他们编写启动停止管理脚本,写入和维护可能很痛苦,而且进程在异常崩溃结束时,许多程序都不会正确重新启动的。Supervisord启动管理的程序进程是作为其子进程来运行的,并且可以配置为在进程崩溃停止时自动重新启动它们。

准确:在Unix上的进程通常很难获得准确的up/down状态。Pidfiles经常说谎。Supervisord 将进程作为子进程启动,所以它总是知道其子进程的正确的up/down状态,可以方便的对这些数据进行查询。

进程分组:进程支持分组启动和停止,也支持启动顺序,即‘优先级’,supervisor允许为进程分配优先级,并允许用户通过supervisorctl客户端发出命令,如“全部启动”和”重新启动所有“,它们以预先分配的优先级顺序启动。还可以将进程分为”进程组“,一组逻辑关联的进程可以作为一个单元停止或启动。

Supervisor 特点

简单:supervisor通过简单的INI风格的配置文件进行配置管理,易于学习,并提供了许多每个进程选项,如重新启动失败的进程和日志的自动切割。

集中:supervisor提供一个start、stop和监控进程的地方,进程可以单独或分组进行控制。可以通过supervisor的本地或远程命令行管理和web管理(一般为了安全,web通常需要禁用)

高效:supervisor通过fork/exec启动子进程,子进程需要前台运行,操作系统进程终止时去通知supervisor,而不像一些我们需要写脚本去定期轮询PID文件来重新启动失败的进程。

可扩展:supervisor有一个简单的事件(event)通知协议,还有一个用于控制的XML-RPC接口,可以用Python开发人员来扩展构建。

兼容:supervisor由Python编写,在除Windows操作系统以外基本都支持,如linux,Mac OS x,solaris,FreeBSD系统。

Supervisor 组建构成

supervisord:supervisor服务器的进程名是supervisord。它主要负责在自己的调用中启动子程序,响应客户端的命令,重新启动崩溃或退出的进程,记录其子进程stdout和stderr的输出,以及生成和处理对应于子进程生命周期中的”event“。

服务器进程使用的配置文件,通常路径存放在 /etc/supervisord.conf 中。此配置文件是INI格式的配置文件。

supervisorctl:supervisor命令行的客户端名称是supervisorctl。它为supervisord提供了一个类似于shell的交互界面。使用supervisorctl,用户可以查看不同的supervisord进程列表,获取控制子进程的状态,如停止和启动子进程。

web服务器:一个可以通过Web界面来查看和控制进程的状态,默认监听在9001上。

1、安装以及服务启动

yum install epel-release     # 安装企业版 Linux 附加软件包(EPEL)
yum install -y supervisor    # yum 安装 supervisor
systemctl enable supervisord # 开机自启动
systemctl start supervisord  # 启动supervisord服务(服务端)
systemctl status supervisord # 查看supervisord服务状态(服务端)
ps -ef|grep supervisord      # 查看是否存在supervisord进程

生成配置文件

用supervisor自带命令生成一个标准的配置文件

/usr/bin/echo_supervisord_conf > /etc/supervisord.conf
~]# cat /etc/supervisord.conf
; Sample supervisor config file.
;
; For more information on the config file, please see:
; http://supervisord.org/configuration.html
;
; Notes:
;  - Shell expansion ("~" or "$HOME") is not supported.  Environment
;    variables can be expanded using this syntax: "%(ENV_HOME)s".
;  - Comments must have a leading space: "a=b ;comment" not "a=b;comment".

[unix_http_server]
file=/tmp/supervisor.sock   ; (socket 文件的路径)
;chmod=0700                 ; socket 文件权限 (default 0700)
;chown=nobody:nogroup       ; socket 文件属主:属组
;username=user              ; (启动http的用户 (open server))
;password=123               ; (默认的密码 (open server))

;[inet_http_server]         ; 默认禁用tcp监听的http 服务
;port=127.0.0.1:9001        ; (指定监听在本机ip地址和端口)
;username=user              ; (默认启动http服务的用户)
;password=123               ; (默认的密码)

[supervisord]
logfile=/tmp/supervisord.log ; (主日志文件的存放位置,默认在程序的工作启动目录)
logfile_maxbytes=50MB        ; (主日志文件的最大值,之后进行切割;默认 50MB)
logfile_backups=10           ; (主日志文件备份的数目;默认 10)
loglevel=info                ; (日志级别;默认是info; 其它: debug,warn,trace)
pidfile=/tmp/supervisord.pid ; (supervisord 运行时的pidfile路径;默认 supervisord.pid)
nodaemon=false               ; (如果为true,程序就以前台运行;默认是 false)
minfds=1024                  ; (min. 启动有效的文件描述符数目;默认 1024)
minprocs=200                 ; (min. 有效进程描述符;默认 200)
;umask=022                   ; (进程文件创建的默认权限;默认 022)
;user=chrism                 ; (默认是当前启动的用户)
;identifier=supervisor       ; (supervisord 标识符, 默认是'supervisor')
;directory=/tmp              ; (默认启动时间不会切换)
;nocleanup=true              ; (在启动时不清理临时文件;默认值为false)
;childlogdir=/tmp            ; ('AUTO' 子进程日志目录, 默认 $TEMP)
;environment=KEY="value"     ; (增加一个环境变量键值对:key=”value“)
;strip_ansi=false            ; (在log日志里去掉ansi转义编码; 默认是 false)

; 下面的部分选项必须保留在RPC的配置文件中
; (supervisorctl/web 接口) 使用以下配置来管理
; added by defining them in separate rpcinterface: sections
[rpcinterface:supervisor]
supervisor.rpcinterface_factory = supervisor.rpcinterface:make_main_rpcinterface

[supervisorctl]
serverurl=unix:///tmp/supervisor.sock ; use a unix:// URL  for a unix socket
;serverurl=http://127.0.0.1:9001 ; use an http:// url to specify an inet socket
;username=chris              ; should be same as http_username if set
;password=123                ; should be same as http_password if set
;prompt=mysupervisor         ; cmd line prompt (default "supervisor")
;history_file=~/.sc_history  ; use readline history if available

; 以下是被管理的示例程序显示所有可能用到的配置。
; 创建一个或“多个”程序: 要遵循以下的键值对规则。
; supervisor.

;[program:theprogramname]
;command=/bin/cat              ; 程序的启动命令 (使用绝对路径)
;process_name=%(program_name)s ; process_name 表示 (默认是 %(program_name)s)
;numprocs=1                    ; 启动时的进程数 (默认 1)
;directory=/tmp                ; 执行时切换到的目录 (def no cwd)
;umask=022                     ; umask for process (default None)
;priority=999                  ; 相对启动优先级(default 999)
;autostart=true                ; 是否跟随supervisord程序启动该监控程序 (default: true)
;startsecs=1                   ; # 在设定时间内,程序必须保持运行 (def. 1)
;startretries=3                ; 当启动失败时尝试的最大次数(default 3)
;autorestart=unexpected        ; 如果退出后,什么状态退出的去重启,默认非意外的(def: unexpected)
;exitcodes=0,2                 ; 'expected' 符合退出代码之后去重启 (default 0,2)
;stopsignal=QUIT               ; 用于杀死进程的信号 (default TERM)
;stopwaitsecs=10               ; 最大等待秒数 SIGKILL (default 10)
;stopasgroup=false             ; 发送停止信号到Unix进程组 (default false)
;killasgroup=false             ; SIGKILL UNIX进程组 (def false)
;user=chrism                   ; setuid to this UNIX account to run the program
;redirect_stderr=true          ; 是否开启程序标准错误输出的重定向 (default false)
;stdout_logfile=/a/path        ; 标准输出路径; default AUTO
;stdout_logfile_maxbytes=1MB   ; 文件最大大小 # 日志文件进行切割 (default 50MB)
;stdout_logfile_backups=10     ; # 日志文件备份数目 (default 10)
;stdout_capture_maxbytes=1MB   ; ‘捕获模式’中的字节数 (default 0)
;stdout_events_enabled=false   ; 在标准输出写入文件时发出事件 (default false)
;stderr_logfile=/a/path        ; 标准错误输出, NONE for none; default AUTO
;stderr_logfile_maxbytes=1MB   ; 文件最大大小 # logfile bytes b4 rotation (default 50MB)
;stderr_logfile_backups=10     ; # of stderr logfile backups (default 10)
;stderr_capture_maxbytes=1MB   ; number of bytes in 'capturemode' (default 0)
;stderr_events_enabled=false   ; emit events on stderr writes (default false)
;environment=A="1",B="2"       ; 添加进程环境变量 (def no adds)
;serverurl=AUTO                ; 覆盖serverurl计算 (childutils)

;下面是event事件部分所有可能设置的值,大部分同上面一样。
; eventlistener subsection values, create one or more 'real'
; eventlistener: sections to be able to handle event notifications
; sent by supervisor.

;[eventlistener:theeventlistenername]
;command=/bin/eventlistener    ; the program (relative uses PATH, can take args)
;process_name=%(program_name)s ; process_name expr (default %(program_name)s)
;numprocs=1                    ; number of processes copies to start (def 1)
;events=EVENT                  ; event notif. types to subscribe to (req'd)
;buffer_size=10                ; event buffer queue size (default 10)
;directory=/tmp                ; directory to cwd to before exec (def no cwd)
;umask=022                     ; umask for process (default None)
;priority=-1                   ; the relative start priority (default -1)
;autostart=true                ; start at supervisord start (default: true)
;startsecs=1                   ; # of secs prog must stay up to be running (def. 1)
;startretries=3                ; max # of serial start failures when starting (default 3)
;autorestart=unexpected        ; autorestart if exited after running (def: unexpected)
;exitcodes=0,2                 ; 'expected' exit codes used with autorestart (default 0,2)
;stopsignal=QUIT               ; signal used to kill process (default TERM)
;stopwaitsecs=10               ; max num secs to wait b4 SIGKILL (default 10)
;stopasgroup=false             ; send stop signal to the UNIX process group (default false)
;killasgroup=false             ; SIGKILL the UNIX process group (def false)
;user=chrism                   ; setuid to this UNIX account to run the program
;redirect_stderr=false         ; redirect_stderr=true is not allowed for eventlisteners
;stdout_logfile=/a/path        ; stdout log path, NONE for none; default AUTO
;stdout_logfile_maxbytes=1MB   ; max # logfile bytes b4 rotation (default 50MB)
;stdout_logfile_backups=10     ; # of stdout logfile backups (default 10)
;stdout_events_enabled=false   ; emit events on stdout writes (default false)
;stderr_logfile=/a/path        ; stderr log path, NONE for none; default AUTO
;stderr_logfile_maxbytes=1MB   ; max # logfile bytes b4 rotation (default 50MB)
;stderr_logfile_backups=10     ; # of stderr logfile backups (default 10)
;stderr_events_enabled=false   ; emit events on stderr writes (default false)
;environment=A="1",B="2"       ; process environment additions
;serverurl=AUTO                ; override serverurl computation (childutils)

; The below sample group section shows all possible group values,
; create one or more 'real' group: sections to create "heterogeneous"
; process groups.

;[group:thegroupname]
;programs=progname1,progname2  ; 这里的progname1,progname2就是定义的监控管理程序的名字,如[program:x]这里就是x
;priority=999                  ; the relative start priority (default 999)

;  下面的 [include] 选项只能包含一个files 设置,功能是定义supervisor管理程序的配置文件,可以单独的移除去,和主配置文件分开,方便。; setting can list multiple files (separated by whitespace or
; newlines).  It can also contain wildcards.  The filenames are
; interpreted as relative to this file.  Included files *cannot*
; include files themselves.

;[include]
;files = relative/directory/*.ini         ;定义管理监控程序的配置文件的路径

管理命令

supervisorctl status # 查看所有进程的状态
supervisorctl stop program_name    # 停止
supervisorctl start program_name   # 启动
supervisorctl restart program_name # 重启某个程序
supervisorctl stop all             # 停止所有
supervisorctl reload  # 重新启动配置中的所有程序
supervisorctl update  # 更新新的配置到supervisord
supervisord -c /etc/supervisord.conf # yum 安装后,启动默认方式
supervisorctl start/stop/restart/status group       # 管理所有属于名为 group 这个分组的进程
supervisorctl start/stop/restart/status group:name1 # 管理分组里指定的进程

添加配置实例

; 设置进程的名称,使用 supervisorctl 来管理进程时需要使用该进程名 我这里就叫做nginx了!
[program:nginx] 
; 需要执行的命令
command=/usr/sbin/nginx -g 'daemon off;' 
; 命令执行的目录或者说执行 command 之前,先切换到工作目录 可以理解为在执行命令前会切换到这个目录 在我这基本没啥用
directory=/etc/nginx
; 是否自动启动
autostart=true
; 程序意外退出是否自动重启
autorestart=true
; 如果为true,则stderr的日志会被写入stdout日志文件中,  理解为重定向输出的日志
redirect_stderr=true
; 启动优先级 越大优先级越高
priority=10
; 子进程的stdout的日志路径 输出日志文件,如果日志路径不存在,会导致无法启动错误
stdout_logfile=/data/logs/supervisord/nginx.log
; 错误日志文件 当 redirect_stderr=true 时,stderr_logfile就不用
stderr_logfile=/data/logs/supervisord/nginx.err.log 

修改后,更新 supervisor


supervisorctl reread # 重新读取配置
supervisorctl update # 更新配置
supervisorctl restart nginx  # 重启 nginx
killall nginx  # 杀掉所有的 nginx 进程. 已经杀不死了 说明守护进程配置成功

宝塔Supervisor管理器生成实例

[program:test-call-serve-excel]
command=redis xxxxxxx
directory=/www/wwwroot/test-call
autorestart=true
startsecs=3
startretries=3
stdout_logfile=/www/server/panel/plugin/supervisor/log/test-serve.out.log
stderr_logfile=/www/server/panel/plugin/supervisor/log/test-serve.err.log
stdout_logfile_maxbytes=2MB
stderr_logfile_maxbytes=2MB
user=www
priority=999
numprocs=4
process_name=%(program_name)s_%(process_num)02d
]]>
0 https://ascwh.com/399.html#comments https://ascwh.com/feed/
Linux WireGuard 一键安装脚本 https://ascwh.com/398.html https://ascwh.com/398.html Thu, 19 May 2022 22:35:00 +0800 ASCWH 安装安装
wget https://raw.githubusercontent.com/yobabyshark/wireguard/master/wireguard_install.sh && chmod +x wireguard_install.sh && ./wireguard_install.sh

# 验证是否安装成功
[root@localhost ~]# modprobe wireguard && lsmod | grep wireguard
wireguard             204791  0 
ip6_udp_tunnel         12755  1 wireguard
udp_tunnel             14423  1 wireguard

安装后配置文件目录(/etc/wireguard/)里的文件:

wireguard/
├── client.conf     #客户端配置文件
├── cprivatekey     #客户端私钥
├── cpublickey         #客户端公钥
├── sprivatekey     #服务端私钥
├── spublickey         #服务端公钥
└── wg0.conf         #服务端配置信息

配置说明

WireGuard的多用户配置特别简单,只需要生成一对客户端密匙(公匙+私匙),在服务端配置文件中新增一段 [Peer] 写上新的客户端公匙和客户端的内网IP地址即可。(当然我们可以用命令快捷添加,或者手动修改服务端配置文件也行。)

而各客户端账号配置文件的区别也只是 [Interface] 中的客户端私匙和客户端内网IP地址不同罢了。

另外我们需要明白一个对等原则:

服务端配置文件中的 [Interface] 是保存自己的服务端私匙,而客户端配置文件中的 [Interface] 同样保存自己的客户端私匙。
服务端配置文件中的 [Peer] 是保存客户端的公匙,而客户端配置文件中的 [Peer] 是保存服务端的公匙。
服务端与客户端都是互相保存自己的私匙在 [Interface] 中,互相保存对方公匙在 [Peer] 中。

如果在Centos7操作,注意跑下yum update(对内核版本有要求)
系统转发注意开启

net.ipv4.ip_forward = 1

#服务器常用命令:
    wg show wg0                     #查看wg0虚拟网卡当前状态
    systemctl status wg-quick@wg0
    systemctl restart wg-quick@wg0

#开启关闭wg指令:
wg-quick up wg0
wg-quick down wg0

# 默认wg0.conf文件:
[Interface]
Address = 10.0.0.1/24
SaveConfig = true
PostUp = firewall-cmd --zone=public --add-port 3306/udp && firewall-cmd --zone=public --add-masquerade
PostDown = firewall-cmd --zone=public --remove-port 3306/udp && firewall-cmd --zone=public --remove-masquerade
ListenPort = 3306
PrivateKey = 4KQWlx+Md0I5uHj+D9DjJZAwpTFFjG7E03/9FFjjJ3o=

[Peer]
PublicKey = 6enJBddxwwaIrD4lV6g9Uiz+RDmh7NGmyYrFbGpJuF0=
AllowedIPs = 10.0.0.2/32

WireGuard 配置说明

[Interface]
ListenPort = 8002  #本机监听 WireGuard 端口
PrivateKey = 填写本机的 privatekey 内容  #本机加密私钥
DNS = 1.1.1.1  #强制本机使用 DNS 服务器
MTU = 1300  #强制本机使用指定 MTU 值
#Table = 100 #强制本机将注入系统的路由的 table 指定为 100 ( 系统默认为 255 和 254 )
#PostUp = ip rule add from 10.10.1.0/24 table 100 ; ip rule add from 10.10.2.0/24 table 100
#PreDown = ip rule delete from 10.10.1.0/24 table 100 ; ip rule delete from 10.10.2.0/24 table 100

[Peer]
PublicKey = 填写对端的 publickey 内容  #本机加密的对端公钥(加密后数据仅对端可以解密)
AllowedIPs = 172.16.1.11/32  #本机允许的对端设备的 IP 地址段,其实就是在本机中这个虚拟网卡对端都有哪些设备 IP 地址(多 peer 不可重复)
Endpoint = another.domain.name:8001  #对端 WireGuard 的外部 IP(可以有一侧的 IP 地址是虚假的公网 IP)
PersistentKeepalive = 25   #当会话存在一端 IP 地址为 NAT 地址或虚假公网 IP 地址时,由该方阶段性每 25 秒发送 keepalive 报文保持会话的可用性,防止被设备终止。

在上方基础上,额外要注意的是
1、如果你存在多个 [Peer] ,则在下面直接增加一个新的 [Peer] 栏目
2、如果多个 Peer 存在不同的 IP,请不要让 AllowedIPs 存在重叠的 IP 地址段(比如配置多个相同/24 只有一个生效)
3、Endpoint 既支持以域名的方式访问,也支持以 IP 的方式访问。
4、会话链接的建立只要保证两端数据在一台设备上成功协商,即使有一端 IP 地址后期变更过也不会影响 VPN 的稳定性。
5、使用命令 wg 可以看到当前服务的运行情况。
6、如果你是本来打算连服务器的移动端,则在配置中你需要将移动端的 "AllowedIPs =" 补写一条内容 0.0.0.0/0,::0 以允许所有流量
7、前面标注 # 的三条内容,作为 iproute2 的使用方式,通过指定路由条目的 table 与 ip rule 联动控制策略流量转发。
8、PreUp,PostUp,PreDown,PostDown 这四个命令参数,是作为 wg-quick 快速设置/删除接口之前/之后由 bash(1)执行的四条命令,常用于配置自定义 DNS 或防火墙规则。 特殊字符串 %i 作为变量替代所控制的 INTERFACE 配置名。 每个命令参数都支持多条命令,参数内的多条命令将按前后顺序依次执行。

]]>
0 https://ascwh.com/398.html#comments https://ascwh.com/feed/
Mac 配置 WireGuard 客户端 https://ascwh.com/397.html https://ascwh.com/397.html Thu, 19 May 2022 21:14:00 +0800 ASCWH WireGuard 是一个易于配置、快速且安全的开源组网,它利用了最新的加密技术。目的是提供一种更快、更简单、更精简的通用 VPN,它可以轻松地在树莓派这类低端设备到高端服务器上部署。
IPsec 和 OpenVPN 等大多数其他解决方案是几十年前开发的。安全研究人员和内核开发人员 Jason Donenfeld 意识到它们速度慢且难以正确配置和管理。
这让他创建了一个新的开源 VPN 协议和解决方案,它更加快速、安全、易于部署和管理。
WireGuard 最初是为 Linux 开发的,但现在可用于 Windows、macOS、BSD、iOS 和 Android。它仍在活跃开发中

安装 WireGuard 客户端

brew install wireguard-tools

配置 WireGurad 客户端

sudo mkdir /usr/local/etc/wireguard
sudo touch /usr/local/etc/wireguard/wg0.conf

wg0.conf 文件如下:

[Interface]
Address = 10.200.200.2/32
PrivateKey = <client_private_key>
MTU = 1420

[Peer]
PublicKey = <server_public_key>
Endpoint = <SERVER_IP:SERVER_PORT>
AllowedIPs = 10.200.0.0/16
PersistentKeepalive = 25

启动 WireGuard

sudo wg-quick up wg0  # 启动Wireguard
sudo wg-quick down wg0 #关闭WIreguard
sudo wg # 查看Wireguard状态
]]>
0 https://ascwh.com/397.html#comments https://ascwh.com/feed/
PHP获取今天、明天、昨天、上周、本周、上月、本月 的基础方法 https://ascwh.com/396.html https://ascwh.com/396.html Tue, 17 May 2022 23:28:30 +0800 ASCWH <?php declare (strict_types=1); namespace app\librarys; /** * Class DateRange 日期范围处理 * @package app\librarys */ class DateRange { /** * @var */ private static $instance; /** * @var */ public $timestamp; /** * @var */ public $currentDate; /** * */ private function __construct() { $this->currentDate = time(); } /** * @title __clone */ private function __clone() { } /** * @title getInstance * @return \app\librarys\DateRange */ public static function getInstance() { if (!self::$instance) { self::$instance = new self(); } return self::$instance; } /** * @title setTimestamp 设置时间格式 默认为时间戳, 可传入 date 方法的 时间参数 * * @param $timestamp * * @return */ public function setTimestamp($timestamp) { $this->timestamp = $timestamp; return $this; } /** * @title setCurrentDate * * @param $date * * @return \app\librarys\DateRange */ public function setCurrentDate($date) { $this->currentDate = $date; return $this; } /** * @title execute 传入时间格式 并返回开始时间和结束时间 * * @param $dateType * * @return array|false[]|float[]|int[]|string[] */ public final function execute($dateType) { switch ($dateType) { case 'today': // 今天 $range = $this->getDateRangeByToday(); break; case 'tomorrow': // 明天 $range = $this->getDateRangeByTomorrowday(); break; case 'yesterday': // 昨天 $range = $this->getDateRangeByYesterday(); break; case 'week': // 本周 $range = $this->getDateRangeByWeek(); break; case 'lastWeek': // 上周 $range = $this->getDateRangeByLastWeek(); break; case 'month': // 本月 $range = $this->getDateRangeByMonth(); break; case 'lastMonth': // 上月 $range = $this->getDateRangeByLastMonth(); break; default: $range = []; } return $range; } /** * @title getDateRangeByToday 返回今天的时间范围 * @return false[]|float[]|int[]|string[] */ protected function getDateRangeByToday() { return [ $this->getTime(strtotime(date('Y-m-d', $this->currentDate))), $this->getTime(strtotime(date('Y-m-d', $this->currentDate)) + 86400 - 1) ]; } /** * @title getDateRangeByYesterday 返回昨日的时间范围 * @return false[]|float[]|int[]|string[] */ protected function getDateRangeByYesterday() { $yesterday = strtotime('-1 day', $this->currentDate); return [ $this->getTime(strtotime(date('Y-m-d', $yesterday))), $this->getTime(strtotime(date('Y-m-d', $yesterday)) + 86400 - 1) ]; } /** * @title getDateRangeByTomorrowday 返回明天的时间范围 * @return array */ protected function getDateRangeByTomorrowday() { $yesterday = strtotime('+1 day', $this->currentDate); return [ $this->getTime(strtotime(date('Y-m-d', $yesterday))), $this->getTime(strtotime(date('Y-m-d', $yesterday)) + 86400 - 1) ]; } /** * @title getDateRangeByWeek 获取本周的时间范围 * @return array */ protected function getDateRangeByWeek() { return [ $this->getTime(strtotime(date('Y-m-d', strtotime('this week Monday', $this->currentDate)))), $this->getTime(strtotime(date('Y-m-d', strtotime('this week Sunday', $this->currentDate))) + 86400 - 1) ]; } /** * @title getDateRangeByLastWeek 获取上周的时间范围 * @return array */ protected function getDateRangeByLastWeek() { return [ $this->getTime(strtotime(date('Y-m-d', strtotime('last week Monday', $this->currentDate)))), $this->getTime(strtotime(date('Y-m-d', strtotime('last week Sunday', $this->currentDate))) + 86400 - 1) ]; } /** * @title getDateRangeByMonth 获取本月的时间范围 * @return array */ protected function getDateRangeByMonth() { return [ $this->getTime(strtotime(date('Y-m-1', $this->currentDate))), $this->getTime(strtotime(date('Y-m-t', $this->currentDate)) + 86400 - 1) ]; } /** * @title getDateRangeByLastMonth 获取上月的时间范围 * @return false[]|float[]|int[]|string[] */ protected function getDateRangeByLastMonth() { return [ $this->getTime(strtotime(date('Y-m-d', strtotime('first day of previous month', $this->currentDate)))), $this->getTime(strtotime(date('Y-m-d', strtotime('last day of previous month', $this->currentDate))) + 86400 - 1) ]; } /** * @title getTime 按 $this->timestamp 返回时间格式 * * @param string $time * * @return false|float|int|string */ public function getTime($time = '') { switch (true) { case (empty($time)): $timeInt = time(); break; case (is_numeric($time)): $timeInt = $time; break; case (is_string($time)): $timeInt = strtotime($time); if (!$timeInt) { $timeInt = time(); } break; default: $timeInt = time(); } return empty($this->timestamp) ? $timeInt : date($this->timestamp, (int)$timeInt); } /** * @title getTimeRangeByDateRange 获取给定的开始时间和结束时间的每天时间 * * @param int $start * @param int $end * * @return array|false */ public function getTimeRangeByDateRange(int $start, int $end) { if ($start > $end) { return false; } $diff = ($end - $start) / 86400; $day = []; for ($i = 0; $i <= $diff; $i++) { $day[] = $this->getTime($start + $i * 86400); } return $day; } } ]]> 0 https://ascwh.com/396.html#comments https://ascwh.com/feed/ Go语言 https://ascwh.com/395.html https://ascwh.com/395.html Thu, 12 May 2022 22:24:00 +0800 ASCWH Go 语言简介
Go是一种新的语言,一种并发的、带垃圾回收的、快速编译的语言。Go是谷歌2009年发布的第二款编程语言。2009年7月份,谷歌曾发布了Simple语言,它是用来开发Android应用的一种BASIC语言。

Go 是一个开源的编程语言,它能让构造简单、可靠且高效的软件变得容易。

Go是从2007年末由Robert Griesemer, Rob Pike, Ken Thompson主持开发,后来还加入了Ian Lance Taylor, Russ Cox等人,并最终于2009年11月开源,在2012年早些时候发布了Go 1稳定版本。现在Go的开发已经是完全开放的,并且拥有一个活跃的社区。

Go 语言特色
简洁、快速、安全
并行、有趣、开源
内存管理、v数组安全、编译迅速
Go 语言用途
Go 语言被设计成一门应用于搭载 Web 服务器,存储集群或类似用途的巨型中央服务器的系统编程语言。

对于高性能分布式系统领域而言,Go 语言无疑比大多数其它语言有着更高的开发效率。它提供了海量并行的支持,这对于游戏服务端的开发而言是再好不过了。

为什么要学习Go语言
如果你要创建系统程序,或者基于网络的程序,Go语言是很不错的选择。作为一种相对较新的语言,它是由经验丰富且受人尊敬的计算机科学家设计的,旨在应对创建大型并发网络程序面临的挑战。

在Go语言出现之前,开发者们总是面临非常艰难的抉择,究竟是使用执行速度快但是编译速度并不理想的语言(如:C++),还是使用编译速度较快但执行效率不佳的语言(如:.NET、Java),或者说开发难度较低但执行速度一般的动态语言呢?显然,Go语言在这 3 个条件之间做到了最佳的平衡:快速编译,高效执行,易于开发。

Go语言支持交叉编译,比如说你可以在运行 Linux 系统的计算机上开发可以在 Windows 上运行的应用程序。这是第一门完全支持 UTF-8 的编程语言,这不仅体现在它可以处理使用 UTF-8 编码的字符串,就连它的源码文件格式都是使用的 UTF-8 编码。Go语言做到了真正的国际化!!

]]>
0 https://ascwh.com/395.html#comments https://ascwh.com/feed/
CentOS 8 结束生命周期如何切换源 https://ascwh.com/393.html https://ascwh.com/393.html Mon, 11 Apr 2022 18:51:00 +0800 ASCWH 公网用户:

mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backup
wget -O /etc/yum.repos.d/CentOS-Base.repo https://mirrors.aliyun.com/repo/Centos-vault-8.5.2111.repo
yum clean all && yum makecache

阿里云ecs用户:

mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backup
wget -O /etc/yum.repos.d/CentOS-Base.repo https://mirrors.aliyun.com/repo/Centos-vault-8.5.2111.repo
sed -i 's/mirrors.cloud.aliyuncs.com/url_tmp/g' /etc/yum.repos.d/CentOS-Base.repo && sed -i 's/mirrors.aliyun.com/mirrors.cloud.aliyuncs.com/g' /etc/yum.repos.d/CentOS-Base.repo && sed -i 's/url_tmp/mirrors.aliyun.com/g' /etc/yum.repos.d/CentOS-Base.repo
yum clean all && yum makecache
]]>
0 https://ascwh.com/393.html#comments https://ascwh.com/feed/
PhpSpreadsheet 导入导出基础应用 https://ascwh.com/392.html https://ascwh.com/392.html Thu, 02 Sep 2021 16:56:00 +0800 ASCWH 由于PHPExcel已经不再维护,PhpSpreadsheet是PHPExcel的下一个版本。PhpSpreadsheet是一个用纯PHP编写的库,并引入了命名空间,PSR规范等。这里简单介绍下PhpSpreadsheet的导入导出功能。

1、安装

使用composer安装:
composer require phpoffice/phpspreadsheet

GitHub下载:https://github.com/PHPOffice/PhpSpreadsheet

2、excel文件导出

    // 命名空间引入
    use PhpOffice\PhpSpreadsheet\IOFactory;
    use PhpOffice\PhpSpreadsheet\Spreadsheet;

    /**
     * @title export 导出
     */
    public function export() {
        $data  = [
            ['title1' => '111', 'title2' => '222'],
            ['title1' => '111', 'title2' => '222'],
            ['title1' => '111', 'title2' => '222']
        ];
        $title = ['标题1', '标题2'];
        array_unshift($data, $title);

        // Create new Spreadsheet object
        $spreadsheet = new Spreadsheet();
        $sheet       = $spreadsheet->getActiveSheet()->setTitle('工作表标题');

        foreach ($data as $k => $item) {
            $column = 1;
            foreach ($item as $value) {
            // 因不想定义 A…  一大串大写字母 所以使用数字单元格坐标设置单元格值。
                $sheet->setCellValueByColumnAndRow($column, $k + 1, $value);
                $column++;
            }
        }

        // Redirect output to a client’s web browser (Xlsx)
        header('Content-Type: application/vnd.openxmlformats-officedocument.spreadsheetml.sheet');
        header('Content-Disposition: attachment;filename="01simple.xlsx"');
        header('Cache-Control: max-age=0');
        // If you're serving to IE 9, then the following may be needed
        header('Cache-Control: max-age=1');
        // If you're serving to IE over SSL, then the following may be needed
        header('Expires: Mon, 26 Jul 1997 05:00:00 GMT');              // Date in the past
        header('Last-Modified: ' . gmdate('D, d M Y H:i:s') . ' GMT'); // always modified
        header('Cache-Control: cache, must-revalidate');               // HTTP/1.1
        header('Pragma: public');                                      // HTTP/1.0
        $writer = IOFactory::createWriter($spreadsheet, 'Xlsx');
        $writer->save('php://output');
        exit;
    }

3、读取excel文件内容

/**
    // 命名空间引入
    use PhpOffice\PhpSpreadsheet\IOFactory;
    
    
     * @title  readExcel 读取excel文件内容
     * @return array
     */
    public function readExcel() {
        $inputFileName = './test.xlsx';
        $spreadsheet   = IOFactory::load($inputFileName);
        //  toArray() 具体参数参考官方文档
        $sheetData = $spreadsheet->getActiveSheet()->toArray(null, true, true, true);
        return $sheetData;
    }
]]>
0 https://ascwh.com/392.html#comments https://ascwh.com/feed/