| 
									
										
										
										
											2021-07-14 08:25:30 +01:00
										 |  |  | // Package geoip implements a max mind database plugin.
 | 
					
						
							|  |  |  | package geoip
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import (
 | 
					
						
							|  |  |  | 	"context"
 | 
					
						
							|  |  |  | 	"fmt"
 | 
					
						
							|  |  |  | 	"net"
 | 
					
						
							|  |  |  | 	"path/filepath"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/plugin"
 | 
					
						
							|  |  |  | 	clog "github.com/coredns/coredns/plugin/pkg/log"
 | 
					
						
							|  |  |  | 	"github.com/coredns/coredns/request"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/miekg/dns"
 | 
					
						
							|  |  |  | 	"github.com/oschwald/geoip2-golang"
 | 
					
						
							|  |  |  | )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var log = clog.NewWithPlugin(pluginName)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // GeoIP is a plugin that add geo location data to the request context by looking up a maxmind
 | 
					
						
							|  |  |  | // geoIP2 database, and which data can be later consumed by other middlewares.
 | 
					
						
							|  |  |  | type GeoIP struct {
 | 
					
						
							| 
									
										
										
										
											2022-05-02 19:25:02 +02:00
										 |  |  | 	Next  plugin.Handler
 | 
					
						
							|  |  |  | 	db    db
 | 
					
						
							|  |  |  | 	edns0 bool
 | 
					
						
							| 
									
										
										
										
											2021-07-14 08:25:30 +01:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type db struct {
 | 
					
						
							|  |  |  | 	*geoip2.Reader
 | 
					
						
							|  |  |  | 	// provides defines the schemas that can be obtained by querying this database, by using
 | 
					
						
							|  |  |  | 	// bitwise operations.
 | 
					
						
							|  |  |  | 	provides int
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const (
 | 
					
						
							|  |  |  | 	city = 1 << iota
 | 
					
						
							|  |  |  | )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var probingIP = net.ParseIP("127.0.0.1")
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-02 19:25:02 +02:00
										 |  |  | func newGeoIP(dbPath string, edns0 bool) (*GeoIP, error) {
 | 
					
						
							| 
									
										
										
										
											2021-07-14 08:25:30 +01:00
										 |  |  | 	reader, err := geoip2.Open(dbPath)
 | 
					
						
							|  |  |  | 	if err != nil {
 | 
					
						
							|  |  |  | 		return nil, fmt.Errorf("failed to open database file: %v", err)
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	db := db{Reader: reader}
 | 
					
						
							|  |  |  | 	schemas := []struct {
 | 
					
						
							|  |  |  | 		provides int
 | 
					
						
							|  |  |  | 		name     string
 | 
					
						
							|  |  |  | 		validate func() error
 | 
					
						
							|  |  |  | 	}{
 | 
					
						
							|  |  |  | 		{name: "city", provides: city, validate: func() error { _, err := reader.City(probingIP); return err }},
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	// Query the database to figure out the database type.
 | 
					
						
							|  |  |  | 	for _, schema := range schemas {
 | 
					
						
							|  |  |  | 		if err := schema.validate(); err != nil {
 | 
					
						
							|  |  |  | 			// If we get an InvalidMethodError then we know this database does not provide that schema.
 | 
					
						
							|  |  |  | 			if _, ok := err.(geoip2.InvalidMethodError); !ok {
 | 
					
						
							|  |  |  | 				return nil, fmt.Errorf("unexpected failure looking up database %q schema %q: %v", filepath.Base(dbPath), schema.name, err)
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		} else {
 | 
					
						
							| 
									
										
										
										
											2021-11-24 16:24:49 +08:00
										 |  |  | 			db.provides |= schema.provides
 | 
					
						
							| 
									
										
										
										
											2021-07-14 08:25:30 +01:00
										 |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if db.provides&city == 0 {
 | 
					
						
							|  |  |  | 		return nil, fmt.Errorf("database does not provide city schema")
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-02 19:25:02 +02:00
										 |  |  | 	return &GeoIP{db: db, edns0: edns0}, nil
 | 
					
						
							| 
									
										
										
										
											2021-07-14 08:25:30 +01:00
										 |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ServeDNS implements the plugin.Handler interface.
 | 
					
						
							|  |  |  | func (g GeoIP) ServeDNS(ctx context.Context, w dns.ResponseWriter, r *dns.Msg) (int, error) {
 | 
					
						
							|  |  |  | 	return plugin.NextOrFailure(pluginName, g.Next, ctx, w, r)
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Metadata implements the metadata.Provider Interface in the metadata plugin, and is used to store
 | 
					
						
							|  |  |  | // the data associated with the source IP of every request.
 | 
					
						
							|  |  |  | func (g GeoIP) Metadata(ctx context.Context, state request.Request) context.Context {
 | 
					
						
							|  |  |  | 	srcIP := net.ParseIP(state.IP())
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-02 19:25:02 +02:00
										 |  |  | 	if g.edns0 {
 | 
					
						
							|  |  |  | 		if o := state.Req.IsEdns0(); o != nil {
 | 
					
						
							|  |  |  | 			for _, s := range o.Option {
 | 
					
						
							|  |  |  | 				if e, ok := s.(*dns.EDNS0_SUBNET); ok {
 | 
					
						
							|  |  |  | 					srcIP = e.Address
 | 
					
						
							|  |  |  | 					break
 | 
					
						
							|  |  |  | 				}
 | 
					
						
							|  |  |  | 			}
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-14 08:25:30 +01:00
										 |  |  | 	switch {
 | 
					
						
							|  |  |  | 	case g.db.provides&city == city:
 | 
					
						
							|  |  |  | 		data, err := g.db.City(srcIP)
 | 
					
						
							|  |  |  | 		if err != nil {
 | 
					
						
							|  |  |  | 			log.Debugf("Setting up metadata failed due to database lookup error: %v", err)
 | 
					
						
							|  |  |  | 			return ctx
 | 
					
						
							|  |  |  | 		}
 | 
					
						
							|  |  |  | 		g.setCityMetadata(ctx, data)
 | 
					
						
							|  |  |  | 	}
 | 
					
						
							|  |  |  | 	return ctx
 | 
					
						
							|  |  |  | }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Name implements the Handler interface.
 | 
					
						
							|  |  |  | func (g GeoIP) Name() string { return pluginName }
 |