Skip to content
Snippets Groups Projects
Commit b079f2f0 authored by Cam Saül's avatar Cam Saül
Browse files

Fix intermittent BigQuery CI failures :unamused:

parent c9d606fa
No related branches found
No related tags found
No related merge requests found
......@@ -35,9 +35,13 @@
(defn- execute
"`execute` REQUEST, and catch any `GoogleJsonResponseException` is
throws, converting them to `ExceptionInfo` and rethrowing them."
throws, converting them to `ExceptionInfo` and rethrowing them.
This automatically retries any failed requests a single time."
[^AbstractGoogleClientRequest request]
(try (.execute request)
;; automatically retry each request at least once if it fails
(try (u/auto-retry 1
(.execute request))
(catch GoogleJsonResponseException e
(let [^GoogleJsonError error (.getDetails e)]
(throw (ex-info (or (.getMessage error)
......@@ -181,8 +185,7 @@
"STRING" identity
"TIMESTAMP" parse-timestamp-str})
(def ^:private ^:const query-timeout-error-message "Query timed out.")
(def ^:private ^:const query-default-timeout-seconds 20)
(def ^:private ^:const query-default-timeout-seconds 15)
(defn- post-process-native
([^QueryResponse response]
......@@ -192,7 +195,7 @@
;; 99% of the time by the time this is called `.getJobComplete` will return `true`. On the off chance it doesn't, wait a few seconds for the job to finish.
(do
(when (zero? timeout-seconds)
(throw (ex-info query-timeout-error-message (into {} response))))
(throw (ex-info "Query timed out." (into {} response))))
(Thread/sleep 1000)
(post-process-native response (dec timeout-seconds)))
;; Otherwise the job *is* complete
......@@ -211,17 +214,10 @@
(parser v)))))}))))
(defn- process-native* [database query-string]
;; Automatically retry the query a single time if it timed out
(let [do-query (fn [] (post-process-native (execute-query database query-string)))]
(try
(do-query)
(catch clojure.lang.ExceptionInfo e
(if (= (.getMessage e) query-timeout-error-message)
;; If this was a timeout error, retry
(do (log/info "Query timed out, retrying...")
(do-query))
;; Otherwise re-throw exception
(throw e))))))
;; automatically retry the query if it times out or otherwise fails. This is on top of the auto-retry added by `execute` so operations going through `process-native*` may be
;; retried up to 3 times.
(u/auto-retry 1
(post-process-native (execute-query database query-string))))
(defn- process-native [{database-id :database, {native-query :query} :native}]
(process-native* (Database database-id) native-query))
......
......@@ -602,3 +602,22 @@
(if (contains? slugify-valid-chars c)
c
\_)))))
(defn do-with-auto-retries
"Execute F, a function that takes no arguments, and return the results.
If F fails with an exception, retry F up to NUM-RETRIES times until it succeeds.
Consider using the `auto-retry` macro instead of calling this function directly."
[num-retries f]
(if (<= num-retries 0)
(f)
(try (f)
(catch Throwable _
(do-with-auto-retries (dec num-retries) f)))))
(defmacro auto-retry
"Execute BODY and return the results.
If BODY fails with an exception, retry execution up to NUM-RETRIES times until it succeeds."
{:style/indent 1}
[num-retries & body]
`(do-with-auto-retries ~num-retries (fn [] ~@body)))
......@@ -90,13 +90,9 @@
(println (u/format-color 'blue "Created BigQuery table '%s.%s'." dataset-id table-id)))
(defn- table-row-count ^Integer [^String dataset-id, ^String table-id]
(let [f (fn [] (first (first (:rows (post-process-native (execute (.query (.jobs bigquery) project-id
(doto (QueryRequest.)
(.setQuery (format "SELECT COUNT(*) FROM [%s.%s]" dataset-id table-id))))))))))]
;; automatically retry the query in case it fails / times out
(try (f)
(catch Throwable _
(f)))))
(first (first (:rows (post-process-native (execute (.query (.jobs bigquery) project-id
(doto (QueryRequest.)
(.setQuery (format "SELECT COUNT(*) FROM [%s.%s]" dataset-id table-id))))))))))
;; This is a dirty HACK
(defn- ^DateTime timestamp-korma-form->GoogleDateTime
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment