mirror of
				https://github.com/bestnite/sub2clash.git
				synced 2025-10-26 17:14:24 +00:00 
			
		
		
		
	feat: 修改短链
update: dockerfile,workflow
This commit is contained in:
		
							
								
								
									
										43
									
								
								api/handler/clash.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								api/handler/clash.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| package handler | ||||
|  | ||||
| import ( | ||||
| 	"net/http" | ||||
| 	"sub2clash/config" | ||||
| 	"sub2clash/model" | ||||
| 	"sub2clash/validator" | ||||
|  | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"gopkg.in/yaml.v3" | ||||
| ) | ||||
|  | ||||
| func SubmodHandler(c *gin.Context) { | ||||
| 	// 从请求中获取参数 | ||||
| 	query, err := validator.ParseQuery(c) | ||||
| 	if err != nil { | ||||
| 		c.String(http.StatusBadRequest, err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	sub, err := BuildSub(model.Clash, query, config.Default.ClashTemplate) | ||||
| 	if err != nil { | ||||
| 		c.String(http.StatusInternalServerError, err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	// 输出 | ||||
| 	if query.NodeListMode { | ||||
| 		nodelist := model.NodeList{} | ||||
| 		nodelist.Proxies = sub.Proxies | ||||
| 		marshal, err := yaml.Marshal(nodelist) | ||||
| 		if err != nil { | ||||
| 			c.String(http.StatusInternalServerError, "YAML序列化失败: "+err.Error()) | ||||
| 			return | ||||
| 		} | ||||
| 		c.String(http.StatusOK, string(marshal)) | ||||
| 		return | ||||
| 	} | ||||
| 	marshal, err := yaml.Marshal(sub) | ||||
| 	if err != nil { | ||||
| 		c.String(http.StatusInternalServerError, "YAML序列化失败: "+err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	c.String(http.StatusOK, string(marshal)) | ||||
| } | ||||
							
								
								
									
										286
									
								
								api/handler/default.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										286
									
								
								api/handler/default.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,286 @@ | ||||
| package handler | ||||
|  | ||||
| import ( | ||||
| 	"crypto/sha256" | ||||
| 	"encoding/hex" | ||||
| 	"errors" | ||||
| 	"net/url" | ||||
| 	"regexp" | ||||
| 	"sort" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sub2clash/logger" | ||||
| 	"sub2clash/model" | ||||
| 	"sub2clash/parser" | ||||
| 	"sub2clash/utils" | ||||
| 	"sub2clash/validator" | ||||
|  | ||||
| 	"go.uber.org/zap" | ||||
| 	"gopkg.in/yaml.v3" | ||||
| ) | ||||
|  | ||||
| func BuildSub(clashType model.ClashType, query validator.SubValidator, template string) ( | ||||
| 	*model.Subscription, error, | ||||
| ) { | ||||
| 	// 定义变量 | ||||
| 	var temp = &model.Subscription{} | ||||
| 	var sub = &model.Subscription{} | ||||
| 	var err error | ||||
| 	var templateBytes []byte | ||||
| 	// 加载模板 | ||||
| 	if query.Template != "" { | ||||
| 		template = query.Template | ||||
| 	} | ||||
| 	if strings.HasPrefix(template, "http") { | ||||
| 		templateBytes, err = utils.LoadSubscription(template, query.Refresh) | ||||
| 		if err != nil { | ||||
| 			logger.Logger.Debug( | ||||
| 				"load template failed", zap.String("template", template), zap.Error(err), | ||||
| 			) | ||||
| 			return nil, errors.New("加载模板失败: " + err.Error()) | ||||
| 		} | ||||
| 	} else { | ||||
| 		unescape, err := url.QueryUnescape(template) | ||||
| 		if err != nil { | ||||
| 			return nil, errors.New("加载模板失败: " + err.Error()) | ||||
| 		} | ||||
| 		templateBytes, err = utils.LoadTemplate(unescape) | ||||
| 		if err != nil { | ||||
| 			logger.Logger.Debug( | ||||
| 				"load template failed", zap.String("template", template), zap.Error(err), | ||||
| 			) | ||||
| 			return nil, errors.New("加载模板失败: " + err.Error()) | ||||
| 		} | ||||
| 	} | ||||
| 	// 解析模板 | ||||
| 	err = yaml.Unmarshal(templateBytes, &temp) | ||||
| 	if err != nil { | ||||
| 		logger.Logger.Debug("parse template failed", zap.Error(err)) | ||||
| 		return nil, errors.New("解析模板失败: " + err.Error()) | ||||
| 	} | ||||
| 	var proxyList []model.Proxy | ||||
| 	// 加载订阅 | ||||
| 	for i := range query.Subs { | ||||
| 		data, err := utils.LoadSubscription(query.Subs[i], query.Refresh) | ||||
| 		subName := "" | ||||
| 		if strings.Contains(query.Subs[i], "#") { | ||||
| 			subName = query.Subs[i][strings.LastIndex(query.Subs[i], "#")+1:] | ||||
| 		} | ||||
| 		if err != nil { | ||||
| 			logger.Logger.Debug( | ||||
| 				"load subscription failed", zap.String("url", query.Subs[i]), zap.Error(err), | ||||
| 			) | ||||
| 			return nil, errors.New("加载订阅失败: " + err.Error()) | ||||
| 		} | ||||
| 		// 解析订阅 | ||||
| 		err = yaml.Unmarshal(data, &sub) | ||||
| 		var newProxies []model.Proxy | ||||
| 		if err != nil { | ||||
| 			reg, _ := regexp.Compile("(ssr|ss|vmess|trojan|vless|hysteria)://") | ||||
| 			if reg.Match(data) { | ||||
| 				p := utils.ParseProxy(strings.Split(string(data), "\n")...) | ||||
| 				newProxies = p | ||||
| 			} else { | ||||
| 				// 如果无法直接解析,尝试Base64解码 | ||||
| 				base64, err := parser.DecodeBase64(string(data)) | ||||
| 				if err != nil { | ||||
| 					logger.Logger.Debug( | ||||
| 						"parse subscription failed", zap.String("url", query.Subs[i]), | ||||
| 						zap.String("data", string(data)), | ||||
| 						zap.Error(err), | ||||
| 					) | ||||
| 					return nil, errors.New("加载订阅失败: " + err.Error()) | ||||
| 				} | ||||
| 				p := utils.ParseProxy(strings.Split(base64, "\n")...) | ||||
| 				newProxies = p | ||||
| 			} | ||||
| 		} else { | ||||
| 			newProxies = sub.Proxies | ||||
| 		} | ||||
| 		if subName != "" { | ||||
| 			for i := range newProxies { | ||||
| 				newProxies[i].SubName = subName | ||||
| 			} | ||||
| 		} | ||||
| 		proxyList = append(proxyList, newProxies...) | ||||
| 	} | ||||
| 	// 添加自定义节点 | ||||
| 	if len(query.Proxies) != 0 { | ||||
| 		proxyList = append(proxyList, utils.ParseProxy(query.Proxies...)...) | ||||
| 	} | ||||
| 	// 给节点添加订阅名称 | ||||
| 	for i := range proxyList { | ||||
| 		if proxyList[i].SubName != "" { | ||||
| 			proxyList[i].Name = strings.TrimSpace(proxyList[i].SubName) + " " + strings.TrimSpace(proxyList[i].Name) | ||||
| 		} | ||||
| 	} | ||||
| 	// 去掉配置相同的节点 | ||||
| 	proxies := make(map[string]*model.Proxy) | ||||
| 	newProxies := make([]model.Proxy, 0, len(proxyList)) | ||||
| 	for i := range proxyList { | ||||
| 		key := proxyList[i].Server + ":" + strconv.Itoa(proxyList[i].Port) + ":" + proxyList[i].Type | ||||
| 		if _, exist := proxies[key]; !exist { | ||||
| 			proxies[key] = &proxyList[i] | ||||
| 			newProxies = append(newProxies, proxyList[i]) | ||||
| 		} | ||||
| 	} | ||||
| 	proxyList = newProxies | ||||
| 	// 删除节点 | ||||
| 	if strings.TrimSpace(query.Remove) != "" { | ||||
| 		newProxyList := make([]model.Proxy, 0, len(proxyList)) | ||||
| 		for i := range proxyList { | ||||
| 			removeReg, err := regexp.Compile(query.Remove) | ||||
| 			if err != nil { | ||||
| 				logger.Logger.Debug("remove regexp compile failed", zap.Error(err)) | ||||
| 				return nil, errors.New("remove 参数非法: " + err.Error()) | ||||
| 			} | ||||
| 			// 删除匹配到的节点 | ||||
| 			if removeReg.MatchString(proxyList[i].Name) { | ||||
| 				continue // 如果匹配到要删除的元素,跳过该元素,不添加到新切片中 | ||||
| 			} | ||||
| 			newProxyList = append(newProxyList, proxyList[i]) // 将要保留的元素添加到新切片中 | ||||
| 		} | ||||
| 		proxyList = newProxyList | ||||
| 	} | ||||
| 	// 重命名 | ||||
| 	if len(query.ReplaceKeys) != 0 { | ||||
| 		// 创建重命名正则表达式 | ||||
| 		replaceRegs := make([]*regexp.Regexp, 0, len(query.ReplaceKeys)) | ||||
| 		for _, v := range query.ReplaceKeys { | ||||
| 			replaceReg, err := regexp.Compile(v) | ||||
| 			if err != nil { | ||||
| 				logger.Logger.Debug("replace regexp compile failed", zap.Error(err)) | ||||
| 				return nil, errors.New("replace 参数非法: " + err.Error()) | ||||
| 			} | ||||
| 			replaceRegs = append(replaceRegs, replaceReg) | ||||
| 		} | ||||
| 		for i := range proxyList { | ||||
| 			// 重命名匹配到的节点 | ||||
| 			for j, v := range replaceRegs { | ||||
| 				if v.MatchString(proxyList[i].Name) { | ||||
| 					proxyList[i].Name = v.ReplaceAllString( | ||||
| 						proxyList[i].Name, query.ReplaceTo[j], | ||||
| 					) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	// 重名检测 | ||||
| 	names := make(map[string]int) | ||||
| 	for i := range proxyList { | ||||
| 		if _, exist := names[proxyList[i].Name]; exist { | ||||
| 			names[proxyList[i].Name] = names[proxyList[i].Name] + 1 | ||||
| 			proxyList[i].Name = proxyList[i].Name + " " + strconv.Itoa(names[proxyList[i].Name]) | ||||
| 		} else { | ||||
| 			names[proxyList[i].Name] = 0 | ||||
| 		} | ||||
| 	} | ||||
| 	// trim | ||||
| 	for i := range proxyList { | ||||
| 		proxyList[i].Name = strings.TrimSpace(proxyList[i].Name) | ||||
| 	} | ||||
| 	// 将新增节点都添加到临时变量 t 中,防止策略组排序错乱 | ||||
| 	var t = &model.Subscription{} | ||||
| 	utils.AddProxy(t, query.AutoTest, query.Lazy, clashType, proxyList...) | ||||
| 	// 排序策略组 | ||||
| 	switch query.Sort { | ||||
| 	case "sizeasc": | ||||
| 		sort.Sort(model.ProxyGroupsSortBySize(t.ProxyGroups)) | ||||
| 	case "sizedesc": | ||||
| 		sort.Sort(sort.Reverse(model.ProxyGroupsSortBySize(t.ProxyGroups))) | ||||
| 	case "nameasc": | ||||
| 		sort.Sort(model.ProxyGroupsSortByName(t.ProxyGroups)) | ||||
| 	case "namedesc": | ||||
| 		sort.Sort(sort.Reverse(model.ProxyGroupsSortByName(t.ProxyGroups))) | ||||
| 	default: | ||||
| 		sort.Sort(model.ProxyGroupsSortByName(t.ProxyGroups)) | ||||
| 	} | ||||
| 	// 合并新节点和模板 | ||||
| 	MergeSubAndTemplate(temp, t, query.IgnoreCountryGrooup) | ||||
| 	// 处理自定义规则 | ||||
| 	for _, v := range query.Rules { | ||||
| 		if v.Prepend { | ||||
| 			utils.PrependRules(temp, v.Rule) | ||||
| 		} else { | ||||
| 			utils.AppendRules(temp, v.Rule) | ||||
| 		} | ||||
| 	} | ||||
| 	// 处理自定义 ruleProvider | ||||
| 	for _, v := range query.RuleProviders { | ||||
| 		hash := sha256.Sum224([]byte(v.Url)) | ||||
| 		name := hex.EncodeToString(hash[:]) | ||||
| 		provider := model.RuleProvider{ | ||||
| 			Type:     "http", | ||||
| 			Behavior: v.Behavior, | ||||
| 			Url:      v.Url, | ||||
| 			Path:     "./" + name + ".yaml", | ||||
| 			Interval: 3600, | ||||
| 		} | ||||
| 		if v.Prepend { | ||||
| 			utils.PrependRuleProvider( | ||||
| 				temp, v.Name, v.Group, provider, | ||||
| 			) | ||||
| 		} else { | ||||
| 			utils.AppenddRuleProvider( | ||||
| 				temp, v.Name, v.Group, provider, | ||||
| 			) | ||||
| 		} | ||||
| 	} | ||||
| 	return temp, nil | ||||
| } | ||||
|  | ||||
| func MergeSubAndTemplate(temp *model.Subscription, sub *model.Subscription, igcg bool) { | ||||
| 	// 只合并节点、策略组 | ||||
| 	// 统计所有国家策略组名称 | ||||
| 	var countryGroupNames []string | ||||
| 	for _, proxyGroup := range sub.ProxyGroups { | ||||
| 		if proxyGroup.IsCountryGrop { | ||||
| 			countryGroupNames = append( | ||||
| 				countryGroupNames, proxyGroup.Name, | ||||
| 			) | ||||
| 		} | ||||
| 	} | ||||
| 	var proxyNames []string | ||||
| 	for _, proxy := range sub.Proxies { | ||||
| 		proxyNames = append(proxyNames, proxy.Name) | ||||
| 	} | ||||
| 	// 将订阅中的节点添加到模板中 | ||||
| 	temp.Proxies = append(temp.Proxies, sub.Proxies...) | ||||
| 	// 将订阅中的策略组添加到模板中 | ||||
| 	for i := range temp.ProxyGroups { | ||||
| 		if temp.ProxyGroups[i].IsCountryGrop { | ||||
| 			continue | ||||
| 		} | ||||
| 		newProxies := make([]string, 0) | ||||
| 		countryGroupMap := make(map[string]model.ProxyGroup) | ||||
| 		for _, v := range sub.ProxyGroups { | ||||
| 			if v.IsCountryGrop { | ||||
| 				countryGroupMap[v.Name] = v | ||||
| 			} | ||||
| 		} | ||||
| 		for j := range temp.ProxyGroups[i].Proxies { | ||||
| 			reg := regexp.MustCompile("<(.*?)>") | ||||
| 			if reg.Match([]byte(temp.ProxyGroups[i].Proxies[j])) { | ||||
| 				key := reg.FindStringSubmatch(temp.ProxyGroups[i].Proxies[j])[1] | ||||
| 				switch key { | ||||
| 				case "all": | ||||
| 					newProxies = append(newProxies, proxyNames...) | ||||
| 				case "countries": | ||||
| 					newProxies = append(newProxies, countryGroupNames...) | ||||
| 				default: | ||||
| 					if len(key) == 2 { | ||||
| 						newProxies = append( | ||||
| 							newProxies, countryGroupMap[utils.GetContryName(key)].Proxies..., | ||||
| 						) | ||||
| 					} | ||||
| 				} | ||||
| 			} else { | ||||
| 				newProxies = append(newProxies, temp.ProxyGroups[i].Proxies[j]) | ||||
| 			} | ||||
| 		} | ||||
| 		temp.ProxyGroups[i].Proxies = newProxies | ||||
| 	} | ||||
| 	if !igcg { | ||||
| 		temp.ProxyGroups = append(temp.ProxyGroups, sub.ProxyGroups...) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										44
									
								
								api/handler/meta.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								api/handler/meta.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| package handler | ||||
|  | ||||
| import ( | ||||
| 	_ "embed" | ||||
| 	"net/http" | ||||
| 	"sub2clash/config" | ||||
| 	"sub2clash/model" | ||||
| 	"sub2clash/validator" | ||||
|  | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"gopkg.in/yaml.v3" | ||||
| ) | ||||
|  | ||||
| func SubHandler(c *gin.Context) { | ||||
| 	// 从请求中获取参数 | ||||
| 	query, err := validator.ParseQuery(c) | ||||
| 	if err != nil { | ||||
| 		c.String(http.StatusBadRequest, err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	sub, err := BuildSub(model.ClashMeta, query, config.Default.MetaTemplate) | ||||
| 	if err != nil { | ||||
| 		c.String(http.StatusInternalServerError, err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	// 输出 | ||||
| 	if query.NodeListMode { | ||||
| 		nodelist := model.NodeList{} | ||||
| 		nodelist.Proxies = sub.Proxies | ||||
| 		marshal, err := yaml.Marshal(nodelist) | ||||
| 		if err != nil { | ||||
| 			c.String(http.StatusInternalServerError, "YAML序列化失败: "+err.Error()) | ||||
| 			return | ||||
| 		} | ||||
| 		c.String(http.StatusOK, string(marshal)) | ||||
| 		return | ||||
| 	} | ||||
| 	marshal, err := yaml.Marshal(sub) | ||||
| 	if err != nil { | ||||
| 		c.String(http.StatusInternalServerError, "YAML序列化失败: "+err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	c.String(http.StatusOK, string(marshal)) | ||||
| } | ||||
							
								
								
									
										137
									
								
								api/handler/short_link.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										137
									
								
								api/handler/short_link.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,137 @@ | ||||
| package handler | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sub2clash/config" | ||||
| 	"sub2clash/logger" | ||||
| 	"sub2clash/model" | ||||
| 	"sub2clash/utils" | ||||
| 	"sub2clash/utils/database" | ||||
| 	"sub2clash/validator" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/gin-gonic/gin" | ||||
| 	"go.uber.org/zap" | ||||
| 	"gorm.io/gorm" | ||||
| ) | ||||
|  | ||||
| func ShortLinkGenHandler(c *gin.Context) { | ||||
| 	// 从请求中获取参数 | ||||
| 	var params validator.ShortLinkGenValidator | ||||
| 	if err := c.ShouldBind(¶ms); err != nil { | ||||
| 		c.String(400, "参数错误: "+err.Error()) | ||||
| 	} | ||||
| 	if strings.TrimSpace(params.Url) == "" { | ||||
| 		c.String(400, "参数错误") | ||||
| 		return | ||||
| 	} | ||||
| 	// 生成hash | ||||
| 	hash := utils.RandomString(config.Default.ShortLinkLength) | ||||
| 	var item model.ShortLink | ||||
| 	result := database.FindShortLinkByUrl(params.Url, &item) | ||||
| 	if result.Error == nil { | ||||
| 		if item.Password != params.Password { | ||||
| 			item.Password = params.Password | ||||
| 			database.SaveShortLink(&item) | ||||
| 			c.String(200, item.Hash+"?password="+params.Password) | ||||
| 		} else { | ||||
| 			c.String(200, item.Hash) | ||||
| 		} | ||||
| 		return | ||||
| 	} else { | ||||
| 		if !errors.Is(result.Error, gorm.ErrRecordNotFound) { | ||||
| 			c.String(500, "数据库错误: "+result.Error.Error()) | ||||
| 			return | ||||
| 		} | ||||
| 	} | ||||
| 	// 如果记录存在则重新生成hash,直到记录不存在 | ||||
| 	result = database.FindShortLinkByHash(hash, &item) | ||||
| 	for result.Error == nil { | ||||
| 		hash = utils.RandomString(config.Default.ShortLinkLength) | ||||
| 		result = database.FindShortLinkByHash(hash, &item) | ||||
| 	} | ||||
| 	// 创建记录 | ||||
| 	database.FirstOrCreateShortLink( | ||||
| 		&model.ShortLink{ | ||||
| 			Hash:            hash, | ||||
| 			Url:             params.Url, | ||||
| 			LastRequestTime: -1, | ||||
| 			Password:        params.Password, | ||||
| 		}, | ||||
| 	) | ||||
| 	// 返回短链接 | ||||
| 	if params.Password != "" { | ||||
| 		hash += "?password=" + params.Password | ||||
| 	} | ||||
| 	c.String(200, hash) | ||||
| } | ||||
|  | ||||
| func ShortLinkGetHandler(c *gin.Context) { | ||||
| 	// 获取动态路由 | ||||
| 	hash := c.Param("hash") | ||||
| 	password := c.Query("password") | ||||
| 	if strings.TrimSpace(hash) == "" { | ||||
| 		c.String(400, "参数错误") | ||||
| 		return | ||||
| 	} | ||||
| 	// 查询数据库 | ||||
| 	var shortLink model.ShortLink | ||||
| 	result := database.FindShortLinkByHash(hash, &shortLink) | ||||
| 	// 重定向 | ||||
| 	if result.Error != nil { | ||||
| 		c.String(404, "未找到短链接或密码错误") | ||||
| 		return | ||||
| 	} | ||||
| 	if shortLink.Password != "" && shortLink.Password != password { | ||||
| 		c.String(404, "未找到短链接或密码错误") | ||||
| 		return | ||||
| 	} | ||||
| 	// 更新最后访问时间 | ||||
| 	shortLink.LastRequestTime = time.Now().Unix() | ||||
| 	database.SaveShortLink(&shortLink) | ||||
| 	get, err := utils.Get("http://localhost:" + strconv.Itoa(config.Default.Port) + "/" + shortLink.Url) | ||||
| 	if err != nil { | ||||
| 		logger.Logger.Debug("get short link data failed", zap.Error(err)) | ||||
| 		c.String(500, "请求错误: "+err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	all, err := io.ReadAll(get.Body) | ||||
| 	if err != nil { | ||||
| 		logger.Logger.Debug("read short link data failed", zap.Error(err)) | ||||
| 		c.String(500, "读取错误: "+err.Error()) | ||||
| 		return | ||||
| 	} | ||||
| 	c.String(http.StatusOK, string(all)) | ||||
| } | ||||
|  | ||||
| func ShortLinkUpdateHandler(c *gin.Context) { | ||||
| 	var params validator.ShortLinkUpdateValidator | ||||
| 	if err := c.ShouldBind(¶ms); err != nil { | ||||
| 		c.String(400, "参数错误: "+err.Error()) | ||||
| 	} | ||||
| 	if strings.TrimSpace(params.Url) == "" { | ||||
| 		c.String(400, "参数错误") | ||||
| 		return | ||||
| 	} | ||||
| 	var shortLink model.ShortLink | ||||
| 	result := database.FindShortLinkByHash(params.Hash, &shortLink) | ||||
| 	if result.Error != nil { | ||||
| 		c.String(404, "未找到短链接") | ||||
| 		return | ||||
| 	} | ||||
| 	if shortLink.Password == "" { | ||||
| 		c.String(403, "无法修改无密码短链接") | ||||
| 		return | ||||
| 	} | ||||
| 	if shortLink.Password != params.Password { | ||||
| 		c.String(403, "密码错误") | ||||
| 		return | ||||
| 	} | ||||
| 	shortLink.Url = params.Url | ||||
| 	database.SaveShortLink(&shortLink) | ||||
| 	c.String(200, "更新成功") | ||||
| } | ||||
		Reference in New Issue
	
	Block a user