Compare commits

...

14 Commits

9 changed files with 130 additions and 32 deletions

2
.gitignore vendored
View File

@ -1,4 +1,4 @@
.idea/
.vscode/
.devcontainer/
qqwry.dat
assets/qqwry.dat

View File

@ -5,8 +5,7 @@ Golang QQWry高性能纯真IP查询库。
# 使用须知
1. 仅支持ipv4查询。
2. city可能是城市,也可能是国家。
3. area可能是区域也可能是运营商。
2. city也可能是国家。
# 使用说明
@ -24,8 +23,8 @@ func main() {
panic(err)
}
// 从内存或缓存查询IP
city, area, err := qqwry.QueryIP("1.1.1.1")
log.Printf("城市:%s区域%s错误%v", city, area, err)
city, isp, err := qqwry.QueryIP("1.1.1.1")
log.Printf("城市:%s运营商%s错误%v", city, isp, err)
}
```
@ -33,6 +32,18 @@ func main() {
- [https://aite.xyz/share-file/qqwry/qqwry.dat](https://aite.xyz/share-file/qqwry/qqwry.dat)
# 编译说明
1. 下载IP数据库并放置于assets目录中。
2. client和server需要go1.16的内嵌资源特性。
3. 作为库使用请直接引包并不需要go1.16+才能编译。
# 服务接口
1. 自行根据需要调整server下源码。
2. 可以通过-listen参数指定http服务地址。
3. json apicurl http://127.0.0.1/ip/1.1.1.1
# 特别感谢
- 感谢[纯真IP库](https://www.cz88.net/)一直坚持为大家提供免费IP数据库。

6
assets/assets.go Normal file
View File

@ -0,0 +1,6 @@
package assets
import _ "embed"
//go:embed qqwry.dat
var QQWryDat []byte

25
client/client.go Normal file
View File

@ -0,0 +1,25 @@
package main
import (
"fmt"
"github.com/xiaoqidun/qqwry"
"github.com/xiaoqidun/qqwry/assets"
"os"
)
func init() {
qqwry.LoadData(assets.QQWryDat)
}
func main() {
if len(os.Args) < 2 {
return
}
queryIp := os.Args[1]
city, isp, err := qqwry.QueryIP(queryIp)
if err != nil {
fmt.Printf("错误:%v\n", err)
return
}
fmt.Printf("城市:%s运营商%s\n", city, isp)
}

4
go.mod
View File

@ -1,5 +1,5 @@
module github.com/xiaoqidun/qqwry
go 1.16
go 1.20
require golang.org/x/text v0.3.5
require golang.org/x/text v0.9.0

5
go.sum
View File

@ -1,3 +1,2 @@
golang.org/x/text v0.3.5 h1:i6eZZ+zk0SOf0xgBpEpPD18qWcJda6q1sxt3S0kzyUQ=
golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
golang.org/x/text v0.9.0 h1:2sjJmO8cDvYveuX97RDLsxlyUxLl+GHoLxBiRdHllBE=
golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=

View File

@ -6,8 +6,10 @@ import (
"errors"
"golang.org/x/text/encoding/simplifiedchinese"
"golang.org/x/text/transform"
"io/ioutil"
"io"
"net"
"os"
"strings"
"sync"
)
@ -25,7 +27,7 @@ const (
type cache struct {
City string
Area string
Isp string
}
func byte3ToUInt32(data []byte) uint32 {
@ -38,15 +40,15 @@ func byte3ToUInt32(data []byte) uint32 {
func gb18030Decode(src []byte) string {
in := bytes.NewReader(src)
out := transform.NewReader(in, simplifiedchinese.GB18030.NewDecoder())
d, _ := ioutil.ReadAll(out)
d, _ := io.ReadAll(out)
return string(d)
}
// QueryIP 从内存或缓存查询IP
func QueryIP(queryIp string) (city string, area string, err error) {
func QueryIP(queryIp string) (city string, isp string, err error) {
if v, ok := ipCache.Load(queryIp); ok {
city = v.(cache).City
area = v.(cache).Area
isp = v.(cache).Isp
return
}
ip := net.ParseIP(queryIp).To4()
@ -87,7 +89,7 @@ func QueryIP(queryIp string) (city string, area string, err error) {
}
posM := offset + 4
mode := data[posM]
var areaPos uint32
var ispPos uint32
switch mode {
case redirectMode1:
posC := byte3ToUInt32(data[posM+1 : posM+4])
@ -106,7 +108,7 @@ func QueryIP(queryIp string) (city string, area string, err error) {
if mode != redirectMode2 {
posC += uint32(len(city) + 1)
}
areaPos = posC
ispPos = posC
case redirectMode2:
posCA := byte3ToUInt32(data[posM+1 : posM+4])
for i := posCA; i < dataLen; i++ {
@ -115,7 +117,7 @@ func QueryIP(queryIp string) (city string, area string, err error) {
break
}
}
areaPos = offset + 8
ispPos = offset + 8
default:
posCA := offset + 4
for i := posCA; i < dataLen; i++ {
@ -124,23 +126,31 @@ func QueryIP(queryIp string) (city string, area string, err error) {
break
}
}
areaPos = offset + uint32(5+len(city))
ispPos = offset + uint32(5+len(city))
}
areaMode := data[areaPos]
if areaMode == redirectMode1 || areaMode == redirectMode2 {
areaPos = byte3ToUInt32(data[areaPos+1 : areaPos+4])
if city != "" {
city = strings.TrimSpace(gb18030Decode([]byte(city)))
}
if areaPos > 0 {
for i := areaPos; i < dataLen; i++ {
ispMode := data[ispPos]
if ispMode == redirectMode1 || ispMode == redirectMode2 {
ispPos = byte3ToUInt32(data[ispPos+1 : ispPos+4])
}
if ispPos > 0 {
for i := ispPos; i < dataLen; i++ {
if data[i] == 0 {
area = string(data[areaPos:i])
isp = string(data[ispPos:i])
if isp != "" {
if strings.Contains(isp, "CZ88.NET") {
isp = ""
} else {
isp = strings.TrimSpace(gb18030Decode([]byte(isp)))
}
}
break
}
}
}
city = gb18030Decode([]byte(city))
area = gb18030Decode([]byte(area))
ipCache.Store(queryIp, cache{City: city, Area: area})
ipCache.Store(queryIp, cache{City: city, Isp: isp})
return
}
@ -152,7 +162,7 @@ func LoadData(database []byte) {
// LoadFile 从文件加载IP数据库
func LoadFile(filepath string) (err error) {
data, err = ioutil.ReadFile(filepath)
data, err = os.ReadFile(filepath)
if err != nil {
return
}

View File

@ -5,16 +5,16 @@ import (
)
func init() {
if err := LoadFile("qqwry.dat"); err != nil {
if err := LoadFile("assets/qqwry.dat"); err != nil {
panic(err)
}
}
func TestQueryIP(t *testing.T) {
queryIp := "1.1.1.1"
city, area, err := QueryIP(queryIp)
city, isp, err := QueryIP(queryIp)
if err != nil {
t.Fatal(err)
}
t.Logf("城市:%s区域%s", city, area)
t.Logf("城市:%s运营商%s", city, isp)
}

47
server/server.go Normal file
View File

@ -0,0 +1,47 @@
package main
import (
"encoding/json"
"flag"
"github.com/xiaoqidun/qqwry"
"github.com/xiaoqidun/qqwry/assets"
"net"
"net/http"
)
type resp struct {
IP string `json:"ip"`
Err string `json:"err"`
City string `json:"city"`
Isp string `json:"isp"`
}
func init() {
qqwry.LoadData(assets.QQWryDat)
}
func main() {
listen := flag.String("listen", "127.0.0.1:80", "http server listen addr")
flag.Parse()
http.HandleFunc("/ip/", IpAPI)
if err := http.ListenAndServe(*listen, nil); err != nil {
panic(err)
}
}
func IpAPI(writer http.ResponseWriter, request *http.Request) {
ip := request.URL.Path[4:]
if ip == "" {
ip, _, _ = net.SplitHostPort(request.RemoteAddr)
}
rw := &resp{IP: ip}
city, isp, err := qqwry.QueryIP(ip)
if err != nil {
rw.Err = err.Error()
} else {
rw.City = city
rw.Isp = isp
}
b, _ := json.MarshalIndent(rw, "", " ")
_, _ = writer.Write(b)
}