2017-09-30 03:30:36 +08:00
|
|
|
// +build js
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
"strings"
|
|
|
|
|
2020-04-15 04:47:30 +08:00
|
|
|
"github.com/StackExchange/dnscontrol/v3/pkg/spflib"
|
2017-09-30 03:30:36 +08:00
|
|
|
"github.com/gopherjs/jquery"
|
|
|
|
)
|
|
|
|
|
|
|
|
type gResolver struct{}
|
|
|
|
|
|
|
|
type gResp struct {
|
|
|
|
Status int
|
|
|
|
Answer []struct {
|
|
|
|
Data string `json:"data"`
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-03 22:03:38 +08:00
|
|
|
func (g gResolver) GetSPF(fqdn string) (string, error) {
|
2017-09-30 03:30:36 +08:00
|
|
|
resp, err := http.Get("https://dns.google.com/resolve?type=txt&name=" + fqdn)
|
|
|
|
if err != nil {
|
2018-05-03 22:03:38 +08:00
|
|
|
return "", err
|
2017-09-30 03:30:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
defer resp.Body.Close()
|
|
|
|
dec := json.NewDecoder(resp.Body)
|
|
|
|
dat := &gResp{}
|
|
|
|
if err = dec.Decode(dat); err != nil {
|
2018-05-03 22:03:38 +08:00
|
|
|
return "", err
|
2017-09-30 03:30:36 +08:00
|
|
|
}
|
|
|
|
for _, a := range dat.Answer {
|
2018-05-03 22:03:38 +08:00
|
|
|
val := strings.Trim(a.Data, "\"")
|
|
|
|
if strings.HasPrefix(val, "v=spf1 ") {
|
|
|
|
return val, nil
|
|
|
|
}
|
2017-09-30 03:30:36 +08:00
|
|
|
}
|
2018-05-03 22:03:38 +08:00
|
|
|
return "", fmt.Errorf("No spf records found")
|
2017-09-30 03:30:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
var jq = jquery.NewJQuery
|
|
|
|
var parsed *spflib.SPFRecord
|
|
|
|
var domain string
|
|
|
|
|
2018-05-03 22:03:38 +08:00
|
|
|
var resolver = gResolver{}
|
|
|
|
|
2017-09-30 03:30:36 +08:00
|
|
|
func main() {
|
|
|
|
jq(func() {
|
|
|
|
jq("#lookup_btn").On(jquery.CLICK, func(e jquery.Event) {
|
|
|
|
go func() {
|
|
|
|
domain = jq("#domain").Val()
|
2018-05-03 22:03:38 +08:00
|
|
|
text, err := resolver.GetSPF(domain)
|
|
|
|
fmt.Println(text)
|
2017-09-30 03:30:36 +08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
2018-05-03 22:03:38 +08:00
|
|
|
parsed, err = spflib.Parse(text, resolver)
|
2017-09-30 03:30:36 +08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
jq("#results").SetHtml(buildHTML(parsed, domain))
|
|
|
|
jq(".cb").On(jquery.CHANGE, func(e jquery.Event) {
|
|
|
|
updateDisabledChecks()
|
|
|
|
renderResults()
|
|
|
|
})
|
|
|
|
updateDisabledChecks()
|
|
|
|
renderResults()
|
|
|
|
}()
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func updateDisabledChecks() {
|
|
|
|
jq("input:checkbox").Each(func(i int, el interface{}) {
|
|
|
|
fmt.Println(jq(el).Prop("disabled"))
|
|
|
|
jq(el).SetProp("disabled", false)
|
|
|
|
})
|
|
|
|
jq("input:checkbox:not(:checked)").Each(func(i int, el interface{}) {
|
|
|
|
fmt.Println(jq(el).Attr("id"))
|
|
|
|
jq(el).Next().Next().Find("input:checkbox").Each(func(i int, el interface{}) {
|
|
|
|
fmt.Println("^^", jq(el).Attr("id"))
|
|
|
|
jq(el).SetProp("disabled", true)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func renderResults() {
|
|
|
|
content := ""
|
|
|
|
addFlattened := func(mode string, filter string) {
|
|
|
|
flat := parsed.Flatten(filter)
|
|
|
|
lookups := 0
|
|
|
|
if filter != "*" {
|
|
|
|
lookups = parsed.Lookups() - len(strings.Split(filter, ","))
|
|
|
|
}
|
|
|
|
content += fmt.Sprintf(`
|
|
|
|
<h3> %s flattened (length %d, %d lookups)</h3><code>%s</code>
|
|
|
|
`, mode, len(flat.TXT()), lookups, flat.TXT())
|
|
|
|
split := flat.TXTSplit("_spf%d." + domain)
|
|
|
|
if len(split) > 1 {
|
|
|
|
lookups += len(split) - 1
|
|
|
|
content += fmt.Sprintf("<h3>%s flattened split (%d lookups)</h3>", mode, lookups)
|
|
|
|
for k, v := range split {
|
|
|
|
content += fmt.Sprintf("<h4>%s</h4><code>%s</code>", k, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
addFlattened("Fully", "*")
|
|
|
|
|
|
|
|
// look for selected divs
|
|
|
|
filters := []string{}
|
|
|
|
jq("input:checked").Each(func(i int, el interface{}) {
|
|
|
|
filters = append(filters, jq(el).Attr("id"))
|
|
|
|
})
|
|
|
|
if len(filters) > 0 {
|
|
|
|
addFlattened("Selectively", strings.Join(filters, ","))
|
|
|
|
}
|
|
|
|
|
|
|
|
jq("#flattened").SetHtml(content)
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildHTML(rec *spflib.SPFRecord, domain string) string {
|
|
|
|
h := "<h1>" + domain + "</h1>"
|
|
|
|
h += fmt.Sprintf("<h2>%d lookups</h2>", rec.Lookups())
|
|
|
|
return h + genRoot(rec)
|
|
|
|
}
|
|
|
|
|
|
|
|
// html based on https://codepen.io/khoama/pen/hpljA
|
|
|
|
func genRoot(rec *spflib.SPFRecord) string {
|
|
|
|
h := fmt.Sprintf(`
|
|
|
|
<ul>
|
|
|
|
<li class='root'>%s</li>
|
|
|
|
`, rec.TXT())
|
|
|
|
for _, p := range rec.Parts {
|
|
|
|
h += genPart(p)
|
|
|
|
}
|
|
|
|
h += "</ul>"
|
|
|
|
return h
|
|
|
|
}
|
|
|
|
|
|
|
|
func genPart(rec *spflib.SPFPart) string {
|
|
|
|
if !rec.IsLookup {
|
|
|
|
return fmt.Sprintf(`<li>%s</li>`, rec.Text)
|
|
|
|
}
|
|
|
|
h := fmt.Sprintf(`<li>
|
|
|
|
<input type="checkbox" class='cb' id="%s" name="%s" />
|
|
|
|
<label for="%s">%s(%d lookups)</label>`, rec.IncludeDomain, rec.IncludeDomain, rec.IncludeDomain, rec.Text, rec.IncludeRecord.Lookups()+1)
|
|
|
|
h += fmt.Sprintf("<ul>")
|
|
|
|
for _, p := range rec.IncludeRecord.Parts {
|
|
|
|
h += genPart(p)
|
|
|
|
}
|
|
|
|
h += "</ul>"
|
|
|
|
h += "</li>"
|
|
|
|
return h
|
|
|
|
}
|