From 5d60bb05ab98c4ed31a9e5a189c4a6dfeb62d817 Mon Sep 17 00:00:00 2001
From: Simon Zolin <s.zolin@adguard.com>
Date: Tue, 25 Jun 2019 15:55:09 +0300
Subject: [PATCH 1/3] * /control/version.json: add "recheck_now" parameter

---
 AGHTechDoc.md          |  6 +++++-
 home/control_update.go | 34 ++++++++++++++++++++++++----------
 2 files changed, 29 insertions(+), 11 deletions(-)

diff --git a/AGHTechDoc.md b/AGHTechDoc.md
index e4462ad9..cfe0304f 100644
--- a/AGHTechDoc.md
+++ b/AGHTechDoc.md
@@ -257,7 +257,11 @@ Server can only auto-update if the current version is equal or higher than `self
 
 Request:
 
-	GET /control/version.json
+	POST /control/version.json
+
+	{
+		"recheck_now": true | false // if false, server will check for a new version data only once in several hours
+	}
 
 Response:
 
diff --git a/home/control_update.go b/home/control_update.go
index f910b055..763a0b48 100644
--- a/home/control_update.go
+++ b/home/control_update.go
@@ -51,6 +51,10 @@ func getVersionResp(data []byte) []byte {
 	return d
 }
 
+type getVersionJSONRequest struct {
+	RecheckNow bool `json:"recheck_now"`
+}
+
 // Get the latest available version from the Internet
 func handleGetVersionJSON(w http.ResponseWriter, r *http.Request) {
 	log.Tracef("%s %v", r.Method, r.URL)
@@ -60,19 +64,29 @@ func handleGetVersionJSON(w http.ResponseWriter, r *http.Request) {
 		return
 	}
 
-	now := time.Now()
-	controlLock.Lock()
-	cached := now.Sub(versionCheckLastTime) <= versionCheckPeriod && len(versionCheckJSON) != 0
-	data := versionCheckJSON
-	controlLock.Unlock()
-
-	if cached {
-		// return cached copy
-		w.Header().Set("Content-Type", "application/json")
-		w.Write(getVersionResp(data))
+	req := getVersionJSONRequest{}
+	err := json.NewDecoder(r.Body).Decode(&req)
+	if err != nil {
+		httpError(w, http.StatusBadRequest, "JSON parse: %s", err)
 		return
 	}
 
+	now := time.Now()
+	if !req.RecheckNow {
+		controlLock.Lock()
+		cached := now.Sub(versionCheckLastTime) <= versionCheckPeriod && len(versionCheckJSON) != 0
+		data := versionCheckJSON
+		controlLock.Unlock()
+
+		if cached {
+			log.Tracef("Returning cached data")
+			w.Header().Set("Content-Type", "application/json")
+			w.Write(getVersionResp(data))
+			return
+		}
+	}
+
+	log.Tracef("Downloading data from %s", versionCheckURL)
 	resp, err := client.Get(versionCheckURL)
 	if err != nil {
 		httpError(w, http.StatusBadGateway, "Couldn't get version check json from %s: %T %s\n", versionCheckURL, err, err)

From 0e9df33a409dbffeb2d5d45b7f31f30587a2e4f7 Mon Sep 17 00:00:00 2001
From: Ildar Kamalov <i.kamalov@adguard.com>
Date: Tue, 25 Jun 2019 17:56:50 +0300
Subject: [PATCH 2/3] + client: add button for check updates

---
 client/src/__locales/en.json                |   4 ++-
 client/src/actions/index.js                 |   7 +++--
 client/src/api/Api.js                       |  10 +++++--
 client/src/components/App/index.js          |   3 +-
 client/src/components/Dashboard/index.js    |  22 ++++++++++++--
 client/src/components/Header/Header.css     |   9 ++++++
 client/src/components/Header/Version.js     |  31 +++++++++++++++-----
 client/src/components/Header/index.js       |  12 ++++----
 client/src/components/Settings/Settings.css |   7 +++++
 client/src/components/ui/Card.css           |  15 ----------
 client/src/components/ui/Icons.js           | Bin 11004 -> 11293 bytes
 client/src/reducers/index.js                |   1 +
 12 files changed, 84 insertions(+), 37 deletions(-)

diff --git a/client/src/__locales/en.json b/client/src/__locales/en.json
index 63301db5..1e66a0e8 100644
--- a/client/src/__locales/en.json
+++ b/client/src/__locales/en.json
@@ -312,5 +312,7 @@
     "access_disallowed_desc": "A list of CIDR or IP addresses. If configured, AdGuard Home will drop requests from these IP addresses.",
     "access_blocked_title": "Blocked domains",
     "access_blocked_desc": "Don't confuse this with filters. AdGuard Home will drop DNS queries with these domains in query's question.",
-    "access_settings_saved": "Access settings successfully saved"
+    "access_settings_saved": "Access settings successfully saved",
+    "updates_checked": "Updates successfully checked",
+    "check_updates_now": "Check updates now"
 }
\ No newline at end of file
diff --git a/client/src/actions/index.js b/client/src/actions/index.js
index 3ceed2c4..aca824c6 100644
--- a/client/src/actions/index.js
+++ b/client/src/actions/index.js
@@ -145,11 +145,14 @@ export const getVersionRequest = createAction('GET_VERSION_REQUEST');
 export const getVersionFailure = createAction('GET_VERSION_FAILURE');
 export const getVersionSuccess = createAction('GET_VERSION_SUCCESS');
 
-export const getVersion = () => async (dispatch) => {
+export const getVersion = (recheck = false) => async (dispatch) => {
     dispatch(getVersionRequest());
     try {
-        const newVersion = await apiClient.getGlobalVersion();
+        const newVersion = await apiClient.getGlobalVersion({ recheck_now: recheck });
         dispatch(getVersionSuccess(newVersion));
+        if (recheck) {
+            dispatch(addSuccessToast('updates_checked'));
+        }
     } catch (error) {
         dispatch(addErrorToast({ error }));
         dispatch(getVersionFailure());
diff --git a/client/src/api/Api.js b/client/src/api/Api.js
index ad4e03fb..76b17888 100644
--- a/client/src/api/Api.js
+++ b/client/src/api/Api.js
@@ -36,7 +36,7 @@ export default class Api {
     GLOBAL_QUERY_LOG_DISABLE = { path: 'querylog_disable', method: 'POST' };
     GLOBAL_SET_UPSTREAM_DNS = { path: 'set_upstreams_config', method: 'POST' };
     GLOBAL_TEST_UPSTREAM_DNS = { path: 'test_upstream_dns', method: 'POST' };
-    GLOBAL_VERSION = { path: 'version.json', method: 'GET' };
+    GLOBAL_VERSION = { path: 'version.json', method: 'POST' };
     GLOBAL_ENABLE_PROTECTION = { path: 'enable_protection', method: 'POST' };
     GLOBAL_DISABLE_PROTECTION = { path: 'disable_protection', method: 'POST' };
     GLOBAL_UPDATE = { path: 'update', method: 'POST' };
@@ -125,9 +125,13 @@ export default class Api {
         return this.makeRequest(path, method, config);
     }
 
-    getGlobalVersion() {
+    getGlobalVersion(data) {
         const { path, method } = this.GLOBAL_VERSION;
-        return this.makeRequest(path, method);
+        const config = {
+            data,
+            headers: { 'Content-Type': 'application/json' },
+        };
+        return this.makeRequest(path, method, config);
     }
 
     enableGlobalProtection() {
diff --git a/client/src/components/App/index.js b/client/src/components/App/index.js
index f77097a0..6489649c 100644
--- a/client/src/components/App/index.js
+++ b/client/src/components/App/index.js
@@ -65,8 +65,7 @@ class App extends Component {
 
     render() {
         const { dashboard, encryption } = this.props;
-        const updateAvailable =
-            !dashboard.processingVersions && dashboard.isCoreRunning && dashboard.isUpdateAvailable;
+        const updateAvailable = dashboard.isCoreRunning && dashboard.isUpdateAvailable;
 
         return (
             <HashRouter hashType="noslash">
diff --git a/client/src/components/Dashboard/index.js b/client/src/components/Dashboard/index.js
index 2509b7c7..ad207ba0 100644
--- a/client/src/components/Dashboard/index.js
+++ b/client/src/components/Dashboard/index.js
@@ -50,8 +50,26 @@ class Dashboard extends Component {
             dashboard.processingClients ||
             dashboard.processingTopStats;
 
-        const refreshFullButton = <button type="button" className="btn btn-outline-primary btn-sm" onClick={() => this.getAllStats()}><Trans>refresh_statics</Trans></button>;
-        const refreshButton = <button type="button" className="btn btn-outline-primary btn-sm card-refresh" onClick={() => this.getAllStats()} />;
+        const refreshFullButton = (
+            <button
+                type="button"
+                className="btn btn-outline-primary btn-sm"
+                onClick={() => this.getAllStats()}
+            >
+                <Trans>refresh_statics</Trans>
+            </button>
+        );
+        const refreshButton = (
+            <button
+                type="button"
+                className="btn btn-icon btn-outline-primary btn-sm"
+                onClick={() => this.getAllStats()}
+            >
+                <svg className="icons">
+                    <use xlinkHref="#refresh" />
+                </svg>
+            </button>
+        );
 
         return (
             <Fragment>
diff --git a/client/src/components/Header/Header.css b/client/src/components/Header/Header.css
index 35e8a3cc..759594ac 100644
--- a/client/src/components/Header/Header.css
+++ b/client/src/components/Header/Header.css
@@ -75,7 +75,11 @@
 }
 
 .nav-version__value {
+    max-width: 110px;
     font-weight: 600;
+    text-overflow: ellipsis;
+    white-space: nowrap;
+    overflow: hidden;
 }
 
 .nav-version__link {
@@ -85,6 +89,11 @@
     cursor: pointer;
 }
 
+.nav-version__text {
+    display: flex;
+    justify-content: flex-end;
+}
+
 .header-brand-img {
     height: 32px;
 }
diff --git a/client/src/components/Header/Version.js b/client/src/components/Header/Version.js
index be2158e9..6ac4f1ab 100644
--- a/client/src/components/Header/Version.js
+++ b/client/src/components/Header/Version.js
@@ -4,12 +4,26 @@ import { Trans, withNamespaces } from 'react-i18next';
 
 import { getDnsAddress } from '../../helpers/helpers';
 
-function Version(props) {
-    const { dnsVersion, dnsAddresses, dnsPort } = props;
+const Version = (props) => {
+    const {
+        dnsVersion, dnsAddresses, dnsPort, processingVersion, t,
+    } = props;
+
     return (
         <div className="nav-version">
             <div className="nav-version__text">
-                <Trans>version</Trans>: <span className="nav-version__value">{dnsVersion}</span>
+                <Trans>version</Trans>:&nbsp;<span className="nav-version__value" title={dnsVersion}>{dnsVersion}</span>
+                <button
+                    type="button"
+                    className="btn btn-icon btn-icon-sm btn-outline-primary btn-sm ml-2"
+                    onClick={() => props.getVersion(true)}
+                    disabled={processingVersion}
+                    title={t('check_updates_now')}
+                >
+                    <svg className="icons">
+                        <use xlinkHref="#refresh" />
+                    </svg>
+                </button>
             </div>
             <div className="nav-version__link">
                 <div className="popover__trigger popover__trigger--address">
@@ -17,20 +31,23 @@ function Version(props) {
                 </div>
                 <div className="popover__body popover__body--address">
                     <div className="popover__list">
-                        {dnsAddresses
-                            .map(ip => <li key={ip}>{getDnsAddress(ip, dnsPort)}</li>)
-                        }
+                        {dnsAddresses.map(ip => (
+                            <li key={ip}>{getDnsAddress(ip, dnsPort)}</li>
+                        ))}
                     </div>
                 </div>
             </div>
         </div>
     );
-}
+};
 
 Version.propTypes = {
     dnsVersion: PropTypes.string.isRequired,
     dnsAddresses: PropTypes.array.isRequired,
     dnsPort: PropTypes.number.isRequired,
+    getVersion: PropTypes.func.isRequired,
+    processingVersion: PropTypes.bool.isRequired,
+    t: PropTypes.func.isRequired,
 };
 
 export default withNamespaces()(Version);
diff --git a/client/src/components/Header/index.js b/client/src/components/Header/index.js
index e2d47fe6..07e64241 100644
--- a/client/src/components/Header/index.js
+++ b/client/src/components/Header/index.js
@@ -23,7 +23,7 @@ class Header extends Component {
     };
 
     render() {
-        const { dashboard } = this.props;
+        const { dashboard, getVersion, location } = this.props;
         const { isMenuOpen } = this.state;
         const badgeClass = classnames({
             'badge dns-status': true,
@@ -51,7 +51,7 @@ class Header extends Component {
                             </div>
                         </div>
                         <Menu
-                            location={this.props.location}
+                            location={location}
                             isMenuOpen={isMenuOpen}
                             toggleMenuOpen={this.toggleMenuOpen}
                             closeMenu={this.closeMenu}
@@ -59,7 +59,8 @@ class Header extends Component {
                         {!dashboard.processing &&
                             <div className="col col-sm-6 col-lg-3">
                                 <Version
-                                    { ...this.props.dashboard }
+                                    { ...dashboard }
+                                    getVersion={getVersion}
                                 />
                             </div>
                         }
@@ -71,8 +72,9 @@ class Header extends Component {
 }
 
 Header.propTypes = {
-    dashboard: PropTypes.object,
-    location: PropTypes.object,
+    dashboard: PropTypes.object.isRequired,
+    location: PropTypes.object.isRequired,
+    getVersion: PropTypes.func.isRequired,
 };
 
 export default withNamespaces()(Header);
diff --git a/client/src/components/Settings/Settings.css b/client/src/components/Settings/Settings.css
index 7e410a0c..48acf4eb 100644
--- a/client/src/components/Settings/Settings.css
+++ b/client/src/components/Settings/Settings.css
@@ -88,3 +88,10 @@
     width: 30px;
     height: 30px;
 }
+
+.btn-icon-sm {
+    width: 23px;
+    height: 23px;
+    min-width: 23px;
+    padding: 5px;
+}
diff --git a/client/src/components/ui/Card.css b/client/src/components/ui/Card.css
index 33b69c2d..6794a791 100644
--- a/client/src/components/ui/Card.css
+++ b/client/src/components/ui/Card.css
@@ -33,21 +33,6 @@
     text-align: center;
 }
 
-.card-refresh {
-    height: 26px;
-    width: 26px;
-    background-size: 14px;
-    background-position: center;
-    background-repeat: no-repeat;
-    background-image: url("");
-}
-
-.card-refresh:hover,
-.card-refresh:not(:disabled):not(.disabled):active,
-.card-refresh:focus:active {
-    background-image: url("");
-}
-
 .card-title-stats {
     font-size: 13px;
     color: #9aa0ac;
diff --git a/client/src/components/ui/Icons.js b/client/src/components/ui/Icons.js
index 58a5af9f73a69b4456df732ccc112a723c4ef095..92a9efa402d9cc9c51186d493afb37274276e9d7 100644
GIT binary patch
delta 128
zcmewpIyYj&b*;(k6-C*LQqzi3i!&zos&uFL8W}5?l$mAdn)w<k7#Wo5nq`<N>D$>9
zB$i|-q}VF?8ta)FDp)33Dp)EQ0HL9Rp^2V_sjjh}v6&B0ub}}@pP@;PiJqB>f(cO2
U5vI}rXq=vjrGla9<bzt)0KT;$#{d8T

