package entity import ( "net/netip" "sort" "github.com/zu1k/nali/internal/db" "github.com/zu1k/nali/internal/re" "github.com/zu1k/nali/pkg/dbif" ) // ParseLine parse a line into entities func ParseLine(line string) Entities { ip4sLoc := re.IPv4Re.FindAllStringIndex(line, -1) ip6sLoc := re.IPv6Re.FindAllStringIndex(line, -1) domainsLoc := re.DomainRe.FindAllStringIndex(line, -1) tmp := make(Entities, 0, len(ip4sLoc)+len(ip6sLoc)+len(domainsLoc)) for _, e := range ip4sLoc { tmp = append(tmp, &Entity{ Loc: e, Type: TypeIPv4, Text: line[e[0]:e[1]], }) } for _, e := range ip6sLoc { text := line[e[0]:e[1]] if ip, _ := netip.ParseAddr(text); !ip.Is4In6() { tmp = append(tmp, &Entity{ Loc: e, Type: TypeIPv6, Text: text, }) } } for _, e := range domainsLoc { tmp = append(tmp, &Entity{ Loc: e, Type: TypeDomain, Text: line[e[0]:e[1]], }) } sort.Sort(tmp) es := make(Entities, 0, len(tmp)) idx := 0 for _, e := range tmp { start := e.Loc[0] if start >= idx { if start > idx { es = append(es, &Entity{ Loc: []int{idx, start}, Type: TypePlain, Text: line[idx:start], }) } e.Info = db.Find(dbif.QueryType(e.Type), e.Text) es = append(es, e) idx = e.Loc[1] } } if total := len(line); idx < total { es = append(es, &Entity{ Loc: []int{idx, total}, Type: TypePlain, Text: line[idx:total], }) } return es }