Projects STRLCPY 0dayex-checker Commits 250c7903
🤬
  • ■ ■ ■ ■ ■ ■
    .gitignore
     1 +app.exe
     2 +output.log
     3 + 
  • ■ ■ ■ ■ ■ ■
    go.mod
     1 +module app
     2 + 
     3 +go 1.17
     4 + 
     5 +require (
     6 + github.com/jchv/go-webview2 v0.0.0-20220925002352-a49836573706
     7 + github.com/tidwall/sjson v1.2.5
     8 + github.com/valyala/fasthttp v1.40.0
     9 +)
     10 + 
     11 +require (
     12 + github.com/andybalholm/brotli v1.0.4 // indirect
     13 + github.com/jchv/go-winloader v0.0.0-20200815041850-dec1ee9a7fd5 // indirect
     14 + github.com/klauspost/compress v1.15.0 // indirect
     15 + github.com/tidwall/gjson v1.14.2 // indirect
     16 + github.com/tidwall/match v1.1.1 // indirect
     17 + github.com/tidwall/pretty v1.2.0 // indirect
     18 + github.com/valyala/bytebufferpool v1.0.0 // indirect
     19 + golang.org/x/sys v0.0.0-20220227234510-4e6760a101f9 // indirect
     20 +)
     21 + 
  • ■ ■ ■ ■ ■ ■
    go.sum
     1 +github.com/andybalholm/brotli v1.0.4 h1:V7DdXeJtZscaqfNuAdSRuRFzuiKlHSC/Zh3zl9qY3JY=
     2 +github.com/andybalholm/brotli v1.0.4/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig=
     3 +github.com/jchv/go-webview2 v0.0.0-20220925002352-a49836573706 h1:alsgT8d4r2H1gBFqEYXDdZlPaQeou5m8dGR4i9tQf3M=
     4 +github.com/jchv/go-webview2 v0.0.0-20220925002352-a49836573706/go.mod h1:/BNVc0Sw3Wj6Sz9uSxPwhCEUhhWs92hPde75K2YV24A=
     5 +github.com/jchv/go-winloader v0.0.0-20200815041850-dec1ee9a7fd5 h1:pdFFlHXY9tZXmJz+tRSm1DzYEH4ebha7cffmm607bMU=
     6 +github.com/jchv/go-winloader v0.0.0-20200815041850-dec1ee9a7fd5/go.mod h1:alcuEEnZsY1WQsagKhZDsoPCRoOijYqhZvPwLG0kzVs=
     7 +github.com/klauspost/compress v1.15.0 h1:xqfchp4whNFxn5A4XFyyYtitiWI8Hy5EW59jEwcyL6U=
     8 +github.com/klauspost/compress v1.15.0/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk=
     9 +github.com/tidwall/gjson v1.14.2 h1:6BBkirS0rAHjumnjHF6qgy5d2YAJ1TLIaFE2lzfOLqo=
     10 +github.com/tidwall/gjson v1.14.2/go.mod h1:/wbyibRr2FHMks5tjHJ5F8dMZh3AcwJEMf5vlfC0lxk=
     11 +github.com/tidwall/match v1.1.1 h1:+Ho715JplO36QYgwN9PGYNhgZvoUSc9X2c80KVTi+GA=
     12 +github.com/tidwall/match v1.1.1/go.mod h1:eRSPERbgtNPcGhD8UCthc6PmLEQXEWd3PRB5JTxsfmM=
     13 +github.com/tidwall/pretty v1.2.0 h1:RWIZEg2iJ8/g6fDDYzMpobmaoGh5OLl4AXtGUGPcqCs=
     14 +github.com/tidwall/pretty v1.2.0/go.mod h1:ITEVvHYasfjBbM0u2Pg8T2nJnzm8xPwvNhhsoaGGjNU=
     15 +github.com/tidwall/sjson v1.2.5 h1:kLy8mja+1c9jlljvWTlSazM7cKDRfJuR/bOJhcY5NcY=
     16 +github.com/tidwall/sjson v1.2.5/go.mod h1:Fvgq9kS/6ociJEDnK0Fk1cpYF4FIW6ZF7LAe+6jwd28=
     17 +github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw=
     18 +github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc=
     19 +github.com/valyala/fasthttp v1.40.0 h1:CRq/00MfruPGFLTQKY8b+8SfdK60TxNztjRMnH0t1Yc=
     20 +github.com/valyala/fasthttp v1.40.0/go.mod h1:t/G+3rLek+CyY9bnIE+YlMRddxVAAGjhxndDB4i4C0I=
     21 +github.com/valyala/tcplisten v1.0.0/go.mod h1:T0xQ8SeCZGxckz9qRXTfG43PvQ/mcWh7FwZEA7Ioqkc=
     22 +golang.org/x/crypto v0.0.0-20220214200702-86341886e292/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
     23 +golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
     24 +golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
     25 +golang.org/x/sys v0.0.0-20200810151505-1b9f1253b3ed/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
     26 +golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
     27 +golang.org/x/sys v0.0.0-20210218145245-beda7e5e158e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
     28 +golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
     29 +golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
     30 +golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
     31 +golang.org/x/sys v0.0.0-20220227234510-4e6760a101f9 h1:nhht2DYV/Sn3qOayu8lM+cU1ii9sTLUeBQwQQfUHtrs=
     32 +golang.org/x/sys v0.0.0-20220227234510-4e6760a101f9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
     33 +golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
     34 +golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
     35 +golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
     36 +golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
     37 +golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
     38 + 
  • ■ ■ ■ ■ ■ ■
    http-utils.go
     1 +package main
     2 +
     3 +import (
     4 + "bytes"
     5 + "crypto/tls"
     6 + "errors"
     7 + "log"
     8 + "net"
     9 + "strings"
     10 + "time"
     11 +
     12 + "github.com/valyala/fasthttp"
     13 +)
     14 +
     15 +var httpClientTimeout = 7 * time.Second
     16 +var dialTimeout = 7 * time.Second
     17 +var httpClient = &fasthttp.Client{
     18 + TLSConfig: &tls.Config{
     19 + InsecureSkipVerify: true,
     20 + },
     21 + MaxIdemponentCallAttempts: 5, // retry if empty resp
     22 + ReadTimeout: httpClientTimeout,
     23 + MaxConnsPerHost: 233,
     24 + MaxIdleConnDuration: 15 * time.Minute,
     25 + ReadBufferSize: 1024 * 8,
     26 + Dial: func(addr string) (net.Conn, error) {
     27 + // no suitable address found => ipv6 can not dial to ipv4,..
     28 + hostname, port, err := net.SplitHostPort(addr)
     29 + if err != nil {
     30 + if err1, ok := err.(*net.AddrError); ok && strings.Index(err1.Err, "missing port") != -1 {
     31 + hostname, port, err = net.SplitHostPort(strings.TrimRight(addr, ":") + ":80")
     32 + }
     33 + if err != nil {
     34 + return nil, err
     35 + }
     36 + }
     37 + if port == "" || port == ":" {
     38 + port = "80"
     39 + }
     40 + return fasthttp.DialDualStackTimeout("["+hostname+"]:"+port, dialTimeout)
     41 + },
     42 +}
     43 +
     44 +var errEncodingNotSupported = errors.New("response content encoding not supported")
     45 +
     46 +func getResponseBody(resp *fasthttp.Response) ([]byte, error) {
     47 + var contentEncoding = resp.Header.Peek("Content-Encoding")
     48 + if len(contentEncoding) < 1 {
     49 + return resp.Body(), nil
     50 + }
     51 + if bytes.Equal(contentEncoding, []byte("gzip")) {
     52 + return resp.BodyGunzip()
     53 + }
     54 + if bytes.Equal(contentEncoding, []byte("deflate")) {
     55 + return resp.BodyInflate()
     56 + }
     57 + return nil, errEncodingNotSupported
     58 +}
     59 +
     60 +func acquireRequest(url string) *fasthttp.Request {
     61 + req := fasthttp.AcquireRequest()
     62 + normalizeRequest(req)
     63 + return req
     64 +}
     65 +
     66 +func normalizeRequest(req *fasthttp.Request) {
     67 + req.Header.Set(`User-Agent`, `Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:106.0) Gecko/20100101 Firefox/106.0`)
     68 + req.Header.Set(`Accept`, `text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,*/*;q=0.8`)
     69 + req.Header.Set(`Accept-Language`, `en-US,en;q=0.5`)
     70 + req.Header.Set(`Accept-Encoding`, `gzip, deflate`)
     71 + req.Header.Set(`Connection`, `close`)
     72 + req.Header.Set(`Upgrade-Insecure-Requests`, `1`)
     73 + req.Header.Set(`Sec-Fetch-Dest`, `document`)
     74 + req.Header.Set(`Sec-Fetch-Mode`, `navigate`)
     75 + req.Header.Set(`Sec-Fetch-Site`, `cross-site`)
     76 + req.Header.Set(`Pragma`, `no-cache`)
     77 + req.Header.Set(`Cache-Control`, `no-cache`)
     78 + req.Header.Set(`TE`, `trailers`)
     79 +}
     80 +
     81 +func doRequestFollowRedirects(req *fasthttp.Request, resp *fasthttp.Response, maxRedirectsCount int, f func(*fasthttp.Response)) (err error) {
     82 + redirectsCount := 0
     83 +
     84 + for {
     85 + if err = httpClient.DoTimeout(req, resp, httpClientTimeout); err != nil {
     86 + log.Println(err)
     87 + break
     88 + }
     89 + if f != nil {
     90 + f(resp)
     91 + }
     92 + if maxRedirectsCount == 1 {
     93 + break
     94 + }
     95 + statusCode := resp.Header.StatusCode()
     96 + if !fasthttp.StatusCodeIsRedirect(statusCode) {
     97 + break
     98 + }
     99 +
     100 + redirectsCount++
     101 + if redirectsCount > maxRedirectsCount {
     102 + err = fasthttp.ErrTooManyRedirects
     103 + break
     104 + }
     105 + location := resp.Header.Peek(`location`)
     106 + if len(location) == 0 {
     107 + err = fasthttp.ErrMissingLocation
     108 + break
     109 + }
     110 + req.URI().UpdateBytes(location)
     111 + resp.Reset()
     112 + }
     113 +
     114 + return err
     115 +}
     116 + 
  • ■ ■ ■ ■ ■ ■
    main.go
     1 +package main
     2 + 
     3 +// go build -ldflags "-H windowsgui"
     4 + 
     5 +import (
     6 + "bytes"
     7 + jsongo "encoding/json"
     8 + "errors"
     9 + "log"
     10 + "net/url"
     11 + "os"
     12 + "strings"
     13 + 
     14 + "github.com/jchv/go-webview2"
     15 + "github.com/valyala/fasthttp"
     16 +)
     17 + 
     18 +func checkExchange(url string) (string, error) {
     19 + // 1. kiểm tra email exchage
     20 + req := fasthttp.AcquireRequest()
     21 + defer fasthttp.ReleaseRequest(req)
     22 + resp := fasthttp.AcquireResponse()
     23 + defer fasthttp.ReleaseResponse(resp)
     24 + req.SetRequestURI(url)
     25 + req.URI().Update(`/owa/`)
     26 + normalizeRequest(req)
     27 + var isEx bool
     28 + err := doRequestFollowRedirects(req, resp, 5, func(resp *fasthttp.Response) {
     29 + if !isEx {
     30 + isEx = len(resp.Header.Peek(`x-owa-version`)) != 0
     31 + }
     32 + })
     33 + url = req.URI().String()
     34 + if isEx {
     35 + return url, nil
     36 + }
     37 + if err != nil {
     38 + return "", err
     39 + }
     40 + 
     41 + body, err := getResponseBody(resp)
     42 + if err != nil {
     43 + return "", err
     44 + }
     45 + 
     46 + if bytes.Contains(body, []byte(`/owa/auth.owa`)) || bytes.Contains(body, []byte(`/owa/auth/`)) {
     47 + return url, nil
     48 + }
     49 + return "", nil
     50 +}
     51 + 
     52 +func checkFirewall(url string) (isSafe bool, err error) {
     53 + req := fasthttp.AcquireRequest()
     54 + defer fasthttp.ReleaseRequest(req)
     55 + resp := fasthttp.AcquireResponse()
     56 + defer fasthttp.ReleaseResponse(resp)
     57 + req.SetRequestURI(url)
     58 + randStr := RandStringBytesMaskImprSrc(6, letterBytes)
     59 + req.URI().Update(`/autodiscover/autodiscover.json?@` + randStr + `.com/owa/&Email=autodiscover/autodiscover.json%3f@` + randStr + `.com`)
     60 + log.Println("Request:")
     61 + log.Println(req.URI().String())
     62 + normalizeRequest(req)
     63 + err = doRequestFollowRedirects(req, resp, 3, func(resp *fasthttp.Response) {
     64 + if !isSafe && bytes.Contains(resp.Header.Peek(`location`), []byte(`/errorFE.aspx`)) {
     65 + isSafe = true
     66 + }
     67 + })
     68 + if err != nil {
     69 + if strings.Contains(err.Error(), `timeout`) {
     70 + isSafe = true
     71 + err = nil
     72 + return
     73 + }
     74 + return
     75 + }
     76 + 
     77 + body, err := getResponseBody(resp)
     78 + if err != nil {
     79 + return
     80 + }
     81 + 
     82 + log.Println("Response:")
     83 + log.Println(resp.Header.String())
     84 + log.Println(string(body))
     85 + 
     86 + if resp.StatusCode() == 403 {
     87 + isSafe = true
     88 + return
     89 + }
     90 + 
     91 + if resp.StatusCode() == 500 || resp.StatusCode() == 400 || resp.StatusCode() == 200 {
     92 + return false, err
     93 + }
     94 + 
     95 + if bytes.Contains(body, []byte(`MandatoryParameterMissing`)) {
     96 + return false, nil
     97 + }
     98 + 
     99 + if bytes.Contains(body, []byte(`X-FEServer`)) || bytes.Contains(body, []byte(`.auth_errorfe_aspx`)) {
     100 + return false, nil
     101 + }
     102 + 
     103 + if bytes.Contains(body, []byte(`Server Error in '/owa' Application`)) {
     104 + return false, nil
     105 + }
     106 + return true, nil
     107 +}
     108 + 
     109 +func goCall(w webview2.WebView, name, params string) ([]byte, error) {
     110 + url := params
     111 + 
     112 + if len(url) != 0 {
     113 + // kiểm tra http
     114 + updateMsg(w, `Đang kiểm tra email exchange..`)
     115 + isEx, err := checkExchange(url)
     116 + if err != nil {
     117 + return nil, err
     118 + }
     119 + if len(isEx) == 0 {
     120 + return nil, errors.New(`Địa chỉ này không phải là Microsoft Exchange Server`)
     121 + }
     122 + 
     123 + // kiểm tra firewall
     124 + updateMsg(w, `Đang kiểm tra firewall..`)
     125 + passed, err := checkFirewall(isEx)
     126 + if err != nil {
     127 + return nil, err
     128 + }
     129 + if !passed {
     130 + return nil, errors.New(`Firewall chưa được cập nhật để chặn chuỗi trên URL: <span style="font-weight:bold; color: green">/autodiscover/autodiscover.json?@</span><br>Vui lòng cập nhật firewall để ngăn chặn cuộc tấn công.<br>Tham khảo: <a href="https://www.gteltsc.vn/blog/canh-bao-chien-dich-tan-cong-su-dung-lo-hong-zero-day-tren-microsoft-exchange-server-12714.html" target="_blank">Hướng dẫn</a>`)
     131 + }
     132 + }
     133 + // kiểm tra file system
     134 + // time.Sleep(5 * time.Second)
     135 + 
     136 + return []byte(`Chúc mừng, máy chủ này an toàn!`), nil
     137 +}
     138 + 
     139 +func updateMsg(w webview2.WebView, msg string) {
     140 + webviewDispatch(w, func() {
     141 + jsBytes, _ := jsongo.Marshal(msg)
     142 + w.Eval(`document.querySelector('#msg').innerHTML=` + string(jsBytes))
     143 + })
     144 +}
     145 + 
     146 +func asyncCall(w webview2.WebView, id, name, params string) {
     147 + retBytes, err := goCall(w, name, params)
     148 + webviewDispatch(w, func() {
     149 + var retStr string
     150 + jsId, _ := jsongo.Marshal(id)
     151 + if err != nil {
     152 + retBytes, _ := jsongo.Marshal(err.Error())
     153 + retStr = `{"error":` + string(retBytes) + `}`
     154 + } else {
     155 + retBytes, _ := jsongo.Marshal(retBytes)
     156 + retStr = string(retBytes)
     157 + }
     158 + w.Eval(`window.go_wait[` + string(jsId) + `].resolve(` + retStr + `)`)
     159 + })
     160 +}
     161 + 
     162 +func main() {
     163 + 
     164 + f, err := os.OpenFile("output.log", os.O_RDWR|os.O_CREATE|os.O_APPEND, 0666)
     165 + if err != nil {
     166 + log.Println(err)
     167 + }
     168 + defer f.Close()
     169 + log.SetOutput(f)
     170 + 
     171 + w := webview2.NewWithOptions(webview2.WebViewOptions{
     172 + Debug: true,
     173 + AutoFocus: true,
     174 + WindowOptions: webview2.WindowOptions{
     175 + Width: 450,
     176 + Height: 450,
     177 + Title: "VNCERT/CC - Kiểm tra lỗ hổng 0-day Exchange Server",
     178 + },
     179 + })
     180 + if w == nil {
     181 + log.Panicln("Failed to load webview.")
     182 + }
     183 + defer w.Destroy()
     184 + w.Bind("go_call", func(id, name, params string) {
     185 + go asyncCall(w, id, name, params)
     186 + })
     187 + w.Bind("quit", func() {
     188 + w.Terminate()
     189 + })
     190 + w.Navigate(`data:text/html,` + url.PathEscape(`
     191 + <!doctype html>
     192 + <head>
     193 + <meta charset="utf-8">
     194 + </head>
     195 + <html style="background-color: #fff;">
     196 + <body style="text-align: center">
     197 + <!-- logo -->
     198 + <img style="width: 150px" src="" />
     199 + 
     200 + 
     201 + <h3>Công cụ kiểm tra lỗ hổng 0-day Exchange Server</h3>
     202 + 
     203 + <input value="" type="text" style="font-size: 25px; width: 98%;" id="url" placeholder="Nhập địa chỉ website: https://">
     204 + <br>
     205 + <br>
     206 + <button style="font-size: 25px; width: 50%;" onclick="check_site()">Kiểm tra</button>
     207 + <br>
     208 + <br>
     209 + <div>
     210 + <span id="msg" style="font-size: 20px; color: rgb(27, 65, 95)"></span>
     211 + </div>
     212 + <script>
     213 + // Copyright (c) by vinhjaxt at VNCERT/CC
     214 + //
     215 + function b64DecodeUnicode(str) {
     216 + // Going backwards: from bytestream, to percent-encoding, to original string.
     217 + return decodeURIComponent(atob(str).split('').map(function(c) {
     218 + return '%' + ('00' + c.charCodeAt(0).toString(16)).slice(-2);
     219 + }).join(''));
     220 + }
     221 + window.go_call_id = 0
     222 + window.go_wait = Object.create(null)
     223 + 
     224 + // disable zoom
     225 + window.addEventListener('wheel', e => {
     226 + if (event.ctrlKey == true) event.preventDefault()
     227 + }, { passive: false })
     228 + window.addEventListener('mousewheel', e => {
     229 + if (event.ctrlKey == true) event.preventDefault()
     230 + }, { passive: false })
     231 + window.addEventListener('DOMMouseScroll', e => {
     232 + if (event.ctrlKey == true) event.preventDefault()
     233 + }, { passive: false })
     234 + window.addEventListener('keydown', e => {
     235 + if (event.ctrlKey==true && (event.which == '61' || event.which == '107' || event.which == '173' || event.which == '109' || event.which == '187' || event.which == '189' ) ) event.preventDefault()
     236 + })
     237 + 
     238 + // disable right click (context menu)
     239 + // window.addEventListener('contextmenu', e => e.preventDefault())
     240 + window.addEventListener('mouseup', e => e.preventDefault())
     241 + 
     242 + const msg = document.querySelector('#msg')
     243 + let checking = false
     244 + function check_site(){
     245 + if (checking) return
     246 + 
     247 + url.value = url.value.trim()
     248 + 
     249 + if (url.value.startsWith('//')) url.value = 'http:'+url.value
     250 + if (!/^https?:\/\//.test(url.value)) url.value = 'http://'+url.value
     251 + 
     252 + if (!url.value) {
     253 + msg.innerText = 'Vui lòng nhập Địa chỉ website!'
     254 + msg.style.color = 'blue'
     255 + return
     256 + }
     257 + checking = true
     258 + msg.innerText = 'Đang kiểm tra..'
     259 + msg.style.color = 'blue'
     260 + 
     261 + const id = (++window.go_call_id).toString(36)
     262 + const prm = new Promise((resolve, reject) => {
     263 + window.go_wait[id] = {resolve, reject}
     264 + })
     265 + go_call(id, 'check', url.value).then(() => prm).then(r => {
     266 + console.log(r)
     267 + if (r.error) {
     268 + msg.innerHTML = 'Lỗi: '+r.error
     269 + msg.style.color = 'red'
     270 + return
     271 + }
     272 + msg.innerHTML = b64DecodeUnicode(r)
     273 + msg.style.color = 'green'
     274 + }).catch(e => {
     275 + msg.innerText = 'Lỗi trong quá trình kiểm tra: '+e
     276 + msg.style.color = 'red'
     277 + console.error(e)
     278 + }).then(() => {
     279 + delete window.go_wait[id]
     280 + checking = false
     281 + })
     282 + }
     283 + </script>
     284 + </body>
     285 + </html>
     286 + `))
     287 + w.Run()
     288 +}
     289 + 
  • ■ ■ ■ ■ ■ ■
    rand-str.go
     1 +package main
     2 +
     3 +import (
     4 + "math/rand"
     5 + "time"
     6 +)
     7 +
     8 +var src = rand.NewSource(time.Now().UnixNano())
     9 +
     10 +const letterBytes = "abcdefghijklmnopqrstuvwxyz0123456789-"
     11 +const (
     12 + letterIdxBits = 6 // 6 bits to represent a letter index
     13 + letterIdxMask = 1<<letterIdxBits - 1 // All 1-bits, as many as letterIdxBits
     14 + letterIdxMax = 63 / letterIdxBits // # of letter indices fitting in 63 bits
     15 +)
     16 +
     17 +func RandStringBytesMaskImprSrc(n int, letterBytes string) string {
     18 + b := make([]byte, n)
     19 + // A src.Int63() generates 63 random bits, enough for letterIdxMax characters!
     20 + for i, cache, remain := n-1, src.Int63(), letterIdxMax; i >= 0; {
     21 + if remain == 0 {
     22 + cache, remain = src.Int63(), letterIdxMax
     23 + }
     24 + if idx := int(cache & letterIdxMask); idx < len(letterBytes) {
     25 + b[i] = letterBytes[idx]
     26 + i--
     27 + }
     28 + cache >>= letterIdxBits
     29 + remain--
     30 + }
     31 +
     32 + return string(b)
     33 +}
     34 + 
  • ■ ■ ■ ■ ■ ■
    webview.go
     1 +package main
     2 +
     3 +import (
     4 + "sync/atomic"
     5 + "time"
     6 +
     7 + "github.com/jchv/go-webview2"
     8 +)
     9 +
     10 +func webviewDispatch(w webview2.WebView, f func()) {
     11 + var fired uint32
     12 + dpFunc := func() {
     13 + if !atomic.CompareAndSwapUint32(&fired, 0, 1) {
     14 + return
     15 + }
     16 + f()
     17 + }
     18 + w.Dispatch(dpFunc)
     19 + for atomic.LoadUint32(&fired) == 0 {
     20 + w.Dispatch(dpFunc)
     21 + time.Sleep(100 * time.Millisecond)
     22 + }
     23 +}
     24 + 
Please wait...
Page is in error, reload to recover