Browse Source

Merge pull request #24 from 9seconds/ntp

Add support of ntp time verification
tags/0.10
Sergey Arkhipov 7 years ago
parent
commit
247ee5b20b
No account linked to committer's email address
4 changed files with 93 additions and 2 deletions
  1. 18
    1
      Gopkg.lock
  2. 4
    0
      Gopkg.toml
  3. 11
    1
      main.go
  4. 60
    0
      ntp/ntp.go

+ 18
- 1
Gopkg.lock View File

@@ -16,6 +16,12 @@
16 16
   packages = ["."]
17 17
   revision = "2efee857e7cfd4f3d0138cc3cbb1b4966962b93a"
18 18
 
19
+[[projects]]
20
+  name = "github.com/beevik/ntp"
21
+  packages = ["."]
22
+  revision = "62c80a04de2086884d8296004b6d74ee1846c582"
23
+  version = "v0.2.0"
24
+
19 25
 [[projects]]
20 26
   name = "github.com/davecgh/go-spew"
21 27
   packages = ["spew"]
@@ -77,6 +83,17 @@
77 83
   revision = "eeedf312bc6c57391d84767a4cd413f02a917974"
78 84
   version = "v1.8.0"
79 85
 
86
+[[projects]]
87
+  branch = "master"
88
+  name = "golang.org/x/net"
89
+  packages = [
90
+    "bpf",
91
+    "internal/iana",
92
+    "internal/socket",
93
+    "ipv4"
94
+  ]
95
+  revision = "d0887baf81f4598189d4e12a37c6da86f0bba4d0"
96
+
80 97
 [[projects]]
81 98
   name = "gopkg.in/alecthomas/kingpin.v2"
82 99
   packages = ["."]
@@ -86,6 +103,6 @@
86 103
 [solve-meta]
87 104
   analyzer-name = "dep"
88 105
   analyzer-version = 1
89
-  inputs-digest = "c4fdd3664f683342ad0c2509f4a8bcfe5b267a6e8cdaf36f70d39536bbf89834"
106
+  inputs-digest = "f828340a30ea13c563829f9a37d0ff62974d4578411c9be02e61125dbdf98692"
90 107
   solver-name = "gps-cdcl"
91 108
   solver-version = 1

+ 4
- 0
Gopkg.toml View File

@@ -48,3 +48,7 @@
48 48
 [[constraint]]
49 49
   branch = "master"
50 50
   name = "github.com/dustin/go-humanize"
51
+
52
+[[constraint]]
53
+  name = "github.com/beevik/ntp"
54
+  version = "0.2.0"

+ 11
- 1
main.go View File

@@ -4,20 +4,22 @@ package main
4 4
 
5 5
 import (
6 6
 	"encoding/json"
7
+	"fmt"
7 8
 	"io"
8 9
 	"math/rand"
9 10
 	"os"
10 11
 	"syscall"
11 12
 	"time"
12 13
 
14
+	"github.com/juju/errors"
13 15
 	"go.uber.org/zap"
14 16
 	"go.uber.org/zap/zapcore"
15 17
 	kingpin "gopkg.in/alecthomas/kingpin.v2"
16 18
 
17 19
 	"github.com/9seconds/mtg/config"
20
+	"github.com/9seconds/mtg/ntp"
18 21
 	"github.com/9seconds/mtg/proxy"
19 22
 	"github.com/9seconds/mtg/stats"
20
-	"github.com/juju/errors"
21 23
 )
22 24
 
23 25
 var (
@@ -120,6 +122,14 @@ func main() {
120 122
 
121 123
 	if conf.UseMiddleProxy() {
122 124
 		zap.S().Infow("Use middle proxy connection to Telegram")
125
+		if diff, err := ntp.Fetch(); err != nil {
126
+			zap.S().Warnw("Could not fetch time data from NTP")
127
+		} else {
128
+			if diff >= time.Second {
129
+				usage(fmt.Sprintf("You choose to use middle proxy but your clock drift (%s) is bigger than 1 second. Please, sync your time", diff))
130
+			}
131
+			go ntp.AutoUpdate()
132
+		}
123 133
 	} else {
124 134
 		zap.S().Infow("Use direct connection to Telegram")
125 135
 	}

+ 60
- 0
ntp/ntp.go View File

@@ -0,0 +1,60 @@
1
+package ntp
2
+
3
+import (
4
+	"math/rand"
5
+	"time"
6
+
7
+	"github.com/beevik/ntp"
8
+	"github.com/juju/errors"
9
+	"go.uber.org/zap"
10
+)
11
+
12
+const autoUpdatePeriod = time.Minute
13
+
14
+var ntpEndpoints = []string{
15
+	"0.pool.ntp.org",
16
+	"1.pool.ntp.org",
17
+	"2.pool.ntp.org",
18
+	"3.pool.ntp.org",
19
+}
20
+
21
+// Fetch fetches the data on time drift.
22
+func Fetch() (time.Duration, error) {
23
+	url := ntpEndpoints[rand.Intn(len(ntpEndpoints))]
24
+	resp, err := ntp.Query(url)
25
+	if err != nil {
26
+		return 0, errors.Annotatef(err, "Cannot fetch NTP server %s", url)
27
+	}
28
+
29
+	offsetInt := int64(resp.ClockOffset)
30
+	if offsetInt < 0 {
31
+		offsetInt = -offsetInt
32
+	}
33
+	offset := time.Duration(offsetInt)
34
+
35
+	return offset, nil
36
+}
37
+
38
+// AutoUpdate runs periodic check of current time .drift state.
39
+func AutoUpdate() {
40
+	logger := zap.S().Named("ntp")
41
+
42
+	for range time.Tick(autoUpdatePeriod) {
43
+		diff, err := Fetch()
44
+		if err != nil {
45
+			logger.Debugw("Cannot fetch time from NTP", "error", err)
46
+			continue
47
+		}
48
+
49
+		switch {
50
+		case diff < 400*time.Millisecond:
51
+			logger.Debugw("NTP time drift", "value", diff.String())
52
+		case diff < 600*time.Millisecond:
53
+			logger.Infow("NTP time drift", "value", diff.String())
54
+		case diff < 800*time.Millisecond:
55
+			logger.Warnw("NTP time drift", "value", diff.String())
56
+		default:
57
+			logger.Errorw("NTP time drift", "value", diff.String())
58
+		}
59
+	}
60
+}

Loading…
Cancel
Save