mirror of
https://github.com/siyuan-note/siyuan.git
synced 2026-03-07 13:20:16 +01:00
291 lines
7 KiB
Go
291 lines
7 KiB
Go
// SiYuan - Refactor your thinking
|
||
// Copyright (c) 2020-present, b3log.org
|
||
//
|
||
// This program is free software: you can redistribute it and/or modify
|
||
// it under the terms of the GNU Affero General Public License as published by
|
||
// the Free Software Foundation, either version 3 of the License, or
|
||
// (at your option) any later version.
|
||
//
|
||
// This program is distributed in the hope that it will be useful,
|
||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
// GNU Affero General Public License for more details.
|
||
//
|
||
// You should have received a copy of the GNU Affero General Public License
|
||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||
|
||
package util
|
||
|
||
import (
|
||
"fmt"
|
||
"net"
|
||
"net/http"
|
||
"net/url"
|
||
"os"
|
||
"strconv"
|
||
"strings"
|
||
"time"
|
||
|
||
"github.com/88250/gulu"
|
||
"github.com/88250/lute/ast"
|
||
"github.com/gin-gonic/gin"
|
||
"github.com/imroc/req/v3"
|
||
"github.com/siyuan-note/httpclient"
|
||
"github.com/siyuan-note/logging"
|
||
)
|
||
|
||
// GetPrivateIPv4s 获取本地所有的私有 IPv4 地址(排除虚拟网卡)
|
||
func GetPrivateIPv4s() (ret []string) {
|
||
ret = []string{}
|
||
|
||
interfaces, err := net.Interfaces()
|
||
if err != nil {
|
||
return
|
||
}
|
||
|
||
// 常见的虚拟网卡名称关键字黑名单
|
||
virtualKeywords := []string{"docker", "veth", "br-", "vmnet", "vbox", "utun", "tun", "tap", "bridge", "cloud", "hyper-"}
|
||
|
||
for _, itf := range interfaces {
|
||
// 1. 基础状态过滤:必须是启动状态且不能是回环网卡
|
||
if itf.Flags&net.FlagUp == 0 || itf.Flags&net.FlagLoopback != 0 {
|
||
continue
|
||
}
|
||
|
||
// 2. 硬件地址过滤:物理网卡通常必须有 MAC 地址
|
||
if len(itf.HardwareAddr) == 0 {
|
||
continue
|
||
}
|
||
|
||
// 3. 名称过滤:排除已知虚拟网卡前缀
|
||
name := strings.ToLower(itf.Name)
|
||
isVirtual := false
|
||
for _, kw := range virtualKeywords {
|
||
if strings.Contains(name, kw) {
|
||
isVirtual = true
|
||
break
|
||
}
|
||
}
|
||
if isVirtual {
|
||
continue
|
||
}
|
||
|
||
// 4. 提取并校验 IP
|
||
addrs, err := itf.Addrs()
|
||
if err != nil {
|
||
continue
|
||
}
|
||
|
||
for _, addr := range addrs {
|
||
ipNet, ok := addr.(*net.IPNet)
|
||
if !ok {
|
||
continue
|
||
}
|
||
|
||
ip := ipNet.IP
|
||
// 仅保留 IPv4 且必须是私有局域网地址 (10.x, 172.16.x, 192.168.x)
|
||
if ip.To4() != nil && ip.IsPrivate() {
|
||
ret = append(ret, ip.String())
|
||
}
|
||
}
|
||
}
|
||
return
|
||
}
|
||
|
||
func IsLocalHostname(hostname string) bool {
|
||
if "localhost" == hostname || strings.HasSuffix(hostname, ".localhost") {
|
||
return true
|
||
}
|
||
if ip := net.ParseIP(hostname); nil != ip {
|
||
return ip.IsLoopback()
|
||
}
|
||
return false
|
||
}
|
||
|
||
func IsLocalHost(host string) bool {
|
||
if hostname, _, err := net.SplitHostPort(strings.TrimSpace(host)); err != nil {
|
||
return false
|
||
} else {
|
||
return IsLocalHostname(hostname)
|
||
}
|
||
}
|
||
|
||
func IsLocalOrigin(origin string) bool {
|
||
if u, err := url.Parse(origin); err == nil {
|
||
return IsLocalHostname(u.Hostname())
|
||
}
|
||
return false
|
||
}
|
||
|
||
func IsOnline(checkURL string, skipTlsVerify bool, timeout int) bool {
|
||
if "" == checkURL {
|
||
return false
|
||
}
|
||
|
||
u, err := url.Parse(checkURL)
|
||
if err != nil {
|
||
logging.LogWarnf("invalid check URL [%s]", checkURL)
|
||
return false
|
||
}
|
||
if u.Scheme == "file" {
|
||
filePath := strings.TrimPrefix(checkURL, "file://")
|
||
_, err := os.Stat(filePath)
|
||
return err == nil
|
||
}
|
||
|
||
if isOnline(checkURL, skipTlsVerify, timeout) {
|
||
return true
|
||
}
|
||
|
||
logging.LogWarnf("network is offline [checkURL=%s]", checkURL)
|
||
return false
|
||
}
|
||
|
||
func IsPortOpen(port string) bool {
|
||
timeout := time.Second
|
||
conn, err := net.DialTimeout("tcp", net.JoinHostPort("127.0.0.1", port), timeout)
|
||
if err != nil {
|
||
return false
|
||
}
|
||
if nil != conn {
|
||
conn.Close()
|
||
return true
|
||
}
|
||
return false
|
||
}
|
||
|
||
func isOnline(checkURL string, skipTlsVerify bool, timeout int) (ret bool) {
|
||
c := req.C().
|
||
SetTimeout(time.Duration(timeout) * time.Millisecond).
|
||
SetProxy(httpclient.ProxyFromEnvironment).
|
||
SetUserAgent(UserAgent)
|
||
if skipTlsVerify {
|
||
c.EnableInsecureSkipVerify()
|
||
}
|
||
|
||
for i := 0; i < 2; i++ {
|
||
resp, err := c.R().Get(checkURL)
|
||
if resp.GetHeader("Location") != "" {
|
||
return true
|
||
}
|
||
|
||
switch err.(type) {
|
||
case *url.Error:
|
||
if err.(*url.Error).URL != checkURL {
|
||
// DNS 重定向
|
||
logging.LogWarnf("network is online [DNS redirect, checkURL=%s, retURL=%s]", checkURL, err.(*url.Error).URL)
|
||
return true
|
||
}
|
||
}
|
||
|
||
ret = err == nil
|
||
if ret {
|
||
break
|
||
}
|
||
|
||
logging.LogWarnf("check url [%s] is online failed: %s", checkURL, err)
|
||
time.Sleep(1 * time.Second)
|
||
}
|
||
return
|
||
}
|
||
|
||
func GetRemoteAddr(req *http.Request) string {
|
||
ret := req.Header.Get("X-forwarded-for")
|
||
ret = strings.TrimSpace(ret)
|
||
if "" == ret {
|
||
ret = req.Header.Get("X-Real-IP")
|
||
}
|
||
ret = strings.TrimSpace(ret)
|
||
if "" == ret {
|
||
return req.RemoteAddr
|
||
}
|
||
return strings.Split(ret, ",")[0]
|
||
}
|
||
|
||
func JsonArg(c *gin.Context, result *gulu.Result) (arg map[string]interface{}, ok bool) {
|
||
arg = map[string]interface{}{}
|
||
if err := c.BindJSON(&arg); err != nil {
|
||
result.Code = -1
|
||
result.Msg = "parses request failed"
|
||
return
|
||
}
|
||
|
||
ok = true
|
||
return
|
||
}
|
||
|
||
// ParseJsonArg 使用泛型从 JSON 参数中提取指定键的值。
|
||
// - 如果 required 为 true 但参数缺失,则会在 ret.Msg 中写入 “[key] is required”
|
||
// - 如果参数存在但类型不匹配,则会在 ret.Msg 中写入 “[key] should be [T]”
|
||
// - 返回值 ok 为 false 时,表示提取失败或类型不匹配
|
||
func ParseJsonArg[T any](key string, required bool, arg map[string]interface{}, ret *gulu.Result) (value T, ok bool) {
|
||
raw, exists := arg[key]
|
||
if !exists || raw == nil {
|
||
if required {
|
||
ret.Code = -1
|
||
ret.Msg = key + " is required"
|
||
} else {
|
||
ok = true
|
||
}
|
||
return
|
||
}
|
||
|
||
value, ok = raw.(T)
|
||
if !ok {
|
||
var zero T
|
||
ret.Code = -1
|
||
ret.Msg = fmt.Sprintf("%s should be %T", key, zero)
|
||
}
|
||
return
|
||
}
|
||
|
||
// JsonArgParseFunc 为单次提取函数,用于 ParseJsonArgs 批量提取。
|
||
type JsonArgParseFunc func(arg map[string]interface{}, ret *gulu.Result) bool
|
||
|
||
// BindJsonArg 创建一个提取函数:从 arg 取 key 并写入 dest,供 ParseJsonArgs 使用。
|
||
func BindJsonArg[T any](key string, required bool, dest *T) JsonArgParseFunc {
|
||
return func(arg map[string]interface{}, ret *gulu.Result) bool {
|
||
v, ok := ParseJsonArg[T](key, required, arg, ret)
|
||
if !ok {
|
||
return false
|
||
}
|
||
*dest = v
|
||
return true
|
||
}
|
||
}
|
||
|
||
// ParseJsonArgs 按顺序执行多个提取函数。
|
||
// - 任一失败返回 false 并在 ret 中写入错误信息
|
||
// - 全部成功返回 true
|
||
func ParseJsonArgs(arg map[string]interface{}, ret *gulu.Result, extractors ...JsonArgParseFunc) bool {
|
||
for _, ext := range extractors {
|
||
if !ext(arg, ret) {
|
||
return false
|
||
}
|
||
}
|
||
return true
|
||
}
|
||
|
||
func InvalidIDPattern(idArg string, result *gulu.Result) bool {
|
||
if ast.IsNodeIDPattern(idArg) {
|
||
return false
|
||
}
|
||
|
||
result.Code = -1
|
||
result.Msg = "invalid ID argument"
|
||
return true
|
||
}
|
||
|
||
func initHttpClient() {
|
||
http.DefaultClient = httpclient.GetCloudFileClient2Min()
|
||
http.DefaultTransport = httpclient.NewTransport(false)
|
||
}
|
||
|
||
func ParsePort(portString string) (uint16, error) {
|
||
port, err := strconv.ParseUint(portString, 10, 16)
|
||
if err != nil {
|
||
logging.LogErrorf("parse port [%s] failed: %s", portString, err)
|
||
return 0, err
|
||
}
|
||
return uint16(port), nil
|
||
}
|