delta 11
ScmbOm@h5b{b*;%iv{eBn83oe-

diff --git a/client/src/reducers/index.js b/client/src/reducers/index.js
index 17d82608..94301e25 100644
--- a/client/src/reducers/index.js
+++ b/client/src/reducers/index.js
@@ -137,6 +137,7 @@ const dashboard = handleActions({
                 newVersion,
                 canAutoUpdate,
                 isUpdateAvailable: true,
+                processingVersion: false,
             };
             return newState;
         }

From d2258cb66de32092f145f2803a7be3d7869970f2 Mon Sep 17 00:00:00 2001
From: Simon Zolin <s.zolin@adguard.com>
Date: Thu, 27 Jun 2019 10:52:45 +0300
Subject: [PATCH 3/3] * openapi.yaml: update /version.json

---
 openapi/openapi.yaml | 17 ++++++++++++++++-
 1 file changed, 16 insertions(+), 1 deletion(-)

diff --git a/openapi/openapi.yaml b/openapi/openapi.yaml
index 385ac057..063b85c7 100644
--- a/openapi/openapi.yaml
+++ b/openapi/openapi.yaml
@@ -135,11 +135,19 @@ paths:
                             "192.168.1.104:53535": "Couldn't communicate with DNS server"
 
     /version.json:
-        get:
+        post:
             tags:
                 - global
             operationId: getVersionJson
             summary: 'Gets information about the latest available version of AdGuard'
+            consumes:
+            - application/json
+            parameters:
+            - in: "body"
+              name: "body"
+              required: true
+              schema:
+                $ref: "#/definitions/GetVersionRequest"
             produces:
                 - 'application/json'
             responses:
@@ -994,6 +1002,13 @@ definitions:
                 example:
                     - '||example.org^'
                     - '||example.com^'
+    GetVersionRequest:
+        type: "object"
+        description: "/version.json request data"
+        properties:
+            recheck_now:
+                description: "If false, server will check for a new version data only once in several hours"
+                type: "boolean"
     VersionInfo:
         type: "object"
         description: "Information about the latest available version of AdGuard Home"