Fix plot annotations
[notebooks.git] / old / fmda_kf_local.ipynb
blob2c632bb52cd0067d4e7d34008f964e74003ffc5f
2  "cells": [
3   {
4    "cell_type": "markdown",
5    "metadata": {
6     "id": "khOTxJsYc91W"
7    },
8    "source": [
9     "# Kalman Filtering for Fuel Moisture\n",
10     "## Jan Mandel, University of Colorado Denver\n",
11     "### Instructor for MATH 4779 - Math Clinic\n",
12     "### December 2021"
13    ]
14   },
15   {
16    "cell_type": "markdown",
17    "metadata": {
18     "id": "sXaqfI-EdCEk"
19    },
20    "source": [
21     "## Abstract"
22    ]
23   },
24   {
25    "cell_type": "markdown",
26    "metadata": {
27     "id": "ZbtSiYm4dF7B"
28    },
29    "source": [
30     "''Fuel moisture is an important factor of the spread of wildland fires. Some weather stations have fuel moisture sensors and data are available online. We review a simple model of fuel moisture from atmospheric conditions, and show how to adjust the model using the weather station data."
31    ]
32   },
33   {
34    "cell_type": "markdown",
35    "metadata": {
36     "id": "eZ6dfHlZ63j1"
37    },
38    "source": [
39     "## Table of contents"
40    ]
41   },
42   {
43    "cell_type": "markdown",
44    "metadata": {
45     "id": "WHIgN2uZ689b"
46    },
47    "source": [
48     "1 Introduction\n",
49     "\n",
50     "2 Background\n",
51     "\n",
52     "2.1 Imports\n",
53     "\n",
54     "2.2 Kalman filter\n",
55     "\n",
56     "2.2.1 Overview\n",
57     "\n",
58     "2.2.2 Formulation\n",
59     "\n",
60     "2.2.3 A Kalman filter tester\n",
61     "\n",
62     "2.3 Fuel moisture model\n",
63     "\n",
64     "2.3.1 A simple time lag model\n",
65     "\n",
66     "2.3.1 Fuel moisture model with drying equilibrium, wetting equilibrium, and rain\n",
67     "\n",
68     "3 Methods\n",
69     "\n",
70     "3.1 Kalman filter demonstration on the simple model\n",
71     "\n",
72     "3.1.1 Creating synthetic data\n",
73     "\n",
74     "3.1.2 Running the Kalman filter\n",
75     "\n",
76     "3.2 Acquisition and preprocessing of real data\n",
77     "\n",
78     "3.2.1 Acquisition of fuel moisture observations\n",
79     "\n",
80     "3.2.2 Acquisition of weather data\n",
81     "\n",
82     "3.2.3 Preprocessing and visualization of the weather data\n",
83     "\n",
84     "4 Results\n",
85     "\n",
86     "4.1 Kalman filter with fuel moisture observations, followed by forecasting\n",
87     "\n",
88     "4.2 Model with an augmented state\n",
89     "\n",
90     "4.3 Kalman filter on the augmented model\n",
91     "\n",
92     "4.4 A comment on the information flow in the Kalman filter and in neural networks\n",
93     "\n",
94     "5. Conclusion\n",
95     "\n",
96     "Contributions of Authors\n",
97     "\n",
98     "Acknowledgements\n",
99     "\n",
100     "References\n",
101     "\n",
102     "\n",
103     "\n",
104     "\n",
105     "\n",
106     "\n",
107     "\n"
108    ]
109   },
110   {
111    "cell_type": "markdown",
112    "metadata": {
113     "id": "ZFafUPCTO1N1"
114    },
115    "source": [
116     "## 1 Introduction"
117    ]
118   },
119   {
120    "cell_type": "markdown",
121    "metadata": {
122     "id": "4_RcdWybPFks"
123    },
124    "source": [
125     "The Kalman filter is at the foundation of many technologies in daily use, from GPS to weather forecasting. No model is completely accurate. Think space navigation: the movement of a Apollo 13 between the moon and the earth, subject to gravitational forces and propulsion, with the position ascertained by visual measurements. No matter how accurate the model of spacecraft motion is, the measurements are always burdened with noise. The idea of Kalman filter is to evolve a quantification of the of the state (here, positin and velocity of the spacecraft) in the form of a covariance matrix, and, using an estimate of the uncertainty of the data, adjust the state to split the difference every time measurements are taken. \n",
126     "\n",
127     "Here, we use the Kalman filter to estimate the evolution of fuel (dead wood) moisture content from a simple theoretical model, adjusting the state of the model hourly for measurements from fuel moisture a sensor in a wood stick exposed to the elements. This is needed for forecasting of wildfire progress; for this purpose, we also want to have the filter adjust the model from the data, so that it gives more accurate data for future when we only have hourly weather forecast but no actual data - because the future has not happened yet. "
128    ]
129   },
130   {
131    "cell_type": "markdown",
132    "metadata": {
133     "id": "M2kbwDPBTB7A"
134    },
135    "source": [
136     "## 2 Background"
137    ]
138   },
139   {
140    "cell_type": "markdown",
141    "metadata": {
142     "id": "ar1BbXac49hO"
143    },
144    "source": [
145     "In this section, we take care of preliminaries: we install some packages we need, and then proceed with the Kalman filter."
146    ]
147   },
148   {
149    "cell_type": "markdown",
150    "metadata": {
151     "id": "_5F5CuRqc91X"
152    },
153    "source": [
154     "### 2.1 Imports"
155    ]
156   },
157   {
158    "cell_type": "markdown",
159    "metadata": {
160     "id": "K6sWUMf0c91Y"
161    },
162    "source": [
163     "We may need the pygrib package to read weather data, but pygrib requires current numpy while google colab is using an old numpy version for compatibility with tensorflow. We will upgrade numpy and restart the runtime then the notebook will need to be run again. If numpy is current, we just download and import packages we need."
164    ]
165   },
166   {
167    "cell_type": "markdown",
168    "metadata": {
169     "id": "X9rvlymMZdJg"
170    },
171    "source": [
172     "### 2.2 Kalman filter"
173    ]
174   },
175   {
176    "cell_type": "markdown",
177    "metadata": {
178     "id": "x5E2UE3F5gf2"
179    },
180    "source": [
181     "#### 2.2.1 Overview"
182    ]
183   },
184   {
185    "cell_type": "markdown",
186    "metadata": {
187     "id": "NPgTHlCLAlA-"
188    },
189    "source": [
190     "The Kalman filter provides an estimate $u$ of the time evolution of some unknown process, called \"nature\" or \"truth\". We do not know with certainty what the nature is, but we can observe it at regular intervals (steps) with some error. In each step, model $F$ advances the model state $u$ in time, $ u \\leftarrow F(u)$, and attempts to reconcile the state with an observation $d$ of the true state, so $u \\approx d$. The filter modifies the model state $u$ to balance the uncertainty in the model and the data (this is called *analysis*) and the cycle continues. For that purpose, the filter evolves also an estimate of the uncertainly of the model.\n",
191     "\n",
192     "More generally, instead of $u \\approx d$, only a part of the state is observed, and $Hu \\approx d$ where $H$ is a matrix, or observation function. Basically, $Hu$ is what the data would be if the model was completely accurate. \n",
193     "\n",
194     "See Kalman (1960) for the original publication, Kalnay (2003) for a gentle introduction, and the [Wikipedia article](https://en.wikipedia.org/wiki/Extended_Kalman_filter)."
195    ]
196   },
197   {
198    "cell_type": "markdown",
199    "metadata": {
200     "id": "y6j34L5s5pEL"
201    },
202    "source": [
203     "#### 2.2.2 Formulation\n",
204     "\n",
205     "---\n",
206     "\n"
207    ]
208   },
209   {
210    "cell_type": "markdown",
211    "metadata": {
212     "id": "b3GZW5vP5_o8"
213    },
214    "source": [
215     "We present the Kalman filter in perhaps the most used form, as extended to nonlinear models.\n",
216     " Consider a discrete time model of some natural\n",
217     "process. At time step $k$, the model has state $u_{k}\\in\\mathbb{R}^{n}$, which\n",
218     "can be approximated from the previous step $u_{k-1}$ by applying the model\n",
219     "$\\mathcal{M}$ to get a forecast $u_{k}^{f}=\\mathcal{M}\\left(  u_{k-1}\\right)\n",
220     "$. We model uncertainty in the model itself by adding normally distributed\n",
221     "noise with mean zero and covariance $Q$ to the uncertainty of $u_{k}^{f}$. We\n",
222     "also need to estimate now the uncertainty in the previous state $u_{k-1}$\n",
223     "propagates to the uncertainty of the forecast $u_{k}^{f}$. So, assume that the\n",
224     "model is differentiable and quantify the uncertainty of the state by a\n",
225     "covariance matrix. That is,  assume that at step $k-1$, the state has\n",
226     "(approximately) normal distribution with mean $u_{k-1}$ and covariance\n",
227     "$P_{k-1}$. Using the Taylor expansion of order $1$ of the model operator at\n",
228     "$u_{k-1}$, $\\mathcal{M}\\left(  u\\right)  \\approx\\mathcal{M}\\left(\n",
229     "u_{k-1}\\right)  +\\mathcal{M}^{\\prime}\\left(  u_{k-1}\\right)  \\left(\n",
230     "u-u_{k-1}\\right)  $, where $\\mathcal{M}^{\\prime}\\left(  u_{k-1}\\right)  $ is\n",
231     "the Jacobian matrix of $\\mathcal{M}$ at $u_{k-1}$. It can be shown that the\n",
232     "forecast has then (approximately)\\ normal distribution with mean and\n",
233     "covariance\n",
234     "$$\n",
235     "u_{k}^{f}=\\mathcal{M}\\left(  u_{k-1}\\right)  ,\\ P_{k}^{f}=\\mathcal{M}\\left(\n",
236     "u_{k-1}\\right)  P_{k-1}\\mathcal{M}^{\\prime}\\left(  u_{k-1}\\right)  +Q\n",
237     "$$\n",
238     "At time $k$, we also have an observation $d_{k}\\approx Hu_{k}$, where $H$ is a\n",
239     "given observation operator, and we want to find $u_{k}$ so that both\n",
240     "$$\n",
241     "u_{k}\\approx u_{k}^{f}\\text{ and }d_{k}\\approx Hu_{k}.\n",
242     "$$\n",
243     "We quantify the uncertainly of the error of observation $d_{k}$ by a covariance\n",
244     "matrix $R$: assume that the observation error has normal probability\n",
245     "distribution with a known covariance $R$. Then, the likelihood of state $u$ is\n",
246     "proportional to $e^{-\\left\\Vert d_{k}-Hu\\right\\Vert _{R^{-1}}^{2}/2}$, where\n",
247     "we used the notation for the norm $\\left\\Vert v\\right\\Vert _{A}%\n",
248     "=\\left(v^{\\top}Av\\right)^{1/2}$ induced by a positive definite matrix $A$. Similarly, we quantify the\n",
249     "uncertainty of the state by a covariance matrix $P_{k}$. That is, the forecast\n",
250     "state has (approximately) normal distribution with mean $u_{k}^{f}$  and covariance\n",
251     "$P_{k}^{f}$. From the Bayes theorem of statistics, the probability distribution\n",
252     "of the state after taking the data into account has density\n",
253     "$$\n",
254     "p_{k}\\left(  u\\right) \\propto e^\\frac{-\\left\\Vert d_{k}\n",
255     "-Hu\\right\\Vert_{R^{-1}}^{2}}{2}e^\\frac{-\\left\\Vert u-u_{k}^{f}\\right\\Vert _{\n",
256     "{P_{k}^f}^{-1}  }^{2}}{2}%\n",
257     "$$\n",
258     "where $\\propto$ means proportional.\n",
259     "Note that the probability density at $u$ is maximal when $\\left\\Vert\n",
260     "d_{k}-Hu\\right\\Vert _{R^{-1}}^{2}+\\left\\Vert u-u_{k}\\right\\Vert _{{P_{k}^{f}}^{-1}}^{2}$\n",
261     " is minimal, which quantifies the statement that $d_{k}\\approx\n",
262     "Hu_{k}$ and $u\\approx u_{k}^{f}$.  By a direct computation completing the\n",
263     "square and using the Sherman-Morrison-Woodbury formula, \n",
264     "$$p_{k}\\left(\n",
265     "\t\tu\n",
266     "\t   \\right) \\propto \n",
267     "e^{-\\frac{\n",
268     "\t\\left\\Vert u-u_{k\n",
269     "\t         }\n",
270     "\t\\right\\Vert_\n",
271     "\t\t{P_{k\n",
272     "\t\t      }^{-1}\n",
273     "\t\t}^{2}\n",
274     "\t}\n",
275     "\t{2}},\n",
276     "$$ \n",
277     "which is the density of the normal distribution with the mean\n",
278     "$$\n",
279     "u_{k}^{f}=u_{k}^{f}+K_{k}(d-Hu_{k}^{f}),\\ \\text{where }K_{k}=P_{k}%\n",
280     "^{f}H^{\\mathrm{T}}(HP_{k}^{f}H^{\\mathrm{T}}+R)^{-1}%\n",
281     "$$\n",
282     "and covariance\n",
283     "$$\n",
284     "P_{k}=\\left(  \\left(  P_{k}^{f}\\right)  ^{-1}+H^{\\mathrm{T}}R^{-1}H\\right)\n",
285     "^{-1}=(I-KH)P_{k}^{f}.\n",
286     "$$\n",
287     "\n",
288     "These are the equations of the extended Kalman filter. The original Kalman (1960) filter was\n",
289     "formulated for a linear process. The extension to the\n",
290     "nonlinear case made broad array of applications possible, including the Apollo spacecraft naviation (McGee and Schmidt, 1966),  and is\n",
291     "still a de-facto standard in navigation and GPS.\n"
292    ]
293   },
294   {
295    "cell_type": "code",
296    "execution_count": null,
297    "metadata": {
298     "id": "-bvUtJ_OLwQA"
299    },
300    "outputs": [],
301    "source": [
302     "import numpy as np\n",
303     "def ext_kf(u,P,F,Q=0,d=None,H=None,R=None):\n",
304     "  \"\"\"\n",
305     "  One step of the extended Kalman filter. \n",
306     "  If there is no data, only advance in time.\n",
307     "  :param u:   the state vector, shape n\n",
308     "  :param P:   the state covariance, shape (n,n)\n",
309     "  :param F:   the model function, args vector u, returns F(u) and Jacobian J(u)\n",
310     "  :param Q:   the process model noise covariance, shape (n,n)\n",
311     "  :param d:   data vector, shape (m). If none, only advance in time\n",
312     "  :param H:   observation matrix, shape (m,n)\n",
313     "  :param R:   data error covariance, shape (n,n)\n",
314     "  :return ua: the analysis state vector, shape (n)\n",
315     "  :return Pa: the analysis covariance matrix, shape (n,n)\n",
316     "  \"\"\"\n",
317     "  def d2(a):\n",
318     "    return np.atleast_2d(a) # convert to at least 2d array\n",
319     "\n",
320     "  def d1(a):\n",
321     "    return np.atleast_1d(a) # convert to at least 1d array\n",
322     "\n",
323     "  # forecast\n",
324     "  uf, J  = F(u)          # advance the model state in time and get the Jacobian\n",
325     "  uf = d1(uf)            # if scalar, make state a 1D array\n",
326     "  J = d2(J)              # if scalar, make jacobian a 2D array\n",
327     "  P = d2(P)              # if scalar, make Jacobian as 2D array\n",
328     "  Pf  = d2(J.T @ P) @ J + Q  # advance the state covariance Pf = J' * P * J + Q\n",
329     "  # analysis\n",
330     "  if d is None or not d.size :  # no data, no analysis\n",
331     "    return uf, Pf\n",
332     "  # K = P H' * inverse(H * P * H' + R) = (inverse(H * P * H' + R)*(H P))'\n",
333     "  H = d2(H)\n",
334     "  HP  = d2(H @ P)            # precompute a part used twice  \n",
335     "  K   = d2(np.linalg.solve( d2(HP @ H.T) + R, HP)).T  # Kalman gain\n",
336     "  # print('H',H)\n",
337     "  # print('K',K)\n",
338     "  res = d1(H @ d1(uf) - d)          # res = H*uf - d\n",
339     "  ua = uf - K @ res # analysis mean uf - K*res\n",
340     "  Pa = Pf - K @ d2(H @ P)        # analysis covariance\n",
341     "  return ua, d2(Pa)\n"
342    ]
343   },
344   {
345    "cell_type": "markdown",
346    "metadata": {
347     "id": "Uvsbbv2XZ2Hd"
348    },
349    "source": [
350     "#### 2.2.3 A Kalman filter tester"
351    ]
352   },
353   {
354    "cell_type": "markdown",
355    "metadata": {
356     "id": "gcmGBqPOU1e5"
357    },
358    "source": [
359     "It is a very good idea to make write a simple tester for every piece of code. How else would we know it actually works, and that something basic did not get broken inadvertently, perhaps as a side effect of changing something else? A simple tester may save a great deal of time trying to debug cryptic errors later. And, what better place for a tester that right after the code it is testing so that it gets run every time?"
360    ]
361   },
362   {
363    "cell_type": "code",
364    "execution_count": null,
365    "metadata": {
366     "id": "OsOqvQk6ZXZV"
367    },
368    "outputs": [],
369    "source": [
370     "# a basic ext_kf test\n",
371     "import numpy as np\n",
372     "u = [1,\n",
373     "     2]\n",
374     "P = [[2 , -1],\n",
375     "    [-1 , 2]]\n",
376     "A = [ [1 ,2],\n",
377     "      [3 ,4]]\n",
378     "u = np.array(u)      \n",
379     "Q = np.array([[1,0],[0,1]])\n",
380     "A = np.array(A)\n",
381     "def fun(u):\n",
382     "  return A @ u, A\n",
383     "F = lambda u: fun(u)\n",
384     "H = [[1, 0],\n",
385     "     [0, 1]]\n",
386     "d = [2,\n",
387     "    3]\n",
388     "R = [[2, 0],\n",
389     "    [0, 2]]\n",
390     "H = np.array(H)      \n",
391     "d = np.array(d)\n",
392     "R = np.array(R)\n",
393     "ua,Pa = ext_kf(u,P,F,Q)\n",
394     "print('ua=',ua)\n",
395     "print('Pa=',Pa)\n",
396     "ua,Pa = ext_kf(u,P,F,Q,d,H,R)\n",
397     "print('ua=',ua)\n",
398     "print('Pa=',Pa)\n"
399    ]
400   },
401   {
402    "cell_type": "markdown",
403    "metadata": {
404     "id": "A9ZpmNcdRpmp"
405    },
406    "source": [
407     "### 2.3  Fuel moisture models\n",
408     "\n",
409     "\n"
410    ]
411   },
412   {
413    "cell_type": "markdown",
414    "metadata": {
415     "id": "eZL8gN7ISGVh"
416    },
417    "source": [
418     "#### 2.3.1 A simple fuel moisture model"
419    ]
420   },
421   {
422    "cell_type": "markdown",
423    "metadata": {
424     "id": "1XvOC4kYSQgH"
425    },
426    "source": [
427     "First consider a simplified fuel moisture model without considering the effect of rain.\n",
428     "The evolution of fuel moisture content $m(t)$ is modeled by the time-lag differential equation on interval $\\left[\n",
429     "t_{0},t_{1}\\right]  $,\n",
430     "$$\n",
431     "\\frac{dm}{dt}=\\frac{E-m(t)}{T},\\quad m(t_{0})=m_{0}.\n",
432     "$$\n",
433     "where the initial fuel moisture content $m_{0}=m\\left(  t_{0}\\right)  $ is the\n",
434     "input, and $m_{1}=m(t_{1})$ is the output. Tnus, $m_1=F(m_0)$. The parameters of the model are the\n",
435     "fuel moisture equilibrium $E$, assumed to be constant over the interval $\\left[\n",
436     "t_{0},t_{1}\\right]  $, NS the characteristic decay time $T$. \n",
437     "\n",
438     "We can build the general model later by calling this simple model with different\n",
439     "equilibria and time constants (drying, wetting, rain).\n",
440     "\n",
441     "Since $E$ is constant in time, the solution can be found\n",
442     "analytically,\n",
443     "$$\n",
444     "m\\left(  t\\right)  =E+\\left(  m_{0}-E\\right)  e^{-t/T}%\n",
445     "$$\n",
446     "For convenience, we use $T_{1}=1/T$ instead of $T$, and the model becomes\n",
447     "$$\n",
448     "m_{1}=E+\\left(  m_{0}-E\\right)  e^{-\\left(  t_{1}-t_{0}\\right)  T_{1}}%\n",
449     "$$\n",
450     "In the extended Kalman filter, we will need the partial derivatives of $m_{1}$\n",
451     "with respect to the input and the parameters. Compute\n",
452     "$$\n",
453     "\\frac{dm_{1}}{d_{m0}}=e^{-\\left(  t_{1}-t_{0}\\right)  T_{1}}\n",
454     "$$\n",
455     "$$\n",
456     "\\frac{dm_{1}}{dE}=1-e^{-\\left(  t_{1}-t_{0}\\right)  T_{1}}\n",
457     "$$\n",
458     "$$\n",
459     "\\frac{dm_{1}}{dT_{1}}=-\\left(  m_{0}-E\\right)  \\left(  t_{1}-t_{0}\\right)\n",
460     "e^{-\\left(  t_{1}-t_{0}\\right)  T_{1}}\n",
461     "$$\n",
462     "At the moment, we need only ${dm_{1}}/{dm_{0}}$ but we put in the code all partials for possible use in future.\n"
463    ]
464   },
465   {
466    "cell_type": "code",
467    "execution_count": null,
468    "metadata": {
469     "id": "wuVIAGLiSeR8"
470    },
471    "outputs": [],
472    "source": [
473     "import numpy as np\n",
474     "def model_decay(m0,E,partials=0,T1=0.1,tlen=1):  \n",
475     "  # Arguments: \n",
476     "  #   m0          fuel moisture content at start dimensionless, unit (1)\n",
477     "  #   E           fuel moisture eqilibrium (1)\n",
478     "  #   partials=0: return m1 = fuel moisture contents after time tlen (1)\n",
479     "  #           =1: return m1, dm0/dm0 \n",
480     "  #           =2: return m1, dm1/dm0, dm1/dE\n",
481     "  #           =3: return m1, dm1/dm0, dm1/dE dm1/dT1   \n",
482     "  #   T1          1/T, where T is the time constant approaching the equilibrium\n",
483     "  #               default 0.1/hour\n",
484     "  #   tlen        the time interval length, default 1 hour\n",
485     "\n",
486     "  exp_t = np.exp(-tlen*T1)                  # compute this subexpression only once\n",
487     "  m1 = E + (m0 - E)*exp_t                   # the solution at end\n",
488     "  if partials==0:\n",
489     "    return m1\n",
490     "  dm1_dm0 = exp_t\n",
491     "  if partials==1:\n",
492     "    return m1, dm1_dm0          # return value and Jacobian\n",
493     "  dm1_dE = 1 - exp_t      \n",
494     "  if partials==2:\n",
495     "     return m1, dm1_dm0, dm1_dE \n",
496     "  dm1_dT1 = -(m0 - E)*tlen*exp_t            # partial derivative dm1 / dT1\n",
497     "  if partials==3:\n",
498     "    return m1, dm1_dm0, dm1_dE, dm1_dT1       # return value and all partial derivatives wrt m1 and parameters\n",
499     "  raise('Bad arg partials')\n",
500     "  "
501    ]
502   },
503   {
504    "cell_type": "markdown",
505    "metadata": {
506     "id": "dOARZlj-RUCi"
507    },
508    "source": [
509     "#### 2.3.2 Fuel moisture model with drying equilibrium, wetting equilibrium, and rain"
510    ]
511   },
512   {
513    "cell_type": "markdown",
514    "metadata": {
515     "id": "AJp6FTpTSx5B"
516    },
517    "source": [
518     "Here is a little more realistic fuel moisture model from Mandel et al. (2004). A rain-wetting lag time $t_{\\mathrm{r}}$ is reached for heavy rain only\n",
519     "asymptotically, when the rain intensity $r$ (mm/h) is\n",
520     "large:\n",
521     "$$\n",
522     "\\frac{\\mathrm{d}m}{\\mathrm{d}t}=\\frac{S-m}{t_{\\mathrm{r}}}\\left(1-\\exp\\left(-\\frac{r-r_0}{r_{\\mathrm{s}}}\n",
523     "\\right)  \\right),\\ \\text{if}\\ r>r_0, \n",
524     "$$\n",
525     "where $r_0$ is the threshold rain intensity below which no perceptible\n",
526     "wetting occurs, and $r_{\\mathrm{s}}$ is the saturation rain\n",
527     "intensity. At the saturation rain intensity, $1-1/e\\approx 0.63$ of\n",
528     "the maximal rain-wetting rate is achieved. For 10h fuel, the model takes $S=250\\,{\\%}$,\n",
529     "$t_{\\mathrm{r}}=14$h, $r_0=0.05$mm/h and\n",
530     "$r_{\\mathrm{s}}=8$mm/h. "
531    ]
532   },
533   {
534    "cell_type": "code",
535    "execution_count": null,
536    "metadata": {
537     "id": "ITsKE0psRblG"
538    },
539    "outputs": [],
540    "source": [
541     "### Define model function with drying, wetting, and rain equilibria\n",
542     "\n",
543     "# Parameters\n",
544     "r0 = 0.05                                   # threshold rainfall [mm/h]\n",
545     "rs = 8.0                                    # saturation rain intensity [mm/h]\n",
546     "Tr = 14.0                                   # time constant for rain wetting model [h]\n",
547     "S = 250                                     # saturation intensity [dimensionless]\n",
548     "T = 10.0                                    # time constant for wetting/drying\n",
549     "\n",
550     "def model_moisture(m0,Eqd,Eqw,r,t,partials=0,T=10.0,tlen=1.0):\n",
551     "    # arguments:\n",
552     "    # m0         starting fuel moistureb (%s\n",
553     "    # Eqd        drying equilibrium      (%) \n",
554     "    # Eqw        wetting equilibrium     (%)\n",
555     "    # r          rain intensity          (mm/h)\n",
556     "    # t          time\n",
557     "    # partials = 0, 1, 2\n",
558     "    # returns: same as model_decay\n",
559     "    #   if partials==0: m1 = fuel moisture contents after time 1 hour\n",
560     "    #              ==1: m1, dm1/dm0 \n",
561     "    #              ==2: m1, dm1/dm0, dm1/dE  \n",
562     "    \n",
563     "    if r > r0:\n",
564     "        # print('raining')\n",
565     "        E = S\n",
566     "        T1 =  (1.0 - np.exp(- (r - r0) / rs)) / Tr\n",
567     "    elif m0 <= Eqw: \n",
568     "        # print('wetting')\n",
569     "        E=Eqw\n",
570     "        T1 = 1.0/T\n",
571     "    elif m0 >= Eqd:\n",
572     "        # print('drying')\n",
573     "        E=Eqd\n",
574     "        T1 = 1.0/T\n",
575     "    else: # no change'\n",
576     "        E = m0\n",
577     "        T1=0.0\n",
578     "    exp_t = np.exp(-tlen*T1)\n",
579     "    m1 = E + (m0 - E)*exp_t  \n",
580     "    dm1_dm0 = exp_t\n",
581     "    dm1_dE = 1 - exp_t\n",
582     "    #if t>=933 and t < 940:\n",
583     "    #  print('t,Eqw,Eqd,r,T1,E,m0,m1,dm1_dm0,dm1_dE',\n",
584     "    #        t,Eqw,Eqd,r,T1,E,m0,m1,dm1_dm0,dm1_dE)   \n",
585     "    if partials==0: \n",
586     "        return m1\n",
587     "    if partials==1:\n",
588     "        return m1, dm1_dm0\n",
589     "    if partials==2:\n",
590     "        return m1, dm1_dm0, dm1_dE\n",
591     "    raise('bad partials')"
592    ]
593   },
594   {
595    "cell_type": "markdown",
596    "metadata": {
597     "id": "JDLU3B_jV42l"
598    },
599    "source": [
600     "## 3. Methods"
601    ]
602   },
603   {
604    "cell_type": "markdown",
605    "metadata": {
606     "id": "hLPJT3FcA2a7"
607    },
608    "source": [
609     "### 3.1 Kalman filter demonstration on the simple model"
610    ]
611   },
612   {
613    "cell_type": "markdown",
614    "metadata": {
615     "id": "kIA3X8vluFdd"
616    },
617    "source": [
618     "We demonstrate the Kalman filter for this model on a simple artificial example. The model is solving the differential equation for one hour. The equilibrium $E$ is constant during the hour, but it changes over the day so that it is higher at night and lower during the day, with a 24-hour period.  First, we create the \"truth\" by choosing the equilibrium $E$ and solving the differential aquation every hour, with a small additive noise. The synthetic data is obtained as the values of the \"truth\", with random noise to simulate observation error."
619    ]
620   },
621   {
622    "cell_type": "markdown",
623    "metadata": {
624     "id": "bBv10PTiChhm"
625    },
626    "source": [
627     "#### 3.1.1 Creating synthetic data"
628    ]
629   },
630   {
631    "cell_type": "code",
632    "execution_count": null,
633    "metadata": {
634     "id": "-_pz-wXnCMnP"
635    },
636    "outputs": [],
637    "source": [
638     "import numpy as np, random\n",
639     "days = 20       \n",
640     "hours = days*24\n",
641     "h2 = int(hours/2)\n",
642     "hour = range(hours)\n",
643     "day = np.array(range(hours))/24.\n",
644     "\n",
645     "# artificial equilibrium data\n",
646     "E = np.power(np.sin(np.pi*day),4) # diurnal curve\n",
647     "E = 0.05+0.25*E\n",
648     "# FMC free run\n",
649     "m_f = np.zeros(hours)\n",
650     "m_f[0] = 0.1         # initial FMC\n",
651     "for t in range(hours-1):\n",
652     "  m_f[t+1] = max(0.,model_decay(m_f[t],E[t])  + random.gauss(0,0.005) )\n",
653     "data = m_f + np.random.normal(loc=0,scale=0.02,size=hours)    \n",
654     "\n",
655     "%matplotlib inline\n",
656     "import matplotlib.pyplot as plt \n",
657     "# fig1, ax1 = plt.subplots()\n",
658     "\n",
659     "plt.figure(figsize=(16,4))\n",
660     "plt.plot(hour,E,linestyle='--',c='r',label='Equilibrium')\n",
661     "plt.plot(hour,m_f,linestyle='-',c='k',label='10-h fuel truth')\n",
662     "plt.scatter(hour[:h2],data[:h2],c='b',label='10-h fuel data')\n",
663     "plt.title('Kalman filter example on artificial data')\n",
664     "plt.xlabel('Time (hours)')\n",
665     "plt.ylabel('Fuel moisture content (%)')\n",
666     "plt.legend()\n",
667     " "
668    ]
669   },
670   {
671    "cell_type": "code",
672    "execution_count": null,
673    "metadata": {
674     "id": "ECsvw2KJ7tAX"
675    },
676    "outputs": [],
677    "source": []
678   },
679   {
680    "cell_type": "markdown",
681    "metadata": {
682     "id": "z-3WLAEpD2yJ"
683    },
684    "source": [
685     "#### 3.1.2 Running the Kalman filter"
686    ]
687   },
688   {
689    "cell_type": "markdown",
690    "metadata": {
691     "id": "T4g-RrrYAlBD"
692    },
693    "source": [
694     "We have used the same code for model and for the truth, and run the Kalman filter for 10 days. The graph below shows that the model state was remarkably close to the truth, even if the model is fed only noisy observations. This is because the dynamics of the model and of the truth are the same. After 10 days, we let the model continue without any new data to simulate forecasting the future, and the agreement with the truth was still very good."
695    ]
696   },
697   {
698    "cell_type": "code",
699    "execution_count": null,
700    "metadata": {
701     "id": "_-CjONZkD18n"
702    },
703    "outputs": [],
704    "source": [
705     "import numpy as np\n",
706     "import matplotlib.pyplot as plt \n",
707     "\n",
708     "def kf_example(DeltaE):\n",
709     "  h2 = int(hours/2)\n",
710     "  m = np.zeros(hours)\n",
711     "  m[0]=0.1             # background state  \n",
712     "  P = np.zeros(hours)\n",
713     "  P[0] = 0.03 # background state variance\n",
714     "  Q = np.array([0.02]) # process noise variance\n",
715     "  H = np.array([1.])   # all observed\n",
716     "  R = np.array([0.02]) # data variance\n",
717     "\n",
718     "  for t in range(h2):\n",
719     "    # use lambda construction to pass additional arguments to the model \n",
720     "    m[t+1],P[t+1] = ext_kf(m[t],P[t],lambda u: model_decay(u,E[t]+DeltaE,partials=1),Q,\n",
721     "                    d=data[t],H=H,R=R)\n",
722     "  for t in range(h2,hours - 1):\n",
723     "    m[t+1],P[t+1] = ext_kf(m[t],P[t],lambda u: model_decay(u,E[t]+DeltaE,partials=1))\n",
724     "  \n",
725     "  %matplotlib inline\n",
726     "  plt.figure(figsize=(16,4))\n",
727     "  plt.plot(hour,E,linestyle='--',c='r',label='Equilibrium')\n",
728     "  # print(len(hour),len(m_f))\n",
729     "  plt.plot(hour,m_f,linestyle='-',c='b',label='10-h fuel truth')\n",
730     "  plt.scatter(hour[:h2],data[:h2],c='b',label='10-h fuel data')\n",
731     "  plt.plot(hour[:h2],m[:h2],linestyle='-',c='k',label='filtered')\n",
732     "  plt.plot(hour[h2:hours],m[h2:hours],linestyle='-',c='r',label='forecast')\n",
733     "  plt.title('Kalman filtering and forecast on artificial data')\n",
734     "  plt.xlabel('Time (hours)') \n",
735     "  plt.ylabel('Fuel moisture content (%)')\n",
736     "  plt.legend()\n",
737     " \n",
738     "  return E, P"
739    ]
740   },
741   {
742    "cell_type": "code",
743    "execution_count": null,
744    "metadata": {
745     "id": "d0EFhTPZAlBD",
746     "scrolled": true
747    },
748    "outputs": [],
749    "source": [
750     "DeltaE = 0.0          # bias\n",
751     "E, P = kf_example(DeltaE)"
752    ]
753   },
754   {
755    "cell_type": "markdown",
756    "metadata": {
757     "id": "vqyB2Yz3uCsD"
758    },
759    "source": [
760     "We have recovered the fuel moisture from data with random noise - we **filtered** the noise out. "
761    ]
762   },
763   {
764    "cell_type": "markdown",
765    "metadata": {
766     "id": "Dl7pBZ9B3Nox"
767    },
768    "source": [
769     "Let's have a look at the evolution of the filter's estimate of the variance $P$. A common problem with the Kalman filter is when the variance converges to zero over time, then, since the filter trusts the model too much, it ignores the observations. Of course, once we switch to forecasting mode, the variance is not of interest. We could keep evolving the variance to bridge over periods when there are no observations, but not in this simplified version."
770    ]
771   },
772   {
773    "cell_type": "code",
774    "execution_count": null,
775    "metadata": {
776     "id": "wRJgbmGLc91g"
777    },
778    "outputs": [],
779    "source": [
780     "%matplotlib inline\n",
781     "plt.figure(figsize=(16,4))\n",
782     "plt.plot(P,linestyle='-',c='b',label='Estimated state variance P')\n",
783     "plt.title('Kalman filtering and forecast on artificial data')\n",
784     "plt.xlabel('Time (hours)') \n",
785     "plt.ylabel('Estimated variance of fuel moisture (%^2)')\n",
786     "plt.legend()"
787    ]
788   },
789   {
790    "cell_type": "markdown",
791    "metadata": {
792     "id": "Ccr-uKbmAlBE"
793    },
794    "source": [
795     "Now what if the model is wrong - different from nature? That is always so in reality. Now suppose that the model and the truth are not the same. That is always the case in reality.  Consider a simple case when the model thinks that the equilibrium $E$ is too high."
796    ]
797   },
798   {
799    "cell_type": "code",
800    "execution_count": null,
801    "metadata": {
802     "id": "spMdGW8oAlBE"
803    },
804    "outputs": [],
805    "source": [
806     "DeltaE = -0.05\n",
807     "E, P = kf_example(DeltaE)  "
808    ]
809   },
810   {
811    "cell_type": "markdown",
812    "metadata": {
813     "id": "DQeF7J8T4j2i"
814    },
815    "source": [
816     "We have found a good estimate of the state $m$, while data is available. Also, the estimated state variance $P$ converges with time - we have *learned* the variance that balances the noise. But for forecasting fuel moisture, we need to continue the fuel moisture model into the future, and we can't have any measurements from future. We only have the equilibrium from weather forecast. And the forecast and the truth disagree - as soon as there is no data to attract the simulation, the model is doing its own thing."
817    ]
818   },
819   {
820    "cell_type": "markdown",
821    "metadata": {
822     "id": "6uXVJj9koGF2"
823    },
824    "source": [
825     "### 3.2 Acquisition and preprocessing of real data"
826    ]
827   },
828   {
829    "cell_type": "markdown",
830    "metadata": {
831     "id": "q3BpOBuzc91i"
832    },
833    "source": [
834     "Data assimilation for fuel moisture from Remote Automated Weather Stations (RAWS) was developed in Vejmelka et al. (2016). First, they use regression from all RAWS in a given area to extend the data spatially from RAWS to a grid in the whole area, then they run the extended Kalman filter at each grid node. Here, we are interested in a simplified problem: estimate future fuel moisture at a single RAWS location from weather data.  "
835    ]
836   },
837   {
838    "cell_type": "markdown",
839    "metadata": {
840     "id": "c8Y6bL1Yc91i"
841    },
842    "source": [
843     "#### 3.2.1 Acquisition of fuel moisture observations"
844    ]
845   },
846   {
847    "cell_type": "markdown",
848    "metadata": {
849     "id": "0CuXyWBFc91i"
850    },
851    "source": [
852     "We try to load the data from a saved file first. If that fails, retrieve the fuel moisture data from sensors on weather stations in the Mesowest network. Get all stations with fuel moisture data in a spatial box within one hour, then pick one station and retrieve the whole time series."
853    ]
854   },
855   {
856    "cell_type": "code",
857    "execution_count": null,
858    "metadata": {
859     "id": "LFrlbbMmc91i"
860    },
861    "outputs": [],
862    "source": [
863     "import json\n",
864     "jfile = 'raws.json'; vars='fuel_moisture'\n",
865     "#jfile = 'raws2.json'; vars='fuel_moisture,precip_accum_one_hour'\n",
866     "def json_w(j,f):\n",
867     "  print('writing json file',f)\n",
868     "  json.dump(j,open(f,'w'),indent=4)\n",
869     "try:\n",
870     "    #! wget --no-clobber http://math.ucdenver.edu/~jmandel/data/math4779f21/raws.json\n",
871     "    j = json.load(open(jfile,'r'))\n",
872     "    print('loaded from ',jfile)\n",
873     "    # Take the first station in the boulding box that has data between time_start and time_s2.\n",
874     "    # Then retrieve data for that station between time_start and time_end\n",
875     "    time_start = j['time_start']      # start of data time series\n",
876     "    # time_s2    = j['time_s2']         # end of segment to read coordinates\n",
877     "    time_end  = j['time_end']         # end of data time series\n",
878     "    meso_ts  = j['meso_ts']           # get meso observations time series\n",
879     "    obs_lon =   j['obs_lon']          # where we retrieved observations\n",
880     "    obs_lat =   j['obs_lat']\n",
881     "except:\n",
882     "    print(\"can't read\",jfile,', creating')\n",
883     "    # set up bounds\n",
884     "    time_start = \"201806010800\"  # June 1 2018 08:00 in format yyyymmddHHMM\n",
885     "    time_s2    = \"201806010900\"  # June 1 2018 09:00 in format yyyymmddHHMM \n",
886     "    time_end   = \"201907200900\"  # June 20 2018 09:00 in format yyyymmddHHMM \n",
887     "    #time_start=  \"201810230100\"\n",
888     "    #time_s2=  \"201810230300\"\n",
889     "    #time_end  =  \"201806022300\"\n",
890     "    !pip install MesoPy\n",
891     "    from MesoPy import Meso\n",
892     "    bounding_box = \"-115, 38, -110, 40\"  # min longtitude, latitude\n",
893     "    meso_token=\"b40cb52cbdef43ef81329b84e8fd874f\"       # you should get your own if you do more of this\n",
894     "    m = Meso(meso_token)# create a Meso object\n",
895     "    print('reading MesoWest fuel moisture data')\n",
896     "    json_w(m.variables(),'variables.json')\n",
897     "    meso_obss = m.timeseries(time_start, time_s2, bbox=bounding_box, \n",
898     "                             showemptystations = '0', vars=vars)   # ask the object for data\n",
899     "    json_w(meso_obss,'meso_obss.json')                        \n",
900     "    # pick one station and retrieve the whole time series.\n",
901     "    station=meso_obss['STATION'][0]\n",
902     "    json_w(station,'station.json')\n",
903     "    lon,lat = (float(station['LONGITUDE']),float(station['LATITUDE']))\n",
904     "    print(station['NAME'],'station',station['STID'],'at',lon,lat)\n",
905     "    e = 0.01   # tolerance\n",
906     "    bb = '%s, %s, %s, %s' % (lon - e, lat - e, lon + e, lat + e)\n",
907     "    print('bounding box',bb)\n",
908     "    meso_ts = m.timeseries(time_start, time_end, bbox=bb, showemptystations = '0', vars=vars)   # ask the object for data\n",
909     "    json_w(meso_ts,'meso_ts.json')                        \n",
910     "    obs_lon, obs_lat = (lon, lat)   # remember station coordinates for later\n",
911     "    j={'time_start':time_start,'time_s2':time_s2,'time_end':time_end,\n",
912     "       'meso_ts':meso_ts,'obs_lon':obs_lon,'obs_lat':obs_lat}\n",
913     "    json_w(j,jfile)\n",
914     "    print('done')"
915    ]
916   },
917   {
918    "cell_type": "code",
919    "execution_count": null,
920    "metadata": {
921     "id": "3bXopS3btyz0",
922     "scrolled": true
923    },
924    "outputs": [],
925    "source": [
926     "# process the data retrieved for this station\n",
927     "# print(json.dumps(meso_ts['STATION'][0], indent=4))\n",
928     "from datetime import datetime, timedelta, time\n",
929     "import numpy as np\n",
930     "import matplotlib.pyplot as plt\n",
931     "import pytz\n",
932     "station = meso_ts['STATION'][0]\n",
933     "time_str  = station['OBSERVATIONS']['date_time']\n",
934     "obs_time = [datetime.strptime(t, '%Y-%m-%dT%H:%M:%SZ').replace(tzinfo=pytz.UTC) for t in time_str]\n",
935     "start_time = obs_time[0].replace(minute=0)     # remember obs_time and start_time for later\n",
936     "end_time = obs_time[-1]\n",
937     "obs_data = np.array(station['OBSERVATIONS'][\"fuel_moisture_set_1\"])\n",
938     "# display the data retrieved\n",
939     "#for o_time,o_data in zip (obs_time,obs_data):\n",
940     "#    print(o_time,o_data)\n",
941     "%matplotlib inline\n",
942     "plt.figure(figsize=(16,4))\n",
943     "plt.plot(obs_data,linestyle='-',c='k',label='10-h fuel data')\n",
944     "plt.title(station['STID'] + ' 10 h fuel moisture data')\n",
945     "plt.xlabel('Time (hours)') \n",
946     "plt.ylabel('Fuel moisture content (%)')\n",
947     "plt.legend()\n",
948     " "
949    ]
950   },
951   {
952    "cell_type": "markdown",
953    "metadata": {
954     "id": "pY4hPeATK9wZ"
955    },
956    "source": [
957     "#### 3.2.2 Acquisition of weather data"
958    ]
959   },
960   {
961    "cell_type": "markdown",
962    "metadata": {
963     "id": "xhyjXqxVN6B2"
964    },
965    "source": [
966     "Our weather data are results from atmospheric models, with assimilated observations from weather stations, satellites, radars, etc. The models can be run in reanalysis mode (for the past, with data for the period modeled)  or in forecast mode (for the future, with only past data assimilated - because future data are not here yet). We use the Real-Time Mesoscale Analysis ([RTMA](https://www.nco.ncep.noaa.gov/pmb/products/rtma/)) interpolated to the RAWS location. RTMA is a real-time product, posted hourly, and available only for few days in the past. We have our own collection of selected RAWS data over past few years, obtained as a side effect of running the fuel moisture modeling software [WRFXPY](https://github.com/openwfm/wrfxpy).\n",
967     "\n",
968     "First try to read the data already extracted for this RAWS and staged for download."
969    ]
970   },
971   {
972    "cell_type": "code",
973    "execution_count": null,
974    "metadata": {
975     "id": "WlqJRP8Vc91o"
976    },
977    "outputs": [],
978    "source": [
979     "import json\n",
980     "jfile = 'rtma.json'\n",
981     "try:\n",
982     "    ! wget --no-clobber http://math.ucdenver.edu/~jmandel/data/math4779f21/rtma.json\n",
983     "    j = json.load(open(jfile,'r'))\n",
984     "    print('loaded from ',jfile)\n",
985     "    if j['obs_lat']!=obs_lat or j['obs_lon']!=obs_lon:\n",
986     "      print('lon lat doesnot agree, need to load original RTMA files')\n",
987     "      read_rtma=True\n",
988     "    else:\n",
989     "      read_rtma=False\n",
990     "except:\n",
991     "    print(\"can't read\",jfile,', creating')\n",
992     "    read_rtma=True\n",
993     "\n",
994     "print('')"
995    ]
996   },
997   {
998    "cell_type": "markdown",
999    "metadata": {
1000     "id": "2iBNHQg5hPxB"
1001    },
1002    "source": [
1003     "####<font color=red>Note: If read_rtma==True, the notebook will say it crashed when run the first time. This is because it needs to install different version of some python packages and restart runtime. Simply run it again.</fonr>"
1004    ]
1005   },
1006   {
1007    "cell_type": "code",
1008    "execution_count": null,
1009    "metadata": {
1010     "id": "mxZABVDxt0gd"
1011    },
1012    "outputs": [],
1013    "source": [
1014     "# Set up environment to read RTMA gribs\n",
1015     "# we will need current numpy for pygrib - needed on Colab, tensorflow is using numpy 1.19\\\n",
1016     "if read_rtma:\n",
1017     "  import subprocess,os\n",
1018     "  def load_rtma(path,file,reload=0):\n",
1019     "    url='http://math.ucdenver.edu/~jmandel/rtma/' + path \n",
1020     "    if os.path.exists(file):\n",
1021     "      if reload:\n",
1022     "        print(file + ' already exists, removing')\n",
1023     "        os.remove(file)\n",
1024     "      else:\n",
1025     "        print(file + ' already exists, exiting')\n",
1026     "        # add checking size here\n",
1027     "        return 0\n",
1028     "    try:\n",
1029     "      ret = subprocess.check_output(['wget','--no-clobber','--output-document='+ file, url,],stderr=subprocess.STDOUT).decode() # execute command from python strings\n",
1030     "      if os.path.exists(file):\n",
1031     "        print('loaded ' + url + ' as ' + file)\n",
1032     "        return 0\n",
1033     "      else: \n",
1034     "        print('file transfer completed, but the file is missing? ' + url)  \n",
1035     "      return 1\n",
1036     "    except:\n",
1037     "      print('file transfer failed: ' + url)\n",
1038     "      return 2\n"
1039    ]
1040   },
1041   {
1042    "cell_type": "markdown",
1043    "metadata": {
1044     "id": "dQ-uJI2sy6I3"
1045    },
1046    "source": [
1047     "Create a function to transfer RTMA files in GRIB2 format from the stash. The function returns zero if the file transfer succeeded. If the file is not available, it returns a nonzero value. Note: if needed, maybe in future add more sophisticated checks, check the return code of wget and if the file size is correct."
1048    ]
1049   },
1050   {
1051    "cell_type": "markdown",
1052    "metadata": {
1053     "id": "THI6gElyHOOc"
1054    },
1055    "source": [
1056     "Next, functions to get the files, open as grib, and interpolate to the station coordinates"
1057    ]
1058   },
1059   {
1060    "cell_type": "code",
1061    "execution_count": null,
1062    "metadata": {
1063     "id": "PL3gxK67AlBI"
1064    },
1065    "outputs": [],
1066    "source": [
1067     "if read_rtma:\n",
1068     "  def rtma_grib(t,var):\n",
1069     "    tpath = '%4i%02i%02i/%02i' % (t.year, t.month, t.day, t.hour)  # remote path on server\n",
1070     "    tstr  = '%4i%02i%02i%02i_' % (t.year, t.month, t.day, t.hour)  # time string for local path\n",
1071     "    gribfile = os.path.join('data',tstr + var + '.grib')\n",
1072     "    remote = tpath + '/' + var + '.grib'\n",
1073     "    if load_rtma(remote,gribfile):\n",
1074     "        print('cannot load remote file',remote,'as',gribfile)\n",
1075     "        return []\n",
1076     "    else:\n",
1077     "        try:\n",
1078     "            gf=GribFile(gribfile)\n",
1079     "            v = np.array(gf[1].values())\n",
1080     "        except:\n",
1081     "            print('cannot read grib file',gribfile)\n",
1082     "            return []\n",
1083     "        print('loaded ',gribfile,' containing array shape ',v.shape)\n",
1084     "        return gf[1]   # grib message\n"
1085    ]
1086   },
1087   {
1088    "cell_type": "code",
1089    "execution_count": null,
1090    "metadata": {
1091     "id": "ccp10kurAlBI"
1092    },
1093    "outputs": [],
1094    "source": [
1095     "from scipy.interpolate import LinearNDInterpolator, interpn\n",
1096     "from scipy.optimize import root\n",
1097     "def interp_to_lat_lon_slow(lats,lons,v,lat,lon): \n",
1098     "    # on mesh with coordinates lats and lons interpolate v to given lat lon\n",
1099     "    interp=LinearNDInterpolator(list(zip(lats.flatten(),lons.flatten())),v.flatten())\n",
1100     "    return interp(lat,lon)\n",
1101     "def interp_to_lat_lon(lats,lons,v,lat,lon):\n",
1102     "    # on mesh with coordinates lats and lons interpolate v to given lat lon\n",
1103     "    points=(np.array(range(lats.shape[0]),float),np.array(range(lats.shape[1]),float))  # uniform mesh\n",
1104     "    def res(ij):  # interpolation of lons lats on the uniform mesh, to noninteger coordinates   \n",
1105     "       return np.hstack((interpn(points,lats,ij)-lat, interpn(points,lons,ij)-lon))\n",
1106     "    # solve for xi,xj such that lats(xi,xj)=lat lons(xi,xj)=lon, then interpolate to (xi, xj) on uniform grid \n",
1107     "    result = root(res,(0,0)) # solve res(ij) = 0\n",
1108     "    if not result.success:\n",
1109     "        print(result.message)\n",
1110     "        exit(1)\n",
1111     "    return interpn(points,v,result.x) \n"
1112    ]
1113   },
1114   {
1115    "cell_type": "markdown",
1116    "metadata": {
1117     "id": "jvnpq6S5AlBI"
1118    },
1119    "source": [
1120     "The interpolation function needs to  be tested."
1121    ]
1122   },
1123   {
1124    "cell_type": "code",
1125    "execution_count": null,
1126    "metadata": {
1127     "id": "NVMJBYI7AlBI"
1128    },
1129    "outputs": [],
1130    "source": [
1131     "def interp_to_lat_lon_test(lats,lons):\n",
1132     "    print('testing interp_to_lat_lon')\n",
1133     "    vx, vy = np.meshgrid(range(lats.shape[0]),range(lats.shape[1]),indexing='ij')\n",
1134     "    i, j = (1,2)\n",
1135     "    lat,lon = ((lats[i,j]+lats[i+1,j+1])/2,(lons[i,j]+lons[i+1,j+1])/2)\n",
1136     "    vi = interp_to_lat_lon(lats,lons,vx,lat,lon)\n",
1137     "    vj = interp_to_lat_lon(lats,lons,vy,lat,lon)\n",
1138     "    print(vi,vj,'should be about',i+0.5,j+0.5)\n",
1139     "    test_slow = 0\n",
1140     "    if test_slow:\n",
1141     "        print('Testing against the standard slow method scipy.interpolate.LinearNDInterpolator. Please wait...')\n",
1142     "        vi_slow = interp_to_lat_lon_slow(lats,lons,vx,lat,lon)\n",
1143     "        print(vi_slow)\n",
1144     "        vj_slow = interp_to_lat_lon_slow(lats,lons,vy,lat,lon)\n",
1145     "        print(vj_slow)\n",
1146     "        \n",
1147     "#gf = rtma_grib(start_time,'temp')      #  read the first grib file and use it to test interpolation\n",
1148     "#lats, lons = gf.latlons()\n",
1149     "#interp_to_lat_lon_test(lats,lons)\n"
1150    ]
1151   },
1152   {
1153    "cell_type": "code",
1154    "execution_count": null,
1155    "metadata": {
1156     "id": "vt-Mk8fIc91m"
1157    },
1158    "outputs": [],
1159    "source": [
1160     "#%debug\n"
1161    ]
1162   },
1163   {
1164    "cell_type": "markdown",
1165    "metadata": {
1166     "id": "LQbWB_3GAlBI"
1167    },
1168    "source": [
1169     "Now we are ready for a function to read the RTMA files and interpolate to the station coordinates"
1170    ]
1171   },
1172   {
1173    "cell_type": "code",
1174    "execution_count": null,
1175    "metadata": {
1176     "id": "b3JJH3XPAlBI"
1177    },
1178    "outputs": [],
1179    "source": [
1180     "if read_rtma:\n",
1181     "  import pandas as pd, json\n",
1182     "  def read_interp_rtma(varnames,times,lat,lon):\n",
1183     "    # read RTMA from start_time to end_time and interpolate to obs_lat obs_lon\n",
1184     "    ntimes = len(times)\n",
1185     "    time_str = 'time_str'\n",
1186     "    j={time_str:times.strftime('%Y-%m-%d %H:%M').tolist()}\n",
1187     "    for varname in varnames:\n",
1188     "        j[varname]=np.full(ntimes,np.nan)  # initialize array of nans as list\n",
1189     "    n=0\n",
1190     "    for t in times:\n",
1191     "        tim=t.strftime('%Y-%m-%d %H:%M')\n",
1192     "        should_be = j[time_str][n]\n",
1193     "        if tim != should_be:\n",
1194     "            print('n=',n,'time',tim,'expected',should_be)\n",
1195     "            raise 'Invalid time' \n",
1196     "        for varname in varnames:\n",
1197     "            gf = rtma_grib(t,varname)   # read and create grib object, download if needed\n",
1198     "            if gf:\n",
1199     "                lats,lons = gf.latlons()    # coordinates\n",
1200     "                v = gf.values()\n",
1201     "                vi=interp_to_lat_lon(lats,lons,v,lat,lon) # append to array\n",
1202     "                print(varname,'at',t,'interpolated to',lat,lon,' value ',vi)\n",
1203     "                j[varname][n] = vi\n",
1204     "            else:\n",
1205     "                print(varname,'at',t,' could not be loaded')\n",
1206     "        n = n+1\n",
1207     "    return j"
1208    ]
1209   },
1210   {
1211    "cell_type": "code",
1212    "execution_count": null,
1213    "metadata": {
1214     "id": "OY1oTYKlfd17"
1215    },
1216    "outputs": [],
1217    "source": [
1218     "if read_rtma:\n",
1219     "    times = pd.date_range(start=time_start,end=time_end,freq='1H')\n",
1220     "    varnames=['temp','td','precipa']\n",
1221     "    j =    read_interp_rtma(varnames,times,obs_lat,obs_lon)      # temperature\n",
1222     "    for varname in varnames:\n",
1223     "        j[varname]=j[varname].tolist() \n",
1224     "    j['obs_lat']=obs_lat\n",
1225     "    j['obs_lon']=obs_lon\n",
1226     "    json.dump(j,open('rtma.json','w'),indent=4)\n",
1227     "    print('done')"
1228    ]
1229   },
1230   {
1231    "cell_type": "code",
1232    "execution_count": null,
1233    "metadata": {
1234     "id": "bMpYIZT6c91o"
1235    },
1236    "outputs": [],
1237    "source": [
1238     "# %debug\n"
1239    ]
1240   },
1241   {
1242    "cell_type": "markdown",
1243    "metadata": {
1244     "id": "KVXBjGA0CiXr"
1245    },
1246    "source": [
1247     "#### 3.2.3 Preprocessing and visualization of the weather data"
1248    ]
1249   },
1250   {
1251    "cell_type": "code",
1252    "execution_count": null,
1253    "metadata": {
1254     "id": "fNA3Vbo1c91o"
1255    },
1256    "outputs": [],
1257    "source": [
1258     "rtma = j\n",
1259     "td = np.array(rtma['td'])\n",
1260     "t2 = np.array(rtma['temp'])\n",
1261     "rain=np.array(rtma['precipa'])\n",
1262     "# compute relative humidity\n",
1263     "rh = 100*np.exp(17.625*243.04*(td - t2) / (243.04 + t2 - 273.15) / (243.0 + td - 273.15))\n",
1264     "Ed = 0.924*rh**0.679 + 0.000499*np.exp(0.1*rh) + 0.18*(21.1 + 273.15 - t2)*(1 - np.exp(-0.115*rh))\n",
1265     "Ew = 0.618*rh**0.753 + 0.000454*np.exp(0.1*rh) + 0.18*(21.1 + 273.15 - t2)*(1 - np.exp(-0.115*rh))"
1266    ]
1267   },
1268   {
1269    "cell_type": "code",
1270    "execution_count": null,
1271    "metadata": {
1272     "id": "tZIK59bJAlBJ"
1273    },
1274    "outputs": [],
1275    "source": [
1276     "%matplotlib inline\n",
1277     "plt.figure(figsize=(16,4))\n",
1278     "plt.plot(t2,linestyle='-',c='k',label='Temperature')\n",
1279     "plt.title(station['STID'] + ' Temperature')\n",
1280     "plt.xlabel('Time (hours)') \n",
1281     "plt.ylabel('Temperature (K)')\n",
1282     "plt.legend()"
1283    ]
1284   },
1285   {
1286    "cell_type": "code",
1287    "execution_count": null,
1288    "metadata": {
1289     "id": "LbyqcuXYAlBJ"
1290    },
1291    "outputs": [],
1292    "source": [
1293     "%matplotlib inline\n",
1294     "plt.figure(figsize=(16,4))\n",
1295     "plt.plot(td,linestyle='-',c='k',label='Dew point')\n",
1296     "plt.title(station['STID'] + ' Dew point (K)')\n",
1297     "plt.xlabel('Time (hours)') \n",
1298     "plt.ylabel('Dew point (K)')\n",
1299     "plt.legend()"
1300    ]
1301   },
1302   {
1303    "cell_type": "code",
1304    "execution_count": null,
1305    "metadata": {
1306     "id": "dfoOK2kSc91p"
1307    },
1308    "outputs": [],
1309    "source": [
1310     "%matplotlib inline\n",
1311     "plt.figure(figsize=(16,4))\n",
1312     "plt.plot(rh,linestyle='-',c='k',label='Dew point')\n",
1313     "plt.title(station['STID'] + ' relative humidity')\n",
1314     "plt.xlabel('Time (hours)') \n",
1315     "plt.ylabel('Relative humidity (%)')\n",
1316     "plt.legend()"
1317    ]
1318   },
1319   {
1320    "cell_type": "code",
1321    "execution_count": null,
1322    "metadata": {
1323     "id": "MWTJ5b2kc91p"
1324    },
1325    "outputs": [],
1326    "source": [
1327     "%matplotlib inline\n",
1328     "plt.figure(figsize=(16,4))\n",
1329     "plt.plot(Ed,linestyle='-',c='r',label='drying equilibrium')\n",
1330     "plt.plot(Ew,linestyle=':',c='b',label='wetting equilibrium')\n",
1331     "plt.title(station['STID'] + ' drying and wetting equilibria')\n",
1332     "plt.xlabel('Time (hours)') \n",
1333     "plt.ylabel('Fuel moisture contents (%)')\n",
1334     "plt.legend()"
1335    ]
1336   },
1337   {
1338    "cell_type": "markdown",
1339    "metadata": {
1340     "id": "jY3_eeBRc91p"
1341    },
1342    "source": [
1343     " "
1344    ]
1345   },
1346   {
1347    "cell_type": "code",
1348    "execution_count": null,
1349    "metadata": {
1350     "id": "PQKSRvRSAlBJ"
1351    },
1352    "outputs": [],
1353    "source": [
1354     "%matplotlib inline\n",
1355     "plt.figure(figsize=(16,4))\n",
1356     "plt.plot(rain,linestyle='-',c='k',label='Precipitation')\n",
1357     "plt.title(station['STID'] + ' Precipitation' )\n",
1358     "plt.xlabel('Time (hours)') \n",
1359     "plt.ylabel('Precipitation (mm/hour)')\n",
1360     "plt.legend()"
1361    ]
1362   },
1363   {
1364    "cell_type": "code",
1365    "execution_count": null,
1366    "metadata": {
1367     "id": "Dwbt4UXfro5x"
1368    },
1369    "outputs": [],
1370    "source": [
1371     "print(rain[1900:2000])"
1372    ]
1373   },
1374   {
1375    "cell_type": "markdown",
1376    "metadata": {
1377     "id": "_yRu_7WvHc6P"
1378    },
1379    "source": [
1380     "Precipitation from RTMA is in kg/m${}^2$. 1m water depth over 1m${}^2$ is 1m${}^3$ with mass 1000 kg thus 1 kg/m${}^2$ is the same as 1 mm of precipitation. RTMA values are accumulations over 1 h so these are values in mm/h. So 9999 mm/h = 10m/h makes no sense. Replace anything over 1m/h by nan and try again."
1381    ]
1382   },
1383   {
1384    "cell_type": "code",
1385    "execution_count": null,
1386    "metadata": {
1387     "id": "XPYO_Iuvc91q"
1388    },
1389    "outputs": [],
1390    "source": [
1391     "rain[rain > 1000] = np.NaN"
1392    ]
1393   },
1394   {
1395    "cell_type": "code",
1396    "execution_count": null,
1397    "metadata": {
1398     "id": "GYWTfbBBc91q",
1399     "scrolled": true
1400    },
1401    "outputs": [],
1402    "source": [
1403     "%matplotlib inline\n",
1404     "plt.figure(figsize=(16,4))\n",
1405     "plt.plot(rain,linestyle='-',c='k',label='Precipitation')\n",
1406     "plt.title(station['STID'] + ' Precipitation' )\n",
1407     "plt.xlabel('Time (hours)') \n",
1408     "plt.ylabel('Precipitation (mm/hour)')\n",
1409     "plt.legend()"
1410    ]
1411   },
1412   {
1413    "cell_type": "markdown",
1414    "metadata": {
1415     "id": "Q_L0R2Njc91q"
1416    },
1417    "source": [
1418     "Fix some missing data, then we can use the data for up to 1942 hours until a biger gap."
1419    ]
1420   },
1421   {
1422    "cell_type": "code",
1423    "execution_count": null,
1424    "metadata": {
1425     "id": "_tkU7UJic91q"
1426    },
1427    "outputs": [],
1428    "source": [
1429     "# fix isolated nans\n",
1430     "def fixnan(a,n):\n",
1431     "    for c in range(n):\n",
1432     "        for i in np.where(np.isnan(a)):\n",
1433     "            a[i]=0.5*(a[i-1]+a[i+1])\n",
1434     "        if not any(np.isnan(a)):\n",
1435     "            break\n",
1436     "    return a\n",
1437     "\n",
1438     "rain=fixnan(rain,2)\n",
1439     "t2=fixnan(t2,2)\n",
1440     "rh=fixnan(rh,2)\n",
1441     "obs_data=fixnan(obs_data,2)\n",
1442     "Ed=fixnan(Ed,2)\n",
1443     "Ew=fixnan(Ew,2)\n",
1444     "\n",
1445     "print(np.where(np.isnan(rain)))\n",
1446     "print(np.where(np.isnan(t2)))\n",
1447     "print(np.where(np.isnan(rh)))\n",
1448     "print(np.where(np.isnan(obs_data)))"
1449    ]
1450   },
1451   {
1452    "cell_type": "markdown",
1453    "metadata": {
1454     "id": "XqQYnyI9DIy1"
1455    },
1456    "source": [
1457     "## 4 Results"
1458    ]
1459   },
1460   {
1461    "cell_type": "markdown",
1462    "metadata": {
1463     "id": "2tIC_Tqnc91r"
1464    },
1465    "source": [
1466     "### 4.1 Kalman filter with fuel moisture observations, followed by forecasting\n",
1467     "We run the model first with Kalman filter for 150 hours. The observations are the RAWS data\n",
1468     "After 150 hours, we run in forecast mode - the RAWS data are no longer used, and we run the model from the weather data without the Kalman filter. The weather data are taken to be RTMA interpolated to one RAWS location.\n",
1469     "In a real forecasting application, the model would be run from weather forecast rather than data."
1470    ]
1471   },
1472   {
1473    "cell_type": "code",
1474    "execution_count": null,
1475    "metadata": {
1476     "id": "aXnSQM7wc91r"
1477    },
1478    "outputs": [],
1479    "source": [
1480     "# run KF on an initial data seqment\n",
1481     "import numpy as np\n",
1482     "import matplotlib.pyplot as plt \n",
1483     "\n",
1484     "hours=1200 # total simulation\n",
1485     "h2 = 150\n",
1486     "m = np.zeros(hours) # preallocate\n",
1487     "m[0]= obs_data[0]             # initial state  \n",
1488     "P = np.zeros(hours)\n",
1489     "P[0] = 1e-3 # background state variance\n",
1490     "H = np.array([1.])   # all oQ = np.array([0.02]) # process noise variancebserved\n",
1491     "Q = np.array([1e-3]) # process noise variance\n",
1492     "R = np.array([1e-3]) # data variance\n",
1493     "for t in range(hours-1):\n",
1494     "    # using lambda construction to pass additional arguments to the model \n",
1495     "    if t < h2 and not np.isnan(obs_data[t]) and not np.isnan(Ew[t]) and not np.isnan(rain[t]): # advance model and run KF\n",
1496     "        m[t+1],P[t+1] = ext_kf(m[t],P[t],lambda u: model_moisture(u,Ed[t],Ew[t],rain[t],t,partials=1),Q,\n",
1497     "                    d=obs_data[t],H=H,R=R)\n",
1498     "    else:  # just advance to next hour, no process noise\n",
1499     "        m[t+1],P[t+1] = ext_kf(m[t],P[t],lambda u: model_moisture(u,Ed[t],Ew[t],rain[t],t,partials=1),Q*0.0)"
1500    ]
1501   },
1502   {
1503    "cell_type": "code",
1504    "execution_count": null,
1505    "metadata": {
1506     "id": "peMi-OF3c91r",
1507     "scrolled": true
1508    },
1509    "outputs": [],
1510    "source": [
1511     "%matplotlib inline\n",
1512     "plt.figure(figsize=(16,4))\n",
1513     "plt.plot(Ed[:hours],linestyle='--',c='r',label='Drying Equilibrium')\n",
1514     "plt.plot(Ew[:hours],linestyle='--',c='b',label='Wetting Equilibrium')\n",
1515     "plt.plot(obs_data[:hours],linestyle=':',c='k',label='RAWS data')\n",
1516     "plt.plot(m[:h2],linestyle='-',c='k',label='filtered')\n",
1517     "plt.plot(range(h2,hours),m[h2:hours],linestyle='-',c='r',label='forecast')\n",
1518     "plt.title('Kalman filtering and forecast with real data')\n",
1519     "plt.xlabel('Time (hours)') \n",
1520     "plt.ylabel('Fuel moisture content (%)')\n",
1521     "plt.legend()"
1522    ]
1523   },
1524   {
1525    "cell_type": "markdown",
1526    "metadata": {
1527     "id": "3TnwXYcLc91r"
1528    },
1529    "source": [
1530     "Clearly, there is a problem - the forecast fuel moisture is too high. We need to assimilate also some parameters of the model, not just its output state. "
1531    ]
1532   },
1533   {
1534    "cell_type": "markdown",
1535    "metadata": {
1536     "id": "-WMWCDz4DX45"
1537    },
1538    "source": [
1539     "#### 4.2 Model with an augmented state"
1540    ]
1541   },
1542   {
1543    "cell_type": "markdown",
1544    "metadata": {
1545     "id": "jivOYEhiXMi5"
1546    },
1547    "source": [
1548     "In reality, the equilibrium moisture $E$ computed from atmospheric conditions\n",
1549     "generally does not agree with the data. We want to add a correction $\\Delta\n",
1550     "E$ to $E$ constant in time, and identify the new parameter $\\Delta E$ from data. \n",
1551     "Because the Kalman filter identifies state, add the parameter to the state.\n",
1552     "Define augmented state $u=\\left[\n",
1553     "\\begin{array}\n",
1554     "[c]{c}\n",
1555     "m\\\\\n",
1556     "\\Delta E\n",
1557     "\\end{array}\n",
1558     "\\right]  .$ Since $\\Delta E$ is constant in time, it satisfies the\n",
1559     "differential equation $\\frac{d\\Delta E}{dt}=0.$ So, we want to estimate the\n",
1560     "state $u$ governed by the\n",
1561     "$$\n",
1562     "\\frac{d}{dt}\\left[\n",
1563     "\\begin{array}\n",
1564     "[c]{c}\n",
1565     "m\\\\\n",
1566     "\\Delta E\n",
1567     "\\end{array}\n",
1568     "\\right]  =\\left[\n",
1569     "\\begin{array}\n",
1570     "[c]{c}\n",
1571     "\\frac{E+\\Delta E-m(t)}{T}\\\\\n",
1572     "0\n",
1573     "\\end{array}\n",
1574     "\\right]  ,\n",
1575     "$$\n",
1576     "which we write as $\\frac{du}{dt}=F(u),$ where\n",
1577     "$$\n",
1578     "F(u)=\\left[\n",
1579     "\\begin{array}\n",
1580     "[c]{c}\n",
1581     "F_{1}\\left(  u\\right)  \\\\\n",
1582     "F_{2}\\left(  u\\right)\n",
1583     "\\end{array}\n",
1584     "\\right]  =F\\left(  \\left[\n",
1585     "\\begin{array}\n",
1586     "[c]{c}\n",
1587     "m\\\\\n",
1588     "\\Delta E\n",
1589     "\\end{array}\n",
1590     "\\right]  \\right)  =\\left[\n",
1591     "\\begin{array}\n",
1592     "[c]{c}\n",
1593     "\\left(  E+\\Delta E-m(t)\\right)  T_{1}\\\\\n",
1594     "0\n",
1595     "\\end{array}\n",
1596     "\\right]  ,\\quad T_{1}=\\frac{1}{T}.\n",
1597     "$$\n",
1598     "The Jacobian of $F$ is\n",
1599     "$$\n",
1600     "\\left[\n",
1601     "\\begin{array}\n",
1602     "[c]{cc}\n",
1603     "\\frac{\\partial F_{1}}{\\partial u_{1}} & \\frac{\\partial F_{1}}{\\partial u_{2}\n",
1604     "}\\\\\n",
1605     "\\frac{\\partial F_{2}}{\\partial u_{1}} & \\frac{\\partial F_{2}}{\\partial u_{2}}\n",
1606     "\\end{array}\n",
1607     "\\right]  =\\left[\n",
1608     "\\begin{array}\n",
1609     "[c]{cc}\n",
1610     "\\frac{\\partial m_{1}}{\\partial m_{0}} & \\frac{\\partial m_{1}}{\\partial E}\\\\\n",
1611     "\\frac{\\partial\\Delta E}{\\partial m_{0}} & \\frac{\\partial\\Delta E}\n",
1612     "{\\partial\\Delta E}\n",
1613     "\\end{array}\n",
1614     "\\right]  =\\left[\n",
1615     "\\begin{array}\n",
1616     "[c]{cc}\n",
1617     "\\frac{\\partial m_{1}}{\\partial m_{0}} & \\frac{\\partial m_{1}}{\\partial E}\\\\\n",
1618     "0 & 1\n",
1619     "\\end{array}\n",
1620     "\\right]\n",
1621     "$$\n",
1622     "Here is a function that implements the augmented model $F$. The input is\n",
1623     "$u_{0}$. The output is $u_{1}$ and the Jacobian $du_{1}/du_{0}$."
1624    ]
1625   },
1626   {
1627    "cell_type": "markdown",
1628    "metadata": {
1629     "id": "MJ1C_1Omc91s"
1630    },
1631    "source": [
1632     "\n",
1633     "Define augmented model function with drying, wetting, and rain equilibria"
1634    ]
1635   },
1636   {
1637    "cell_type": "code",
1638    "execution_count": null,
1639    "metadata": {
1640     "id": "GHtAaGp9WSHT"
1641    },
1642    "outputs": [],
1643    "source": [
1644     "def model_augmented(u0,Ed,Ew,r,t):\n",
1645     "    # state u is the vector [m,dE] with dE correction to equilibria Ed and Ew at t\n",
1646     "    # \n",
1647     "    m0, Ec = u0  # decompose state u0\n",
1648     "    # reuse model_moisture(m0,Eqd,Eqw,r,partials=0):\n",
1649     "    # arguments:\n",
1650     "    # m0         starting fuel moistureb (1)\n",
1651     "    # Ed         drying equilibrium      (1) \n",
1652     "    # Ew         wetting equilibrium     (1)\n",
1653     "    # r          rain intensity          (mm/h)\n",
1654     "    # partials = 0, 1, 2\n",
1655     "    # returns: same as model_decay\n",
1656     "    #   if partials==0: m1 = fuel moisture contents after time 1 hour\n",
1657     "    #              ==1: m1, dm0/dm0 \n",
1658     "    #              ==2: m1, dm1/dm0, dm1/dE \n",
1659     "    m1, dm1_dm0, dm1_dE  = model_moisture(m0,Ed + Ec, Ew + Ec, r, t, partials=2)\n",
1660     "    u1 = np.array([m1,Ec])   # dE is just copied\n",
1661     "    J =  np.array([[dm1_dm0, dm1_dE],\n",
1662     "                   [0.     ,     1.]])\n",
1663     "    return u1, J"
1664    ]
1665   },
1666   {
1667    "cell_type": "markdown",
1668    "metadata": {
1669     "id": "8SuVNg8TsW4d"
1670    },
1671    "source": [
1672     "### 4.3 Kalman filter on the augmented model"
1673    ]
1674   },
1675   {
1676    "cell_type": "code",
1677    "execution_count": null,
1678    "metadata": {
1679     "id": "1No3g6HyAEh_"
1680    },
1681    "outputs": [],
1682    "source": [
1683     "u = np.zeros((2,hours))\n",
1684     "u[:,0]=[0.1,0.1]       # initialize,background state  \n",
1685     "P = np.zeros((2,2,hours))\n",
1686     "P[:,:,0] = np.array([[1e-3, 0.],\n",
1687     "                    [0.,  1e-3]]) # background state covariance\n",
1688     "Q = np.array([[1e-3, 0.],\n",
1689     "              [0,  1e-3]]) # process noise covariance\n",
1690     "H = np.array([[1., 0.]])  # first component observed\n",
1691     "R = np.array([1e-3]) # data variance\n",
1692     "\n",
1693     "# ext_kf(u,P,F,Q=0,d=None,H=None,R=None) returns ua, Pa\n",
1694     "\n",
1695     "# print('initial u=',u,'P=',P)\n",
1696     "# print('Q=',Q,'H=',H,'R=',R)\n",
1697     "for t in range(1,h2):\n",
1698     "    # use lambda construction to pass additional arguments to the model \n",
1699     "    u[:,t],P[:,:,t] = ext_kf(u[:,t-1],P[:,:,t-1],\n",
1700     "                                 lambda uu: model_augmented(uu,Ed[t],Ew[t],rain[t],t),\n",
1701     "                                 Q,obs_data[t],H=H,R=R)\n",
1702     "    # print('time',t,'data',obs_data[t],'filtered',u[0,t],'Ec',u[1,t])\n",
1703     "for t in range(h2,hours):\n",
1704     "    u[:,t],P[:,:,t] = ext_kf(u[:,t-1],P[:,:,t-1],\n",
1705     "                                 lambda uu: model_augmented(uu,Ed[t],Ew[t],rain[t],t),\n",
1706     "                                 Q*0.0)\n",
1707     "    # print('time',t,'data',obs_data[t],'forecast',u[0,t],'Ec',u[1,t])\n",
1708     "m,Ec = u  # extract from state"
1709    ]
1710   },
1711   {
1712    "cell_type": "markdown",
1713    "metadata": {
1714     "id": "FYAbWNCfk2wD"
1715    },
1716    "source": [
1717     "Plot the result:\n"
1718    ]
1719   },
1720   {
1721    "cell_type": "code",
1722    "execution_count": null,
1723    "metadata": {
1724     "id": "hlkby_oTlB_f"
1725    },
1726    "outputs": [],
1727    "source": [
1728     "def plot_moisture(hmin,hmax):\n",
1729     "  print('training from 0 to',h2,'plot from',hmin,'to',hmax)\n",
1730     "  plt.figure(figsize=(16,4))\n",
1731     "  plt.plot(range(hmin,hmax),Ed[hmin:hmax],linestyle='--',c='r',label='Drying Equilibrium (%)')\n",
1732     "  plt.plot(range(hmin,hmax),Ew[hmin:hmax],linestyle='--',c='b',label='Wetting Equilibrium (%)')\n",
1733     "  plt.plot(range(hmin,hmax),Ec[hmin:hmax],linestyle='--',c='g',label='Equilibrium Correction (%)')\n",
1734     "  plt.plot(range(hmin,hmax),obs_data[hmin:hmax],linestyle=':',c='k',label='RAWS data (%)')\n",
1735     "  plt.plot(range(hmin,hmax),rain[hmin:hmax],linestyle='-',c='b',label='RTMA rain (mm/h)')\n",
1736     "  if hmin>=h2:\n",
1737     "    plt.plot(m[hmin:h2],linestyle='-',c='k',label='Filtered')\n",
1738     "  h1 = np.maximum(hmin,h2)\n",
1739     "  plt.plot(range(h1,hmax),m[h1:hmax],linestyle='-',c='r',label='Forecast (%)')\n",
1740     "  plt.title('Kalman filtering and forecast with augmented state, real data. Training 0:%i hmax' % h2)\n",
1741     "  plt.xlabel('Time (hours)') \n",
1742     "  plt.ylabel('Fuel moisture content (%)')\n",
1743     "  plt.legend()"
1744    ]
1745   },
1746   {
1747    "cell_type": "code",
1748    "execution_count": null,
1749    "metadata": {
1750     "id": "q-h5omKgnow2"
1751    },
1752    "outputs": [],
1753    "source": [
1754     "plot_moisture(0,hours)"
1755    ]
1756   },
1757   {
1758    "cell_type": "markdown",
1759    "metadata": {
1760     "id": "0w0YtHtqnza5"
1761    },
1762    "source": [
1763     "A detailed view of transition from training to forecast:"
1764    ]
1765   },
1766   {
1767    "cell_type": "code",
1768    "execution_count": null,
1769    "metadata": {
1770     "id": "B7sXGUotc91s"
1771    },
1772    "outputs": [],
1773    "source": [
1774     "plot_moisture(0,300)\n",
1775     "\n"
1776    ]
1777   },
1778   {
1779    "cell_type": "markdown",
1780    "metadata": {
1781     "id": "7W03QTo3c91t"
1782    },
1783    "source": [
1784     "Filtering by extended Kalman filter using RAWS data until 150 hours, then forecasting mode - running the model from interpolated RTMA only. For the first 60 hours the forecast is good, the equilibium correction made the model quite close to data. But then the big spike in equilibrium moisture around 230 hours attracted the solution, and it took a while for it to get back. The spike in the RAWS measurement is there but much smaller. The model becomes inaccurate during periods when the fuel moisture equilibrium is large.\n",
1785     "\n",
1786     "Possible reasons include: 1. There was something in the data we do not know about - maybe it rained but RTMA did not tell us. Try comparing with data from the RAWS itself? 2. The model is too simple, assumes the whole depth of the wood stick is wetting and drying at the same time. Perhaps the moisture got stored in the inside layers of the measurement stick. Try a two-layer model as in van der Kamp (2017) and make the state larger? "
1787    ]
1788   },
1789   {
1790    "cell_type": "markdown",
1791    "metadata": {
1792     "id": "owEI4EtTo7Ek"
1793    },
1794    "source": [
1795     "A detailed vieew of rain episode:"
1796    ]
1797   },
1798   {
1799    "cell_type": "code",
1800    "execution_count": null,
1801    "metadata": {
1802     "id": "C_hoDjgtpMEJ"
1803    },
1804    "outputs": [],
1805    "source": [
1806     "plot_moisture(900,1100)"
1807    ]
1808   },
1809   {
1810    "cell_type": "markdown",
1811    "metadata": {
1812     "id": "DRraWhwdpSkV"
1813    },
1814    "source": [
1815     "It seems there is some rain that the model does not know about."
1816    ]
1817   },
1818   {
1819    "cell_type": "markdown",
1820    "metadata": {
1821     "id": "gVQxv9Blc91t"
1822    },
1823    "source": [
1824     "### 4.4 A comment on the information flow in the Kalman filter and in neural networks"
1825    ]
1826   },
1827   {
1828    "cell_type": "markdown",
1829    "metadata": {
1830     "id": "CZmR4HPAc91t"
1831    },
1832    "source": [
1833     "At each time step $k$,\n",
1834     "\n",
1835     "* input state $u_{k-1}$ size $n$ and its covariance matrix $P_{k-1}$ size $n \\times n$.\n",
1836     "* the model is applied to external data $e_k$ and the input $u_{k-1},P_{k-1}$ produce the forecast $u_k^f$ and its covariance $P^f_k$\n",
1837     "* the new state $u_k$ is found by minimizing $|| u^f_k - u_k||^2_{P^f_k} + ||H u_k - d_k||^2_{R}$   \n",
1838     "* the new state covariance is $P_k = ( (P^f_k)^{-1} + H^\\top R^{-1} H)^{-1}$.\n",
1839     "\n",
1840     "Here, the state consists of \n",
1841     "* the fuel moisture and the adjustment to the equilibrium, dimension 2\n",
1842     "* the covariance matrix of vector of dimension 2, which is symmetric $2 \\times 2$ matrix, given by 3 numbers because it is symmetric\n",
1843     "Thus, the dimension of the state is 2 + 3 = 5. The first component of the state, the fuel moisture, is the quantity of interest, the rest are auxiliary.\n",
1844     "\n",
1845     "\n",
1846     "This can be understood as:\n",
1847     "\n",
1848     "* a mapping $M$ of the 5-dimensional hidden and external data state to a new hidden state:\n",
1849     "$$M:(u_{k-1},P_{k-1},e_k) \\mapsto (u_{k},P_{k})$$\n",
1850     "* taking the output (the quantity of interest) as the first component of the hiddent state\n",
1851     "* feeding the hiddent state back to the mapping $M$ for the next step $k+1$\n",
1852     "* training consists of fitting the hidden state to minimize a loss function\n",
1853     "$$\\ell(u_{k},P_{k},d_k,R_k) \\to \\min$$\n",
1854     "\n",
1855     "Note that in the augmented Kalman filter above, the mapping $M$ is fixed and it has a one component of the hidden state as a parameter. To get a better fit, we could increase the number of parameters, e.g., by modeling the moisture in multiple layers, as in van der Kamp et al. (2017) two-layer model.\n",
1856     "\n",
1857     "A recurrent neural network (RNN) has a similar information flow but it can be more flexible and look for the best model automatically, i.e., build the model from data. \n"
1858    ]
1859   },
1860   {
1861    "cell_type": "markdown",
1862    "metadata": {
1863     "id": "_g_OTEg6ePb9"
1864    },
1865    "source": [
1866     "## 5. Conclusion"
1867    ]
1868   },
1869   {
1870    "cell_type": "markdown",
1871    "metadata": {
1872     "id": "aNxw7xI3FqFt"
1873    },
1874    "source": [
1875     "We have shown how to combine a model and data for improved forecasting of fuel moisture from weather forecast using the Kalman filter. Augmenting the filter state by a model parameter and joint estimation of augmented state resulted in an improvement of the forecast."
1876    ]
1877   },
1878   {
1879    "cell_type": "markdown",
1880    "metadata": {
1881     "id": "IWpmDwUPGElR"
1882    },
1883    "source": [
1884     "## Contributions of authors "
1885    ]
1886   },
1887   {
1888    "cell_type": "markdown",
1889    "metadata": {
1890     "id": "jujW1VFgGOCn"
1891    },
1892    "source": [
1893     "Not applicable."
1894    ]
1895   },
1896   {
1897    "cell_type": "markdown",
1898    "metadata": {
1899     "id": "HWslw7HmGZmP"
1900    },
1901    "source": [
1902     "## Acknowledgements"
1903    ]
1904   },
1905   {
1906    "cell_type": "markdown",
1907    "metadata": {
1908     "id": "xubqDAV2GjkZ"
1909    },
1910    "source": [
1911     "This Math Clinic was sponsored by the team of investigators of the NASA grant no. 80NSSC19K1091 *Coupled Interactive Forecasting of Weather, Fire Behavior, and Smoke Impact for Improved Wildland Fire Decision Making* under the NASA ROSES18 Disasters program. The author would like to thank Brian Zhang from the Math Clinic class for bringing the reference van der Kamp et al. (2017) to his attention."
1912    ]
1913   },
1914   {
1915    "cell_type": "markdown",
1916    "metadata": {
1917     "id": "ZsNZxOv7c91t"
1918    },
1919    "source": [
1920     "## References"
1921    ]
1922   },
1923   {
1924    "cell_type": "markdown",
1925    "metadata": {
1926     "id": "vFY-iS1Wc91t"
1927    },
1928    "source": [
1929     "J. Mandel, S. Amram, J. D. Beezley, G. Kelman, A. K. Kochanski, V. Y. Kondratenko, B. H. Lynn, B. Regev, and M. Vejmelka. *Recent advances and applications of WRF-SFIRE.* Natural Hazards and Earth System Science, 14(10):2829–2845, 2014. [doi:10.5194/nhessd-2-1759-2014](https://doi.org/10.5194/nhessd-2-1759-2014)\n",
1930     "\n",
1931     "R. E. Kalman. *A new approach to linear filtering and prediction problems.* Transactions of the ASME – Journal of Basic Engineering, Series D, 82:35–45, 1960. [doi:10.1115/1.3662552](https://doi.org/10.1115/1.3662552)\n",
1932     "\n",
1933     "E. Kalnay. *Atmospheric Modeling, Data Assimilation and Predictability.* Cambridge University Press, 2003. [doi:10.1017/CBO9780511802270](https://doi.org/10.1017/CBO9780511802270)\n",
1934     "\n",
1935     "D. W. van der Kamp, R. D. Moore, and I. G. McKendry. *A model for simulating the moisture content of standardized fuel sticks of various sizes.* Agricultural and Forest Meteorology, 236:123–134, 2017. [doi:10.1016/j.agrformet.2017.01.013](https://doi.org/10.1016/j.agrformet.2017.01.013)\n",
1936     "\n",
1937     "S. F. Schmidt. *Application of state-space methods to navigation problems.* volume 3 of Advances in Control Systems, C. T.  Leondes, ed., pages 293–340. Elsevier, 1966. [doi:10.1016/B978-1-4831-6716-9.50011-4](https://doi.org/10.1016/B978-1-4831-6716-9.50011-4)\n",
1938     "\n",
1939     "M. Vejmelka, A. K. Kochanski, and J. Mandel. *Data assimilation of dead fuel moisture observations from remote automatic weather stations.* International Journal of Wildland Fire, 25:558– 568, 2016. [doi:10.1071/WF14085](https://doi.org/10.1071/WF14085)\n"
1940    ]
1941   }
1942  ],
1943  "metadata": {
1944   "colab": {
1945    "collapsed_sections": [],
1946    "name": "fmda_kf_local.ipynb",
1947    "provenance": []
1948   },
1949   "kernelspec": {
1950    "display_name": "Python 3 (ipykernel)",
1951    "language": "python",
1952    "name": "python3"
1953   },
1954   "language_info": {
1955    "codemirror_mode": {
1956     "name": "ipython",
1957     "version": 3
1958    },
1959    "file_extension": ".py",
1960    "mimetype": "text/x-python",
1961    "name": "python",
1962    "nbconvert_exporter": "python",
1963    "pygments_lexer": "ipython3",
1964    "version": "3.7.13"
1965   }
1966  },
1967  "nbformat": 4,
1968  "nbformat_minor": 1