mosdns/plugin/executable/reverse_lookup/reverse_lookup.go
dengxiongjian cd761e8145
Some checks are pending
Test mosdns / build (push) Waiting to run
新增Mikrotik API 插入解析ip
2025-07-31 11:28:55 +08:00

190 lines
4.3 KiB
Go

/*
* Copyright (C) 2020-2022, IrineSistiana
*
* This file is part of mosdns.
*
* mosdns is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* mosdns 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
package reverselookup
import (
"context"
"fmt"
"github.com/IrineSistiana/mosdns/v5/coremain"
"github.com/IrineSistiana/mosdns/v5/pkg/cache"
"github.com/IrineSistiana/mosdns/v5/pkg/dnsutils"
"github.com/IrineSistiana/mosdns/v5/pkg/query_context"
"github.com/IrineSistiana/mosdns/v5/pkg/utils"
"github.com/IrineSistiana/mosdns/v5/plugin/executable/sequence"
"github.com/go-chi/chi/v5"
"github.com/miekg/dns"
"net"
"net/http"
"net/netip"
"time"
)
const (
PluginType = "reverse_lookup"
)
func init() {
coremain.RegNewPluginFunc(PluginType, Init, func() any { return new(Args) })
}
var _ sequence.RecursiveExecutable = (*ReverseLookup)(nil)
type Args struct {
Size int `yaml:"size"` // Default is 64*1024
HandlePTR bool `yaml:"handle_ptr"`
TTL int `yaml:"ttl"` // Default is 7200 (2h)
}
func (a *Args) init() {
utils.SetDefaultUnsignNum(&a.Size, 64*1024)
utils.SetDefaultUnsignNum(&a.TTL, 7200)
}
type ReverseLookup struct {
args *Args
c *cache.Cache[key, string]
}
func Init(bp *coremain.BP, args any) (any, error) {
return NewReverseLookup(bp, args.(*Args))
}
func NewReverseLookup(bp *coremain.BP, args *Args) (any, error) {
args.init()
c := cache.New[key, string](cache.Opts{Size: args.Size})
p := &ReverseLookup{
args: args,
c: c,
}
r := chi.NewRouter()
r.Get("/", p.ServeHTTP)
bp.RegAPI(r)
return p, nil
}
func (p *ReverseLookup) Exec(ctx context.Context, qCtx *query_context.Context, next sequence.ChainWalker) error {
q := qCtx.Q()
if r := p.ResponsePTR(q); r != nil {
qCtx.SetResponse(r)
return nil
}
if err := next.ExecNext(ctx, qCtx); err != nil {
return err
}
p.saveIPs(q, qCtx.R())
return nil
}
func (p *ReverseLookup) Close() error {
return p.c.Close()
}
func (p *ReverseLookup) ServeHTTP(w http.ResponseWriter, req *http.Request) {
ipStr := req.URL.Query().Get("ip")
if len(ipStr) == 0 {
http.Error(w, "no 'ip' query parameter found", http.StatusBadRequest)
return
}
addr, err := netip.ParseAddr(ipStr)
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
return
}
d := p.lookup(netip.AddrFrom16(addr.As16()))
if len(d) > 0 {
_, _ = fmt.Fprint(w, d)
}
}
func (p *ReverseLookup) lookup(n netip.Addr) string {
v, _, _ := p.c.Get(key(as16(n)))
return v
}
func (p *ReverseLookup) ResponsePTR(q *dns.Msg) *dns.Msg {
if p.args.HandlePTR && len(q.Question) > 0 && q.Question[0].Qtype == dns.TypePTR {
question := q.Question[0]
addr, _ := dnsutils.ParsePTRQName(question.Name)
// If we cannot parse this ptr name. Just ignore it and pass query to next node.
// PTR standards are a mess.
if !addr.IsValid() {
return nil
}
fqdn := p.lookup(addr)
if len(fqdn) > 0 {
r := new(dns.Msg)
r.SetReply(q)
r.Answer = append(r.Answer, &dns.PTR{
Hdr: dns.RR_Header{
Name: question.Name,
Rrtype: question.Qtype,
Class: question.Qclass,
Ttl: 5,
},
Ptr: fqdn,
})
return r
}
}
return nil
}
func (p *ReverseLookup) saveIPs(q, r *dns.Msg) {
if r == nil {
return
}
now := time.Now()
for _, rr := range r.Answer {
var ip net.IP
switch rr := rr.(type) {
case *dns.A:
ip = rr.A
case *dns.AAAA:
ip = rr.AAAA
default:
continue
}
addr, ok := netip.AddrFromSlice(ip)
if !ok {
continue
}
h := rr.Header()
if int(h.Ttl) > p.args.TTL {
h.Ttl = uint32(p.args.TTL)
}
name := h.Name
if len(q.Question) == 1 {
name = q.Question[0].Name
}
p.c.Store(key(as16(addr)), name, now.Add(time.Duration(p.args.TTL)*time.Second))
}
}
func as16(n netip.Addr) netip.Addr {
if n.Is6() {
return n
}
return netip.AddrFrom16(n.As16())
}