mirror of
https://github.com/TeaOSLab/EdgeAdmin.git
synced 2025-11-03 20:40:26 +08:00
简化IP名单中创建IP操作/支持IP以CIDR方式显示
This commit is contained in:
@@ -3,11 +3,66 @@ package utils
|
|||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"errors"
|
"errors"
|
||||||
|
"github.com/TeaOSLab/EdgeCommon/pkg/iputils"
|
||||||
"github.com/iwind/TeaGo/types"
|
"github.com/iwind/TeaGo/types"
|
||||||
"net"
|
"net"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// ParseIPValue 解析IP值
|
||||||
|
func ParseIPValue(value string) (newValue string, ipFrom string, ipTo string, ok bool) {
|
||||||
|
if len(value) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
newValue = value
|
||||||
|
|
||||||
|
// ip1-ip2
|
||||||
|
if strings.Contains(value, "-") {
|
||||||
|
var pieces = strings.Split(value, "-")
|
||||||
|
if len(pieces) != 2 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ipFrom = strings.TrimSpace(pieces[0])
|
||||||
|
ipTo = strings.TrimSpace(pieces[1])
|
||||||
|
|
||||||
|
if !iputils.IsValid(ipFrom) || !iputils.IsValid(ipTo) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if !iputils.IsSameVersion(ipFrom, ipTo) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if iputils.CompareIP(ipFrom, ipTo) > 0 {
|
||||||
|
ipFrom, ipTo = ipTo, ipFrom
|
||||||
|
newValue = ipFrom + "-" + ipTo
|
||||||
|
}
|
||||||
|
|
||||||
|
ok = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// ip/mask
|
||||||
|
if strings.Contains(value, "/") {
|
||||||
|
cidr, err := iputils.ParseCIDR(value)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
return newValue, cidr.From().String(), cidr.To().String(), true
|
||||||
|
}
|
||||||
|
|
||||||
|
// single value
|
||||||
|
if iputils.IsValid(value) {
|
||||||
|
ipFrom = value
|
||||||
|
ok = true
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// ExtractIP 分解IP
|
// ExtractIP 分解IP
|
||||||
// 只支持D段掩码的CIDR
|
// 只支持D段掩码的CIDR
|
||||||
// 最多只记录255个值
|
// 最多只记录255个值
|
||||||
|
|||||||
@@ -99,6 +99,7 @@ func (this *ListsAction) RunGet(params struct {
|
|||||||
|
|
||||||
itemMaps = append(itemMaps, maps.Map{
|
itemMaps = append(itemMaps, maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
||||||
|
|||||||
@@ -57,6 +57,7 @@ func (this *TestAction) RunPost(params struct {
|
|||||||
if resp.IpItem != nil {
|
if resp.IpItem != nil {
|
||||||
resultMap["item"] = maps.Map{
|
resultMap["item"] = maps.Map{
|
||||||
"id": resp.IpItem.Id,
|
"id": resp.IpItem.Id,
|
||||||
|
"value": resp.IpItem.Value,
|
||||||
"ipFrom": resp.IpItem.IpFrom,
|
"ipFrom": resp.IpItem.IpFrom,
|
||||||
"ipTo": resp.IpItem.IpTo,
|
"ipTo": resp.IpItem.IpTo,
|
||||||
"reason": resp.IpItem.Reason,
|
"reason": resp.IpItem.Reason,
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
package ipadmin
|
package ipadmin
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/TeaOSLab/EdgeAdmin/internal/utils"
|
||||||
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/iputils"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
||||||
"github.com/iwind/TeaGo/actions"
|
"github.com/iwind/TeaGo/actions"
|
||||||
@@ -33,6 +33,7 @@ func (this *UpdateIPPopupAction) RunGet(params struct {
|
|||||||
|
|
||||||
this.Data["item"] = maps.Map{
|
this.Data["item"] = maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"expiredAt": item.ExpiredAt,
|
"expiredAt": item.ExpiredAt,
|
||||||
@@ -50,8 +51,7 @@ func (this *UpdateIPPopupAction) RunPost(params struct {
|
|||||||
FirewallPolicyId int64
|
FirewallPolicyId int64
|
||||||
ItemId int64
|
ItemId int64
|
||||||
|
|
||||||
IpFrom string
|
Value string
|
||||||
IpTo string
|
|
||||||
ExpiredAt int64
|
ExpiredAt int64
|
||||||
Reason string
|
Reason string
|
||||||
Type string
|
Type string
|
||||||
@@ -63,50 +63,25 @@ func (this *UpdateIPPopupAction) RunPost(params struct {
|
|||||||
// 日志
|
// 日志
|
||||||
defer this.CreateLogInfo(codes.WAF_LogUpdateIPFromWAFPolicy, params.FirewallPolicyId, params.ItemId)
|
defer this.CreateLogInfo(codes.WAF_LogUpdateIPFromWAFPolicy, params.FirewallPolicyId, params.ItemId)
|
||||||
|
|
||||||
// TODO 校验ItemId所属用户
|
|
||||||
|
|
||||||
switch params.Type {
|
switch params.Type {
|
||||||
case "ipv4":
|
case "ip":
|
||||||
|
// 校验IP格式
|
||||||
params.Must.
|
params.Must.
|
||||||
Field("ipFrom", params.IpFrom).
|
Field("value", params.Value).
|
||||||
Require("请输入开始IP")
|
Require("请输入IP或IP段")
|
||||||
|
|
||||||
// 校验IP格式(ipFrom/ipTo)
|
_, _, _, ok := utils.ParseIPValue(params.Value)
|
||||||
if !iputils.IsIPv4(params.IpFrom) {
|
if !ok {
|
||||||
this.FailField("ipFrom", "请输入正确的开始IP")
|
this.FailField("value", "请输入正确的IP格式")
|
||||||
}
|
return
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv4(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的结束IP")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
|
||||||
case "ipv6":
|
|
||||||
params.Must.
|
|
||||||
Field("ipFrom", params.IpFrom).
|
|
||||||
Require("请输入正确的开始IP")
|
|
||||||
|
|
||||||
if !iputils.IsIPv6(params.IpFrom) {
|
|
||||||
this.FailField("ipFrom", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv6(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
}
|
||||||
case "all":
|
case "all":
|
||||||
params.IpFrom = "0.0.0.0"
|
params.Value = "0.0.0.0"
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err := this.RPC().IPItemRPC().UpdateIPItem(this.AdminContext(), &pb.UpdateIPItemRequest{
|
_, err := this.RPC().IPItemRPC().UpdateIPItem(this.AdminContext(), &pb.UpdateIPItemRequest{
|
||||||
IpItemId: params.ItemId,
|
IpItemId: params.ItemId,
|
||||||
IpFrom: params.IpFrom,
|
Value: params.Value,
|
||||||
IpTo: params.IpTo,
|
|
||||||
ExpiredAt: params.ExpiredAt,
|
ExpiredAt: params.ExpiredAt,
|
||||||
Reason: params.Reason,
|
Reason: params.Reason,
|
||||||
Type: params.Type,
|
Type: params.Type,
|
||||||
|
|||||||
@@ -109,6 +109,7 @@ func (this *AllowListAction) RunGet(params struct {
|
|||||||
|
|
||||||
itemMaps = append(itemMaps, maps.Map{
|
itemMaps = append(itemMaps, maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
||||||
|
|||||||
@@ -109,6 +109,7 @@ func (this *DenyListAction) RunGet(params struct {
|
|||||||
|
|
||||||
itemMaps = append(itemMaps, maps.Map{
|
itemMaps = append(itemMaps, maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
||||||
|
|||||||
@@ -70,6 +70,7 @@ func (this *TestAction) RunPost(params struct {
|
|||||||
if resp.IpItem != nil {
|
if resp.IpItem != nil {
|
||||||
resultMap["item"] = maps.Map{
|
resultMap["item"] = maps.Map{
|
||||||
"id": resp.IpItem.Id,
|
"id": resp.IpItem.Id,
|
||||||
|
"value": resp.IpItem.Value,
|
||||||
"ipFrom": resp.IpItem.IpFrom,
|
"ipFrom": resp.IpItem.IpFrom,
|
||||||
"ipTo": resp.IpItem.IpTo,
|
"ipTo": resp.IpItem.IpTo,
|
||||||
"reason": resp.IpItem.Reason,
|
"reason": resp.IpItem.Reason,
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
package ipadmin
|
package ipadmin
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/TeaOSLab/EdgeAdmin/internal/utils"
|
||||||
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/iputils"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
||||||
"github.com/iwind/TeaGo/actions"
|
"github.com/iwind/TeaGo/actions"
|
||||||
@@ -33,6 +33,7 @@ func (this *UpdateIPPopupAction) RunGet(params struct {
|
|||||||
|
|
||||||
this.Data["item"] = maps.Map{
|
this.Data["item"] = maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"expiredAt": item.ExpiredAt,
|
"expiredAt": item.ExpiredAt,
|
||||||
@@ -49,8 +50,7 @@ func (this *UpdateIPPopupAction) RunGet(params struct {
|
|||||||
func (this *UpdateIPPopupAction) RunPost(params struct {
|
func (this *UpdateIPPopupAction) RunPost(params struct {
|
||||||
ItemId int64
|
ItemId int64
|
||||||
|
|
||||||
IpFrom string
|
Value string
|
||||||
IpTo string
|
|
||||||
ExpiredAt int64
|
ExpiredAt int64
|
||||||
Reason string
|
Reason string
|
||||||
Type string
|
Type string
|
||||||
@@ -62,50 +62,25 @@ func (this *UpdateIPPopupAction) RunPost(params struct {
|
|||||||
// 日志
|
// 日志
|
||||||
defer this.CreateLogInfo(codes.IPItem_LogUpdateIPItem, params.ItemId)
|
defer this.CreateLogInfo(codes.IPItem_LogUpdateIPItem, params.ItemId)
|
||||||
|
|
||||||
// TODO 校验ItemId所属用户
|
|
||||||
|
|
||||||
switch params.Type {
|
switch params.Type {
|
||||||
case "ipv4":
|
case "ip":
|
||||||
|
// 校验IP格式
|
||||||
params.Must.
|
params.Must.
|
||||||
Field("ipFrom", params.IpFrom).
|
Field("value", params.Value).
|
||||||
Require("请输入开始IP")
|
Require("请输入IP或IP段")
|
||||||
|
|
||||||
// 校验IP格式(ipFrom/ipTo)
|
_, _, _, ok := utils.ParseIPValue(params.Value)
|
||||||
if !iputils.IsIPv4(params.IpFrom) {
|
if !ok {
|
||||||
this.FailField("ipFrom", "请输入正确的开始IP")
|
this.FailField("value", "请输入正确的IP格式")
|
||||||
}
|
return
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv4(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的结束IP")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
|
||||||
case "ipv6":
|
|
||||||
params.Must.
|
|
||||||
Field("ipFrom", params.IpFrom).
|
|
||||||
Require("请输入正确的开始IP")
|
|
||||||
|
|
||||||
if !iputils.IsIPv6(params.IpFrom) {
|
|
||||||
this.FailField("ipFrom", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv6(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
}
|
||||||
case "all":
|
case "all":
|
||||||
params.IpFrom = "0.0.0.0"
|
params.Value = "0.0.0.0"
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err := this.RPC().IPItemRPC().UpdateIPItem(this.AdminContext(), &pb.UpdateIPItemRequest{
|
_, err := this.RPC().IPItemRPC().UpdateIPItem(this.AdminContext(), &pb.UpdateIPItemRequest{
|
||||||
IpItemId: params.ItemId,
|
IpItemId: params.ItemId,
|
||||||
IpFrom: params.IpFrom,
|
Value: params.Value,
|
||||||
IpTo: params.IpTo,
|
|
||||||
ExpiredAt: params.ExpiredAt,
|
ExpiredAt: params.ExpiredAt,
|
||||||
Reason: params.Reason,
|
Reason: params.Reason,
|
||||||
Type: params.Type,
|
Type: params.Type,
|
||||||
|
|||||||
@@ -1,16 +1,14 @@
|
|||||||
package iplists
|
package iplists
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/TeaOSLab/EdgeAdmin/internal/utils"
|
||||||
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/configutils"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/iputils"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/serverconfigs/firewallconfigs"
|
"github.com/TeaOSLab/EdgeCommon/pkg/serverconfigs/firewallconfigs"
|
||||||
"github.com/iwind/TeaGo/actions"
|
"github.com/iwind/TeaGo/actions"
|
||||||
"github.com/iwind/TeaGo/maps"
|
"github.com/iwind/TeaGo/maps"
|
||||||
"github.com/iwind/TeaGo/types"
|
"github.com/iwind/TeaGo/types"
|
||||||
"net"
|
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -50,9 +48,7 @@ func (this *CreateIPPopupAction) RunPost(params struct {
|
|||||||
ListId int64
|
ListId int64
|
||||||
Method string
|
Method string
|
||||||
|
|
||||||
IpFrom string
|
Value string
|
||||||
IpTo string
|
|
||||||
|
|
||||||
IpData string
|
IpData string
|
||||||
|
|
||||||
ExpiredAt int64
|
ExpiredAt int64
|
||||||
@@ -76,29 +72,24 @@ func (this *CreateIPPopupAction) RunPost(params struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type ipData struct {
|
type ipData struct {
|
||||||
|
value string
|
||||||
ipFrom string
|
ipFrom string
|
||||||
ipTo string
|
ipTo string
|
||||||
}
|
}
|
||||||
|
|
||||||
var batchIPs = []*ipData{}
|
var batchIPs = []*ipData{}
|
||||||
switch params.Type {
|
switch params.Type {
|
||||||
case "ipv4":
|
case "ip":
|
||||||
if params.Method == "single" {
|
if params.Method == "single" {
|
||||||
// 校验IP格式(ipFrom/ipTo)
|
// 校验IP格式
|
||||||
params.Must.
|
params.Must.
|
||||||
Field("ipFrom", params.IpFrom).
|
Field("value", params.Value).
|
||||||
Require("请输入开始IP")
|
Require("请输入IP或IP段")
|
||||||
|
|
||||||
if !iputils.IsIPv4(params.IpFrom) {
|
_, _, _, ok := utils.ParseIPValue(params.Value)
|
||||||
this.FailField("ipFrom", "请输入正确的开始IP")
|
if !ok {
|
||||||
}
|
this.FailField("value", "请输入正确的IP格式")
|
||||||
|
return
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv4(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的结束IP")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) != 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
}
|
||||||
} else if params.Method == "batch" {
|
} else if params.Method == "batch" {
|
||||||
if len(params.IpData) == 0 {
|
if len(params.IpData) == 0 {
|
||||||
@@ -107,144 +98,30 @@ func (this *CreateIPPopupAction) RunPost(params struct {
|
|||||||
var lines = strings.Split(params.IpData, "\n")
|
var lines = strings.Split(params.IpData, "\n")
|
||||||
for index, line := range lines {
|
for index, line := range lines {
|
||||||
line = strings.TrimSpace(line)
|
line = strings.TrimSpace(line)
|
||||||
if strings.Contains(line, "/") { // CIDR
|
if len(line) == 0 {
|
||||||
if strings.Contains(line, ":") {
|
continue
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
ipFrom, ipTo, err := configutils.ParseCIDR(line)
|
|
||||||
if err != nil {
|
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
batchIPs = append(batchIPs, &ipData{
|
|
||||||
ipFrom: ipFrom,
|
|
||||||
ipTo: ipTo,
|
|
||||||
})
|
|
||||||
} else if strings.Contains(line, "-") { // IP Range
|
|
||||||
var pieces = strings.Split(line, "-")
|
|
||||||
var ipFrom = strings.TrimSpace(pieces[0])
|
|
||||||
var ipTo = strings.TrimSpace(pieces[1])
|
|
||||||
|
|
||||||
if net.ParseIP(ipFrom) == nil || net.ParseIP(ipTo) == nil || strings.Contains(ipFrom, ":") || strings.Contains(ipTo, ":") {
|
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
if len(ipTo) > 0 && iputils.CompareIP(ipFrom, ipTo) > 0 {
|
|
||||||
ipFrom, ipTo = ipTo, ipFrom
|
|
||||||
}
|
|
||||||
batchIPs = append(batchIPs, &ipData{
|
|
||||||
ipFrom: ipFrom,
|
|
||||||
ipTo: ipTo,
|
|
||||||
})
|
|
||||||
} else if strings.Contains(line, ",") { // IP Range
|
|
||||||
var pieces = strings.Split(line, ",")
|
|
||||||
var ipFrom = strings.TrimSpace(pieces[0])
|
|
||||||
var ipTo = strings.TrimSpace(pieces[1])
|
|
||||||
|
|
||||||
if net.ParseIP(ipFrom) == nil || net.ParseIP(ipTo) == nil || strings.Contains(ipFrom, ":") || strings.Contains(ipTo, ":") {
|
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
if len(ipTo) > 0 && iputils.CompareIP(ipFrom, ipTo) > 0 {
|
|
||||||
ipFrom, ipTo = ipTo, ipFrom
|
|
||||||
}
|
|
||||||
batchIPs = append(batchIPs, &ipData{
|
|
||||||
ipFrom: ipFrom,
|
|
||||||
ipTo: ipTo,
|
|
||||||
})
|
|
||||||
} else if len(line) > 0 {
|
|
||||||
var ipFrom = line
|
|
||||||
if net.ParseIP(ipFrom) == nil || strings.Contains(ipFrom, ":") {
|
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
batchIPs = append(batchIPs, &ipData{
|
|
||||||
ipFrom: ipFrom,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
_, ipFrom, ipTo, ok := utils.ParseIPValue(line)
|
||||||
}
|
if !ok {
|
||||||
case "ipv6":
|
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
||||||
if params.Method == "single" {
|
return
|
||||||
params.Must.
|
|
||||||
Field("ipFrom", params.IpFrom).
|
|
||||||
Require("请输入正确的开始IP")
|
|
||||||
|
|
||||||
if !iputils.IsIPv6(params.IpFrom) {
|
|
||||||
this.FailField("ipFrom", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv6(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
|
||||||
} else if params.Method == "batch" {
|
|
||||||
if len(params.IpData) == 0 {
|
|
||||||
this.FailField("ipData", "请输入IP")
|
|
||||||
}
|
|
||||||
var lines = strings.Split(params.IpData, "\n")
|
|
||||||
for index, line := range lines {
|
|
||||||
line = strings.TrimSpace(line)
|
|
||||||
if strings.Contains(line, "/") { // CIDR
|
|
||||||
if !strings.Contains(line, ":") {
|
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
ipFrom, ipTo, err := configutils.ParseCIDR(line)
|
|
||||||
if err != nil {
|
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
batchIPs = append(batchIPs, &ipData{
|
|
||||||
ipFrom: ipFrom,
|
|
||||||
ipTo: ipTo,
|
|
||||||
})
|
|
||||||
} else if strings.Contains(line, "-") { // IP Range
|
|
||||||
var pieces = strings.Split(line, "-")
|
|
||||||
var ipFrom = strings.TrimSpace(pieces[0])
|
|
||||||
var ipTo = strings.TrimSpace(pieces[1])
|
|
||||||
|
|
||||||
if net.ParseIP(ipFrom) == nil || net.ParseIP(ipTo) == nil || !strings.Contains(ipFrom, ":") || !strings.Contains(ipTo, ":") {
|
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
if len(ipTo) > 0 && iputils.CompareIP(ipFrom, ipTo) > 0 {
|
|
||||||
ipFrom, ipTo = ipTo, ipFrom
|
|
||||||
}
|
|
||||||
batchIPs = append(batchIPs, &ipData{
|
|
||||||
ipFrom: ipFrom,
|
|
||||||
ipTo: ipTo,
|
|
||||||
})
|
|
||||||
} else if strings.Contains(line, ",") { // IP Range
|
|
||||||
var pieces = strings.Split(line, ",")
|
|
||||||
var ipFrom = strings.TrimSpace(pieces[0])
|
|
||||||
var ipTo = strings.TrimSpace(pieces[1])
|
|
||||||
|
|
||||||
if net.ParseIP(ipFrom) == nil || net.ParseIP(ipTo) == nil || !strings.Contains(ipFrom, ":") || !strings.Contains(ipTo, ":") {
|
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
if len(ipTo) > 0 && iputils.CompareIP(ipFrom, ipTo) > 0 {
|
|
||||||
ipFrom, ipTo = ipTo, ipFrom
|
|
||||||
}
|
|
||||||
batchIPs = append(batchIPs, &ipData{
|
|
||||||
ipFrom: ipFrom,
|
|
||||||
ipTo: ipTo,
|
|
||||||
})
|
|
||||||
} else if len(line) > 0 {
|
|
||||||
var ipFrom = line
|
|
||||||
if net.ParseIP(ipFrom) == nil || !strings.Contains(ipFrom, ":") {
|
|
||||||
this.FailField("ipData", "第"+types.String(index+1)+"行IP格式错误:"+line)
|
|
||||||
}
|
|
||||||
batchIPs = append(batchIPs, &ipData{
|
|
||||||
ipFrom: ipFrom,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
batchIPs = append(batchIPs, &ipData{
|
||||||
|
value: line,
|
||||||
|
ipFrom: ipFrom,
|
||||||
|
ipTo: ipTo,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
case "all":
|
case "all":
|
||||||
params.IpFrom = "0.0.0.0"
|
params.Value = "0.0.0.0"
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(batchIPs) > 0 {
|
if len(batchIPs) > 0 {
|
||||||
for _, ip := range batchIPs {
|
for _, ip := range batchIPs {
|
||||||
_, err := this.RPC().IPItemRPC().CreateIPItem(this.AdminContext(), &pb.CreateIPItemRequest{
|
_, err := this.RPC().IPItemRPC().CreateIPItem(this.AdminContext(), &pb.CreateIPItemRequest{
|
||||||
IpListId: params.ListId,
|
IpListId: params.ListId,
|
||||||
|
Value: ip.value,
|
||||||
IpFrom: ip.ipFrom,
|
IpFrom: ip.ipFrom,
|
||||||
IpTo: ip.ipTo,
|
IpTo: ip.ipTo,
|
||||||
ExpiredAt: params.ExpiredAt,
|
ExpiredAt: params.ExpiredAt,
|
||||||
@@ -263,8 +140,7 @@ func (this *CreateIPPopupAction) RunPost(params struct {
|
|||||||
} else {
|
} else {
|
||||||
createResp, err := this.RPC().IPItemRPC().CreateIPItem(this.AdminContext(), &pb.CreateIPItemRequest{
|
createResp, err := this.RPC().IPItemRPC().CreateIPItem(this.AdminContext(), &pb.CreateIPItemRequest{
|
||||||
IpListId: params.ListId,
|
IpListId: params.ListId,
|
||||||
IpFrom: params.IpFrom,
|
Value: params.Value,
|
||||||
IpTo: params.IpTo,
|
|
||||||
ExpiredAt: params.ExpiredAt,
|
ExpiredAt: params.ExpiredAt,
|
||||||
Reason: params.Reason,
|
Reason: params.Reason,
|
||||||
Type: params.Type,
|
Type: params.Type,
|
||||||
|
|||||||
@@ -30,7 +30,6 @@ func (this *ExportDataAction) RunGet(params struct {
|
|||||||
}) {
|
}) {
|
||||||
defer this.CreateLogInfo(codes.IPList_LogExportIPList, params.ListId)
|
defer this.CreateLogInfo(codes.IPList_LogExportIPList, params.ListId)
|
||||||
|
|
||||||
var err error
|
|
||||||
var ext string
|
var ext string
|
||||||
var jsonMaps = []maps.Map{}
|
var jsonMaps = []maps.Map{}
|
||||||
var xlsxFile *xlsx.File
|
var xlsxFile *xlsx.File
|
||||||
@@ -44,20 +43,16 @@ func (this *ExportDataAction) RunGet(params struct {
|
|||||||
case "xlsx":
|
case "xlsx":
|
||||||
ext = ".xlsx"
|
ext = ".xlsx"
|
||||||
xlsxFile = xlsx.NewFile()
|
xlsxFile = xlsx.NewFile()
|
||||||
if err != nil {
|
var err error
|
||||||
this.ErrorPage(err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
xlsxSheet, err = xlsxFile.AddSheet("IP名单")
|
xlsxSheet, err = xlsxFile.AddSheet("IP名单")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
this.ErrorPage(err)
|
this.ErrorPage(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
row := xlsxSheet.AddRow()
|
var row = xlsxSheet.AddRow()
|
||||||
row.SetHeight(26)
|
row.SetHeight(26)
|
||||||
row.AddCell().SetValue("开始IP")
|
row.AddCell().SetValue("IP/IP段")
|
||||||
row.AddCell().SetValue("结束IP")
|
|
||||||
row.AddCell().SetValue("过期时间戳")
|
row.AddCell().SetValue("过期时间戳")
|
||||||
row.AddCell().SetValue("类型")
|
row.AddCell().SetValue("类型")
|
||||||
row.AddCell().SetValue("级别")
|
row.AddCell().SetValue("级别")
|
||||||
@@ -93,27 +88,25 @@ func (this *ExportDataAction) RunGet(params struct {
|
|||||||
for _, item := range itemsResp.IpItems {
|
for _, item := range itemsResp.IpItems {
|
||||||
switch params.Format {
|
switch params.Format {
|
||||||
case "xlsx":
|
case "xlsx":
|
||||||
row := xlsxSheet.AddRow()
|
var row = xlsxSheet.AddRow()
|
||||||
row.SetHeight(26)
|
row.SetHeight(26)
|
||||||
row.AddCell().SetValue(item.IpFrom)
|
row.AddCell().SetValue(item.Value)
|
||||||
row.AddCell().SetValue(item.IpTo)
|
|
||||||
row.AddCell().SetValue(types.String(item.ExpiredAt))
|
row.AddCell().SetValue(types.String(item.ExpiredAt))
|
||||||
row.AddCell().SetValue(item.Type)
|
row.AddCell().SetValue(item.Type)
|
||||||
row.AddCell().SetValue(item.EventLevel)
|
row.AddCell().SetValue(item.EventLevel)
|
||||||
row.AddCell().SetValue(item.Reason)
|
row.AddCell().SetValue(item.Reason)
|
||||||
case "csv":
|
case "csv":
|
||||||
err = csvWriter.Write([]string{item.IpFrom, item.IpTo, types.String(item.ExpiredAt), item.Type, item.EventLevel, item.Reason})
|
err = csvWriter.Write([]string{item.Value, types.String(item.ExpiredAt), item.Type, item.EventLevel, item.Reason})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
this.ErrorPage(err)
|
this.ErrorPage(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
case "txt":
|
case "txt":
|
||||||
data = append(data, item.IpFrom+","+item.IpTo+","+types.String(item.ExpiredAt)+","+item.Type+","+item.EventLevel+","+item.Reason...)
|
data = append(data, item.Value+","+types.String(item.ExpiredAt)+","+item.Type+","+item.EventLevel+","+item.Reason...)
|
||||||
data = append(data, '\n')
|
data = append(data, '\n')
|
||||||
case "json":
|
case "json":
|
||||||
jsonMaps = append(jsonMaps, maps.Map{
|
jsonMaps = append(jsonMaps, maps.Map{
|
||||||
"ipFrom": item.IpFrom,
|
"value": item.Value,
|
||||||
"ipTo": item.IpTo,
|
|
||||||
"expiredAt": item.ExpiredAt,
|
"expiredAt": item.ExpiredAt,
|
||||||
"type": item.Type,
|
"type": item.Type,
|
||||||
"eventLevel": item.EventLevel,
|
"eventLevel": item.EventLevel,
|
||||||
@@ -127,7 +120,7 @@ func (this *ExportDataAction) RunGet(params struct {
|
|||||||
switch params.Format {
|
switch params.Format {
|
||||||
case "xlsx":
|
case "xlsx":
|
||||||
var buf = &bytes.Buffer{}
|
var buf = &bytes.Buffer{}
|
||||||
err = xlsxFile.Write(buf)
|
err := xlsxFile.Write(buf)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
this.ErrorPage(err)
|
this.ErrorPage(err)
|
||||||
return
|
return
|
||||||
@@ -137,6 +130,7 @@ func (this *ExportDataAction) RunGet(params struct {
|
|||||||
csvWriter.Flush()
|
csvWriter.Flush()
|
||||||
data = csvBuffer.Bytes()
|
data = csvBuffer.Bytes()
|
||||||
case "json":
|
case "json":
|
||||||
|
var err error
|
||||||
data, err = json.Marshal(jsonMaps)
|
data, err = json.Marshal(jsonMaps)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
this.ErrorPage(err)
|
this.ErrorPage(err)
|
||||||
|
|||||||
@@ -6,8 +6,8 @@ import (
|
|||||||
"bytes"
|
"bytes"
|
||||||
"encoding/csv"
|
"encoding/csv"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"github.com/TeaOSLab/EdgeAdmin/internal/utils"
|
||||||
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/configutils"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
||||||
"github.com/iwind/TeaGo/actions"
|
"github.com/iwind/TeaGo/actions"
|
||||||
@@ -15,7 +15,6 @@ import (
|
|||||||
"github.com/iwind/TeaGo/types"
|
"github.com/iwind/TeaGo/types"
|
||||||
"github.com/tealeg/xlsx/v3"
|
"github.com/tealeg/xlsx/v3"
|
||||||
"io"
|
"io"
|
||||||
"net"
|
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
@@ -79,9 +78,10 @@ func (this *ImportAction) RunPost(params struct {
|
|||||||
var items = []*pb.IPItem{}
|
var items = []*pb.IPItem{}
|
||||||
switch ext {
|
switch ext {
|
||||||
case ".xlsx":
|
case ".xlsx":
|
||||||
file, err := xlsx.OpenBinary(data)
|
file, openErr := xlsx.OpenBinary(data)
|
||||||
if err != nil {
|
if openErr != nil {
|
||||||
this.Fail("Excel读取错误:" + err.Error())
|
this.Fail("Excel读取错误:" + openErr.Error())
|
||||||
|
return
|
||||||
}
|
}
|
||||||
if len(file.Sheets) > 0 {
|
if len(file.Sheets) > 0 {
|
||||||
var sheet = file.Sheets[0]
|
var sheet = file.Sheets[0]
|
||||||
@@ -136,7 +136,7 @@ func (this *ImportAction) RunPost(params struct {
|
|||||||
if len(line) == 0 {
|
if len(line) == 0 {
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
item := this.createItemFromValues(strings.SplitN(string(line), ",", 6), &countIgnore)
|
item := this.createItemFromValues(strings.SplitN(string(line), ",", 5), &countIgnore)
|
||||||
if item != nil {
|
if item != nil {
|
||||||
items = append(items, item)
|
items = append(items, item)
|
||||||
}
|
}
|
||||||
@@ -150,6 +150,7 @@ func (this *ImportAction) RunPost(params struct {
|
|||||||
for _, item := range items {
|
for _, item := range items {
|
||||||
_, err = this.RPC().IPItemRPC().CreateIPItem(this.AdminContext(), &pb.CreateIPItemRequest{
|
_, err = this.RPC().IPItemRPC().CreateIPItem(this.AdminContext(), &pb.CreateIPItemRequest{
|
||||||
IpListId: params.ListId,
|
IpListId: params.ListId,
|
||||||
|
Value: item.Value,
|
||||||
IpFrom: item.IpFrom,
|
IpFrom: item.IpFrom,
|
||||||
IpTo: item.IpTo,
|
IpTo: item.IpTo,
|
||||||
ExpiredAt: item.ExpiredAt,
|
ExpiredAt: item.ExpiredAt,
|
||||||
@@ -170,60 +171,45 @@ func (this *ImportAction) RunPost(params struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (this *ImportAction) createItemFromValues(values []string, countIgnore *int) *pb.IPItem {
|
func (this *ImportAction) createItemFromValues(values []string, countIgnore *int) *pb.IPItem {
|
||||||
// ipFrom, ipTo, expiredAt, type, eventType, reason
|
// value, expiredAt, type, eventType, reason
|
||||||
|
|
||||||
var item = &pb.IPItem{}
|
var item = &pb.IPItem{}
|
||||||
switch len(values) {
|
switch len(values) {
|
||||||
case 1:
|
case 1:
|
||||||
item.IpFrom = values[0]
|
item.Value = values[0]
|
||||||
case 2:
|
case 2:
|
||||||
item.IpFrom = values[0]
|
item.Value = values[0]
|
||||||
item.IpTo = values[1]
|
item.ExpiredAt = types.Int64(values[1])
|
||||||
case 3:
|
case 3:
|
||||||
item.IpFrom = values[0]
|
item.Value = values[0]
|
||||||
item.IpTo = values[1]
|
item.ExpiredAt = types.Int64(values[1])
|
||||||
item.ExpiredAt = types.Int64(values[2])
|
item.Type = values[2]
|
||||||
case 4:
|
case 4:
|
||||||
item.IpFrom = values[0]
|
item.Value = values[0]
|
||||||
item.IpTo = values[1]
|
item.ExpiredAt = types.Int64(values[1])
|
||||||
item.ExpiredAt = types.Int64(values[2])
|
item.Type = values[2]
|
||||||
item.Type = values[3]
|
item.EventLevel = values[3]
|
||||||
case 5:
|
case 5:
|
||||||
item.IpFrom = values[0]
|
item.Value = values[0]
|
||||||
item.IpTo = values[1]
|
item.ExpiredAt = types.Int64(values[1])
|
||||||
item.ExpiredAt = types.Int64(values[2])
|
item.Type = values[2]
|
||||||
item.Type = values[3]
|
item.EventLevel = values[3]
|
||||||
item.EventLevel = values[4]
|
item.Reason = values[4]
|
||||||
case 6:
|
|
||||||
item.IpFrom = values[0]
|
|
||||||
item.IpTo = values[1]
|
|
||||||
item.ExpiredAt = types.Int64(values[2])
|
|
||||||
item.Type = values[3]
|
|
||||||
item.EventLevel = values[4]
|
|
||||||
item.Reason = values[5]
|
|
||||||
}
|
|
||||||
|
|
||||||
// CIDR
|
|
||||||
if strings.Contains(item.IpFrom, "/") {
|
|
||||||
ipFrom, ipTo, err := configutils.ParseCIDR(item.IpFrom)
|
|
||||||
if err == nil {
|
|
||||||
item.IpFrom = ipFrom
|
|
||||||
item.IpTo = ipTo
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(item.EventLevel) == 0 {
|
if len(item.EventLevel) == 0 {
|
||||||
item.EventLevel = "critical"
|
item.EventLevel = "critical"
|
||||||
}
|
}
|
||||||
|
|
||||||
if net.ParseIP(item.IpFrom) == nil {
|
newValue, ipFrom, ipTo, ok := utils.ParseIPValue(item.Value)
|
||||||
*countIgnore++
|
if !ok {
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if len(item.IpTo) > 0 && net.ParseIP(item.IpTo) == nil {
|
|
||||||
*countIgnore++
|
*countIgnore++
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
item.Value = newValue
|
||||||
|
item.IpFrom = ipFrom
|
||||||
|
item.IpTo = ipTo
|
||||||
|
|
||||||
return item
|
return item
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -172,6 +172,7 @@ func (this *IndexAction) RunGet(params struct {
|
|||||||
|
|
||||||
itemMaps = append(itemMaps, maps.Map{
|
itemMaps = append(itemMaps, maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
||||||
|
|||||||
@@ -118,6 +118,7 @@ func (this *ItemsAction) RunGet(params struct {
|
|||||||
|
|
||||||
itemMaps = append(itemMaps, maps.Map{
|
itemMaps = append(itemMaps, maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
||||||
|
|||||||
@@ -58,6 +58,7 @@ func (this *TestAction) RunPost(params struct {
|
|||||||
if resp.IpItem != nil {
|
if resp.IpItem != nil {
|
||||||
resultMap["item"] = maps.Map{
|
resultMap["item"] = maps.Map{
|
||||||
"id": resp.IpItem.Id,
|
"id": resp.IpItem.Id,
|
||||||
|
"value": resp.IpItem.Value,
|
||||||
"ipFrom": resp.IpItem.IpFrom,
|
"ipFrom": resp.IpItem.IpFrom,
|
||||||
"ipTo": resp.IpItem.IpTo,
|
"ipTo": resp.IpItem.IpTo,
|
||||||
"reason": resp.IpItem.Reason,
|
"reason": resp.IpItem.Reason,
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
package iplists
|
package iplists
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/TeaOSLab/EdgeAdmin/internal/utils"
|
||||||
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/iputils"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
||||||
"github.com/iwind/TeaGo/actions"
|
"github.com/iwind/TeaGo/actions"
|
||||||
@@ -33,6 +33,7 @@ func (this *UpdateIPPopupAction) RunGet(params struct {
|
|||||||
|
|
||||||
this.Data["item"] = maps.Map{
|
this.Data["item"] = maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"expiredAt": item.ExpiredAt,
|
"expiredAt": item.ExpiredAt,
|
||||||
@@ -49,8 +50,7 @@ func (this *UpdateIPPopupAction) RunGet(params struct {
|
|||||||
func (this *UpdateIPPopupAction) RunPost(params struct {
|
func (this *UpdateIPPopupAction) RunPost(params struct {
|
||||||
ItemId int64
|
ItemId int64
|
||||||
|
|
||||||
IpFrom string
|
Value string
|
||||||
IpTo string
|
|
||||||
ExpiredAt int64
|
ExpiredAt int64
|
||||||
Reason string
|
Reason string
|
||||||
Type string
|
Type string
|
||||||
@@ -62,50 +62,25 @@ func (this *UpdateIPPopupAction) RunPost(params struct {
|
|||||||
// 日志
|
// 日志
|
||||||
defer this.CreateLogInfo(codes.IPItem_LogUpdateIPItem, params.ItemId)
|
defer this.CreateLogInfo(codes.IPItem_LogUpdateIPItem, params.ItemId)
|
||||||
|
|
||||||
// TODO 校验ItemId所属用户
|
|
||||||
|
|
||||||
switch params.Type {
|
switch params.Type {
|
||||||
case "ipv4":
|
case "ip":
|
||||||
|
// 校验IP格式
|
||||||
params.Must.
|
params.Must.
|
||||||
Field("ipFrom", params.IpFrom).
|
Field("value", params.Value).
|
||||||
Require("请输入开始IP")
|
Require("请输入IP或IP段")
|
||||||
|
|
||||||
// 校验IP格式(ipFrom/ipTo)
|
_, _, _, ok := utils.ParseIPValue(params.Value)
|
||||||
if !iputils.IsIPv4(params.IpFrom) {
|
if !ok {
|
||||||
this.FailField("ipFrom", "请输入正确的开始IP")
|
this.FailField("value", "请输入正确的IP格式")
|
||||||
}
|
return
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv4(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的结束IP")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
|
||||||
case "ipv6":
|
|
||||||
params.Must.
|
|
||||||
Field("ipFrom", params.IpFrom).
|
|
||||||
Require("请输入正确的开始IP")
|
|
||||||
|
|
||||||
if !iputils.IsIPv6(params.IpFrom) {
|
|
||||||
this.FailField("ipFrom", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv6(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
}
|
||||||
case "all":
|
case "all":
|
||||||
params.IpFrom = "0.0.0.0"
|
params.Value = "0.0.0.0"
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err := this.RPC().IPItemRPC().UpdateIPItem(this.AdminContext(), &pb.UpdateIPItemRequest{
|
_, err := this.RPC().IPItemRPC().UpdateIPItem(this.AdminContext(), &pb.UpdateIPItemRequest{
|
||||||
IpItemId: params.ItemId,
|
IpItemId: params.ItemId,
|
||||||
IpFrom: params.IpFrom,
|
Value: params.Value,
|
||||||
IpTo: params.IpTo,
|
|
||||||
ExpiredAt: params.ExpiredAt,
|
ExpiredAt: params.ExpiredAt,
|
||||||
Reason: params.Reason,
|
Reason: params.Reason,
|
||||||
Type: params.Type,
|
Type: params.Type,
|
||||||
|
|||||||
@@ -109,6 +109,7 @@ func (this *AllowListAction) RunGet(params struct {
|
|||||||
|
|
||||||
itemMaps = append(itemMaps, maps.Map{
|
itemMaps = append(itemMaps, maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
||||||
|
|||||||
@@ -109,6 +109,7 @@ func (this *DenyListAction) RunGet(params struct {
|
|||||||
|
|
||||||
itemMaps = append(itemMaps, maps.Map{
|
itemMaps = append(itemMaps, maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
"createdTime": timeutil.FormatTime("Y-m-d", item.CreatedAt),
|
||||||
|
|||||||
@@ -70,6 +70,7 @@ func (this *TestAction) RunPost(params struct {
|
|||||||
if resp.IpItem != nil {
|
if resp.IpItem != nil {
|
||||||
resultMap["item"] = maps.Map{
|
resultMap["item"] = maps.Map{
|
||||||
"id": resp.IpItem.Id,
|
"id": resp.IpItem.Id,
|
||||||
|
"value": resp.IpItem.Value,
|
||||||
"ipFrom": resp.IpItem.IpFrom,
|
"ipFrom": resp.IpItem.IpFrom,
|
||||||
"ipTo": resp.IpItem.IpTo,
|
"ipTo": resp.IpItem.IpTo,
|
||||||
"reason": resp.IpItem.Reason,
|
"reason": resp.IpItem.Reason,
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
package ipadmin
|
package ipadmin
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/TeaOSLab/EdgeAdmin/internal/utils"
|
||||||
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/iputils"
|
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
"github.com/TeaOSLab/EdgeCommon/pkg/langs/codes"
|
||||||
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
||||||
"github.com/iwind/TeaGo/actions"
|
"github.com/iwind/TeaGo/actions"
|
||||||
@@ -33,6 +33,7 @@ func (this *UpdateIPPopupAction) RunGet(params struct {
|
|||||||
|
|
||||||
this.Data["item"] = maps.Map{
|
this.Data["item"] = maps.Map{
|
||||||
"id": item.Id,
|
"id": item.Id,
|
||||||
|
"value": item.Value,
|
||||||
"ipFrom": item.IpFrom,
|
"ipFrom": item.IpFrom,
|
||||||
"ipTo": item.IpTo,
|
"ipTo": item.IpTo,
|
||||||
"expiredAt": item.ExpiredAt,
|
"expiredAt": item.ExpiredAt,
|
||||||
@@ -49,8 +50,7 @@ func (this *UpdateIPPopupAction) RunGet(params struct {
|
|||||||
func (this *UpdateIPPopupAction) RunPost(params struct {
|
func (this *UpdateIPPopupAction) RunPost(params struct {
|
||||||
ItemId int64
|
ItemId int64
|
||||||
|
|
||||||
IpFrom string
|
Value string
|
||||||
IpTo string
|
|
||||||
ExpiredAt int64
|
ExpiredAt int64
|
||||||
Reason string
|
Reason string
|
||||||
Type string
|
Type string
|
||||||
@@ -62,50 +62,25 @@ func (this *UpdateIPPopupAction) RunPost(params struct {
|
|||||||
// 日志
|
// 日志
|
||||||
defer this.CreateLogInfo(codes.IPItem_LogUpdateIPItem, params.ItemId)
|
defer this.CreateLogInfo(codes.IPItem_LogUpdateIPItem, params.ItemId)
|
||||||
|
|
||||||
// TODO 校验ItemId所属用户
|
|
||||||
|
|
||||||
switch params.Type {
|
switch params.Type {
|
||||||
case "ipv4":
|
case "ip":
|
||||||
|
// 校验IP格式
|
||||||
params.Must.
|
params.Must.
|
||||||
Field("ipFrom", params.IpFrom).
|
Field("value", params.Value).
|
||||||
Require("请输入开始IP")
|
Require("请输入IP或IP段")
|
||||||
|
|
||||||
// 校验IP格式(ipFrom/ipTo)
|
_, _, _, ok := utils.ParseIPValue(params.Value)
|
||||||
if !iputils.IsIPv4(params.IpFrom) {
|
if !ok {
|
||||||
this.FailField("ipFrom", "请输入正确的开始IP")
|
this.FailField("value", "请输入正确的IP格式")
|
||||||
}
|
return
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv4(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的结束IP")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
|
||||||
case "ipv6":
|
|
||||||
params.Must.
|
|
||||||
Field("ipFrom", params.IpFrom).
|
|
||||||
Require("请输入正确的开始IP")
|
|
||||||
|
|
||||||
if !iputils.IsIPv6(params.IpFrom) {
|
|
||||||
this.FailField("ipFrom", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && !iputils.IsIPv6(params.IpTo) {
|
|
||||||
this.FailField("ipTo", "请输入正确的IPv6地址")
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(params.IpTo) > 0 && iputils.CompareIP(params.IpFrom, params.IpTo) > 0 {
|
|
||||||
params.IpTo, params.IpFrom = params.IpFrom, params.IpTo
|
|
||||||
}
|
}
|
||||||
case "all":
|
case "all":
|
||||||
params.IpFrom = "0.0.0.0"
|
params.Value = "0.0.0.0"
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err := this.RPC().IPItemRPC().UpdateIPItem(this.AdminContext(), &pb.UpdateIPItemRequest{
|
_, err := this.RPC().IPItemRPC().UpdateIPItem(this.AdminContext(), &pb.UpdateIPItemRequest{
|
||||||
IpItemId: params.ItemId,
|
IpItemId: params.ItemId,
|
||||||
IpFrom: params.IpFrom,
|
Value: params.Value,
|
||||||
IpTo: params.IpTo,
|
|
||||||
ExpiredAt: params.ExpiredAt,
|
ExpiredAt: params.ExpiredAt,
|
||||||
Reason: params.Reason,
|
Reason: params.Reason,
|
||||||
Type: params.Type,
|
Type: params.Type,
|
||||||
|
|||||||
@@ -140,6 +140,15 @@ Vue.component("datetime-input", {
|
|||||||
resultTimestamp: function () {
|
resultTimestamp: function () {
|
||||||
return this.timestamp
|
return this.timestamp
|
||||||
},
|
},
|
||||||
|
nextYear: function () {
|
||||||
|
let date = new Date()
|
||||||
|
date.setFullYear(date.getFullYear()+1)
|
||||||
|
this.day = date.getFullYear() + "-" + this.leadingZero(date.getMonth() + 1, 2) + "-" + this.leadingZero(date.getDate(), 2)
|
||||||
|
this.hour = this.leadingZero(date.getHours(), 2)
|
||||||
|
this.minute = this.leadingZero(date.getMinutes(), 2)
|
||||||
|
this.second = this.leadingZero(date.getSeconds(), 2)
|
||||||
|
this.change()
|
||||||
|
},
|
||||||
nextDays: function (days) {
|
nextDays: function (days) {
|
||||||
let date = new Date()
|
let date = new Date()
|
||||||
date.setTime(date.getTime() + days * 86400 * 1000)
|
date.setTime(date.getTime() + days * 86400 * 1000)
|
||||||
@@ -171,6 +180,6 @@ Vue.component("datetime-input", {
|
|||||||
<div class="ui field">:</div>
|
<div class="ui field">:</div>
|
||||||
<div class="ui field" :class="{error: hasSecondError}"><input type="text" v-model="second" maxlength="2" style="width:4em" placeholder="秒" @input="change"/></div>
|
<div class="ui field" :class="{error: hasSecondError}"><input type="text" v-model="second" maxlength="2" style="width:4em" placeholder="秒" @input="change"/></div>
|
||||||
</div>
|
</div>
|
||||||
<p class="comment">常用时间:<a href="" @click.prevent="nextHours(1)"> 1小时 </a> <span class="disabled">|</span> <a href="" @click.prevent="nextDays(1)"> 1天 </a> <span class="disabled">|</span> <a href="" @click.prevent="nextDays(3)"> 3天 </a> <span class="disabled">|</span> <a href="" @click.prevent="nextDays(7)"> 1周 </a> <span class="disabled">|</span> <a href="" @click.prevent="nextDays(30)"> 30天 </a> </p>
|
<p class="comment">常用时间:<a href="" @click.prevent="nextHours(1)"> 1小时 </a> <span class="disabled">|</span> <a href="" @click.prevent="nextDays(1)"> 1天 </a> <span class="disabled">|</span> <a href="" @click.prevent="nextDays(3)"> 3天 </a> <span class="disabled">|</span> <a href="" @click.prevent="nextDays(7)"> 1周 </a> <span class="disabled">|</span> <a href="" @click.prevent="nextDays(30)"> 30天 </a> <span class="disabled">|</span> <a href="" @click.prevent="nextYear()"> 1年 </a> </p>
|
||||||
</div>`
|
</div>`
|
||||||
})
|
})
|
||||||
@@ -2,13 +2,13 @@ Vue.component("ip-item-text", {
|
|||||||
props: ["v-item"],
|
props: ["v-item"],
|
||||||
template: `<span>
|
template: `<span>
|
||||||
<span v-if="vItem.type == 'all'">*</span>
|
<span v-if="vItem.type == 'all'">*</span>
|
||||||
<span v-if="vItem.type == 'ipv4' || vItem.type.length == 0">
|
<span v-else>
|
||||||
{{vItem.ipFrom}}
|
<span v-if="vItem.value != null && vItem.value.length > 0">{{vItem.value}}</span>
|
||||||
<span v-if="vItem.ipTo.length > 0">- {{vItem.ipTo}}</span>
|
<span v-else>
|
||||||
</span>
|
{{vItem.ipFrom}}
|
||||||
<span v-if="vItem.type == 'ipv6'">{{vItem.ipFrom}}
|
<span v-if="vItem.ipTo != null &&vItem.ipTo.length > 0">- {{vItem.ipTo}}</span>
|
||||||
<span v-if="vItem.ipTo.length > 0">- {{vItem.ipTo}}</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
<span v-if="vItem.eventLevelName != null && vItem.eventLevelName.length > 0"> 级别:{{vItem.eventLevelName}}</span>
|
<span v-if="vItem.eventLevelName != null && vItem.eventLevelName.length > 0"> 级别:{{vItem.eventLevelName}}</span>
|
||||||
</span>`
|
</span>`
|
||||||
})
|
})
|
||||||
@@ -155,8 +155,12 @@ Vue.component("ip-list-table", {
|
|||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<span v-if="item.type != 'all'" :class="{green: item.list != null && item.list.type == 'white'}">
|
<span v-if="item.type != 'all'" :class="{green: item.list != null && item.list.type == 'white'}">
|
||||||
<keyword :v-word="keyword">{{item.ipFrom}}</keyword> <span> <span class="small red" v-if="item.isRead != null && !item.isRead"> New </span> <a :href="'/servers/iplists?ip=' + item.ipFrom" v-if="vShowSearchButton" title="搜索此IP"><span><i class="icon search small" style="color: #ccc"></i></span></a></span>
|
<span v-if="item.value != null && item.value.length > 0"><keyword :v-word="keyword">{{item.value}}</keyword></span>
|
||||||
<span v-if="item.ipTo.length > 0"> - <keyword :v-word="keyword">{{item.ipTo}}</keyword></span></span>
|
<span v-else>
|
||||||
|
<keyword :v-word="keyword">{{item.ipFrom}}</keyword> <span> <span class="small red" v-if="item.isRead != null && !item.isRead"> New </span> <a :href="'/servers/iplists?ip=' + item.ipFrom" v-if="vShowSearchButton" title="搜索此IP"><span><i class="icon search small" style="color: #ccc"></i></span></a></span>
|
||||||
|
<span v-if="item.ipTo.length > 0"> - <keyword :v-word="keyword">{{item.ipTo}}</keyword></span>
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
<span v-else class="disabled">*</span>
|
<span v-else class="disabled">*</span>
|
||||||
|
|
||||||
<div v-if="item.region != null && item.region.length > 0">
|
<div v-if="item.region != null && item.region.length > 0">
|
||||||
|
|||||||
@@ -22,38 +22,11 @@
|
|||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<!-- IPv4 -->
|
<tbody v-if="type != 'all'">
|
||||||
<tbody v-if="type == 'ipv4'">
|
|
||||||
<tr>
|
<tr>
|
||||||
<td>开始IP *</td>
|
<td>IP或IP段 *</td>
|
||||||
<td>
|
<td>
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x.x.x.x" ref="focus" v-model="item.ipFrom"/>
|
<input type="text" name="value" maxlength="64" placeholder="x.x.x.x" ref="focus" v-model="item.value" style="width: 20em"/>
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x.x.x.x" v-model="item.ipTo"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
|
|
||||||
|
|
||||||
<!-- IPv6 -->
|
|
||||||
<tbody v-if="type == 'ipv6'">
|
|
||||||
<tr>
|
|
||||||
<td>开始IP *</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" v-model="item.ipFrom"/>
|
|
||||||
<p class="comment">IPv6地址,比如 1406:3c00:0:2409:13:58:103:15</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" v-model="item.ipTo"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
|
|||||||
@@ -21,37 +21,11 @@
|
|||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<!-- IPv4 -->
|
<tbody v-if="type != 'all'">
|
||||||
<tbody v-if="type == 'ipv4'">
|
|
||||||
<tr>
|
<tr>
|
||||||
<td>开始IP *</td>
|
<td>IP或IP段 *</td>
|
||||||
<td>
|
<td>
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x.x.x.x" ref="focus" v-model="item.ipFrom"/>
|
<input type="text" name="value" maxlength="64" placeholder="x.x.x.x" ref="focus" v-model="item.value" style="width: 20em"/>
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x.x.x.x" v-model="item.ipTo"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
|
|
||||||
<!-- IPv6 -->
|
|
||||||
<tbody v-if="type == 'ipv6'">
|
|
||||||
<tr>
|
|
||||||
<td>开始IP *</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" v-model="item.ipFrom"/>
|
|
||||||
<p class="comment">IPv6地址,比如 1406:3c00:0:2409:13:58:103:15</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" v-model="item.ipTo"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
|
|||||||
@@ -11,12 +11,10 @@
|
|||||||
<td class="title">类型 *</td>
|
<td class="title">类型 *</td>
|
||||||
<td>
|
<td>
|
||||||
<select class="ui dropdown auto-width" name="type" v-model="type">
|
<select class="ui dropdown auto-width" name="type" v-model="type">
|
||||||
<option value="ipv4">IPv4</option>
|
<option value="ip">单IP或IP段</option>
|
||||||
<option value="ipv6">IPv6</option>
|
|
||||||
<option value="all">所有IP</option>
|
<option value="all">所有IP</option>
|
||||||
</select>
|
</select>
|
||||||
<p class="comment" v-if="type == 'ipv4'">单个IPv4或一个IPv4范围。</p>
|
<p class="comment" v-if="type == 'ip'">单个IP或一个IP范围。</p>
|
||||||
<p class="comment" v-if="type == 'ipv6'">单个IPv6或一个IPv6范围。</p>
|
|
||||||
<p class="comment" v-if="type == 'all'">允许或禁用所有的IP。</p>
|
<p class="comment" v-if="type == 'all'">允许或禁用所有的IP。</p>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
@@ -30,60 +28,23 @@
|
|||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<!-- 单个添加 -->
|
<!-- 单个添加 -->
|
||||||
<!-- IPv4 -->
|
<tbody v-if="method == 'single' && type == 'ip'">
|
||||||
<tbody v-if="method == 'single' && type == 'ipv4'">
|
|
||||||
<tr>
|
<tr>
|
||||||
<td>开始IP *</td>
|
<td>IP或IP段 *</td>
|
||||||
<td>
|
<td>
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x.x.x.x" ref="focus" style="width: 10em"/>
|
<input type="text" name="value" maxlength="64" placeholder="x.x.x.x" ref="focus" style="width: 20em"/>
|
||||||
</td>
|
<p class="comment">支持IPv4和IPv6;支持三种格式:单个IP(比如192.168.1.100)、IP范围(比如192.168.1.1-192.168.1.255)、CIDR(比如192.168.1.1/24)。</p>
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x.x.x.x" style="width: 10em"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
|
|
||||||
<!-- IPv6 -->
|
|
||||||
<tbody v-if="method == 'single' && type == 'ipv6'">
|
|
||||||
<tr>
|
|
||||||
<td>开始IP *</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" style="width: 20em"/>
|
|
||||||
<p class="comment">IPv6地址,比如 1406:3c00:0:2409:13:58:103:15</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" style="width: 20em"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
|
|
||||||
<!-- 批量添加 -->
|
<!-- 批量添加 -->
|
||||||
<!-- IPv4 -->
|
<tbody v-if="method == 'batch' && type == 'ip'">
|
||||||
<tbody v-if="method == 'batch' && type == 'ipv4'">
|
|
||||||
<tr>
|
<tr>
|
||||||
<td>IP列表</td>
|
<td>IP列表</td>
|
||||||
<td>
|
<td>
|
||||||
<textarea name="ipData" style="width: 20em" :placeholder="'x.x.x.x' + '\n'+ 'x.x.x.x/24' + '\n' + 'x.x.x.x-y.y.y.y'" rows="3"></textarea>
|
<textarea name="ipData" style="width: 20em" :placeholder="'x.x.x.x' + '\n' + 'x.x.x.x/24' + '\n' + 'x.x.x.x-y.y.y.y'" rows="3"></textarea>
|
||||||
<p class="comment">每行一个IP,支持三种格式:<code-label :class="'grey'">192.168.1.1</code-label>、<code-label :class="'grey'">192.168.1.1/24</code-label>、<code-label :class="'grey'">192.168.1.1-192.168.1.255</code-label>。</p>
|
<p class="comment">每行一个IP,支持三种格式:<code-label :class="'grey'">192.168.1.100</code-label>、<code-label :class="'grey'">192.168.1.1-192.168.1.255</code-label>、<code-label :class="'grey'">192.168.1.1/24</code-label>。</p>
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
|
|
||||||
<!-- IPv6 -->
|
|
||||||
<tbody v-if="method == 'batch' && type == 'ipv6'">
|
|
||||||
<tr>
|
|
||||||
<td>IP列表</td>
|
|
||||||
<td>
|
|
||||||
<textarea name="ipData" style="width: 20em" :placeholder="'x:x:x:x:x:x:x:x'" rows="3"></textarea>
|
|
||||||
<p class="comment">每行一个IP。</p>
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
Tea.context(function () {
|
Tea.context(function () {
|
||||||
this.type = "ipv4"
|
this.type = "ip"
|
||||||
this.eventLevel = (this.listType == "white") ? "debug" : "critical"
|
this.eventLevel = (this.listType == "white") ? "debug" : "critical"
|
||||||
this.method = "single"
|
this.method = "single"
|
||||||
})
|
})
|
||||||
@@ -7,7 +7,7 @@
|
|||||||
<table class="ui table definition selectable">
|
<table class="ui table definition selectable">
|
||||||
<tr>
|
<tr>
|
||||||
<td class="title">说明</td>
|
<td class="title">说明</td>
|
||||||
<td>导出所有的IP</td>
|
<td>导出所有的IP,并以文件格式下载。注意v1.3.5版本及以后导出的数据不能在之前的版本中导入。</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td>格式</td>
|
<td>格式</td>
|
||||||
|
|||||||
@@ -21,7 +21,7 @@
|
|||||||
<div v-if="result.isFound">
|
<div v-if="result.isFound">
|
||||||
<div v-if="result.item != null">
|
<div v-if="result.item != null">
|
||||||
<div v-if="result.isAllowed">
|
<div v-if="result.isAllowed">
|
||||||
<span class="green">在白名单中 <ip-item-text :v-item="result.item"></ip-item-text> <a href="" @click.prevent="updateItem(result.list.id, result.item.id)" title="查看和修改"><i class="icon pencil small"></i></a></span>
|
<span class="green">在白名单中 <ip-item-text :v-item="result.item"></ip-item-text> <a href="" @click.prevent="updateItem(result.item.id)" title="查看和修改"><i class="icon pencil small"></i></a></span>
|
||||||
</div>
|
</div>
|
||||||
<div v-else>
|
<div v-else>
|
||||||
<span class="red">在黑名单中 <ip-item-text :v-item="result.item"></ip-item-text> <a href="" @click.prevent="updateItem(result.item.id)" title="查看和修改"><i class="icon pencil small"></i></a></span>
|
<span class="red">在黑名单中 <ip-item-text :v-item="result.item"></ip-item-text> <a href="" @click.prevent="updateItem(result.item.id)" title="查看和修改"><i class="icon pencil small"></i></a></span>
|
||||||
|
|||||||
@@ -21,37 +21,11 @@
|
|||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<!-- IPv4 -->
|
<tbody v-if="type != 'all'">
|
||||||
<tbody v-if="type == 'ipv4'">
|
|
||||||
<tr>
|
<tr>
|
||||||
<td>开始IP *</td>
|
<td>IP或IP段 *</td>
|
||||||
<td>
|
<td>
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x.x.x.x" ref="focus" v-model="item.ipFrom"/>
|
<input type="text" name="value" maxlength="64" placeholder="x.x.x.x" ref="focus" v-model="item.value" style="width: 20em"/>
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x.x.x.x" v-model="item.ipTo"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
|
|
||||||
<!-- IPv6 -->
|
|
||||||
<tbody v-if="type == 'ipv6'">
|
|
||||||
<tr>
|
|
||||||
<td>开始IP *</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" v-model="item.ipFrom"/>
|
|
||||||
<p class="comment">IPv6地址,比如 1406:3c00:0:2409:13:58:103:15</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" v-model="item.ipTo"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
|
|||||||
@@ -22,36 +22,11 @@
|
|||||||
</tr>
|
</tr>
|
||||||
|
|
||||||
<!-- IPv4 -->
|
<!-- IPv4 -->
|
||||||
<tbody v-if="type == 'ipv4'">
|
<tbody v-if="type != 'all'">
|
||||||
<tr>
|
<tr>
|
||||||
<td>开始IP *</td>
|
<td>IP或IP段 *</td>
|
||||||
<td>
|
<td>
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x.x.x.x" ref="focus" v-model="item.ipFrom"/>
|
<input type="text" name="value" maxlength="64" placeholder="x.x.x.x" ref="focus" v-model="item.value" style="width: 20em"/>
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x.x.x.x" v-model="item.ipTo"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</tbody>
|
|
||||||
|
|
||||||
<!-- IPv6 -->
|
|
||||||
<tbody v-if="type == 'ipv6'">
|
|
||||||
<tr>
|
|
||||||
<td>开始IP *</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipFrom" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" v-model="item.ipFrom"/>
|
|
||||||
<p class="comment">IPv6地址,比如 1406:3c00:0:2409:13:58:103:15</p>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
<tr>
|
|
||||||
<td>结束IP</td>
|
|
||||||
<td>
|
|
||||||
<input type="text" name="ipTo" maxlength="64" placeholder="x:x:x:x:x:x:x:x" ref="focus" v-model="item.ipTo"/>
|
|
||||||
<p class="comment">只有表示IP段的时候才需要填写此项。</p>
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
|
|||||||
Reference in New Issue
Block a user