Skip to main content
  • Home
  • Development
  • Documentation
  • Donate
  • Operational login
  • Browse the archive

swh logo
SoftwareHeritage
Software
Heritage
Archive
Features
  • Search

  • Downloads

  • Save code now

  • Add forge now

  • Help

https://doi.org/10.5281/zenodo.11373234
23 April 2025, 21:23:49 UTC
  • Code
  • Branches (0)
  • Releases (1)
  • Visits
    • Branches
    • Releases
      • 1
      • 1
    • dbe46e6
    • /
    • saridout-kpd_model_code-79b3e8c
    • /
    • code
    • /
    • models.py
    Raw File Download

    To reference or cite the objects present in the Software Heritage archive, permalinks based on SoftWare Hash IDentifiers (SWHIDs) must be used.
    Select below a type of object currently browsed in order to display its associated SWHID and permalink.

    • content
    • directory
    • snapshot
    • release
    origin badgecontent badge
    swh:1:cnt:704b0579016866b776a627aa96bac2cf05553683
    origin badgedirectory badge
    swh:1:dir:e88e847ae29d3ec5ede68f2414558c09ff1eb79e
    origin badgesnapshot badge
    swh:1:snp:0ca2497b1149c0b82a8c17fc5ad8aab11076782d
    origin badgerelease badge
    swh:1:rel:81e1c51c151fea06f525ddba6ba1535b1ee24796

    This interface enables to generate software citations, provided that the root directory of browsed objects contains a citation.cff or codemeta.json file.
    Select below a type of object currently browsed in order to generate citations for them.

    • content
    • directory
    • snapshot
    • release
    (requires biblatex-software package)
    Generating citation ...
    (requires biblatex-software package)
    Generating citation ...
    (requires biblatex-software package)
    Generating citation ...
    (requires biblatex-software package)
    Generating citation ...
    models.py
    """
    Models are built up from individual reactions.
    Reactions should fix their parameters and array indices they act on at compile time,
    on the basis of given parameters and of a list of substrates.
    """
    
    import numpy as onp
    import jax.numpy as jnp
    from jax import jit
    import diffrax
    from numpy.polynomial import Polynomial
    #from orthax import Polynomial
    
    import pandas as pd
    
    
    def find_index(all_substrates, substrate):
        """
        Args:
            all_substrates (string): string with each character being a substrate (e.g., "GI")
            substrate (string): single character identifying a substrate (e.g, "G")
        Returns:
            int: index of substrate in sorted list of all_substrates
        """
        s = onp.unique(list(all_substrates))
        i = onp.searchsorted(s, substrate)
        assert s[i] == substrate
        return i
    
    
    def N_substrates(all_substrates):
        """
        Args:
            all_substrates (string): string with each character being a substrate (e.g., "GI")
        Returns:
            int: Number of substrates in the model.
        """
        return len(onp.unique([list(all_substrates)]))
    
    
    def hill_catalysis_reaction(all_substrates, params):
        """
        Args:
            all_substrates (string): string with each character being a substrate (e.g., "GI")
            params (dict): dictionary of hill function parameters and identification of reactants
        Returns:
            function: A function that returns contribution to d[all concentrations]/dt,
                from a reaction which converts of r to p with rate k r c^n / (K^n + c^n)
        """
        assert "k" in params
        assert "K" in params
        assert "n" in params
        assert "c" in params
        assert ("r" in params) or ("p" in params)
    
        if "r" in params and params["r"]:
            i = find_index(all_substrates, params["r"])
        if "p" in params and params["p"]:
            j = find_index(all_substrates, params["p"])
        k = find_index(all_substrates, params["c"])
        N = N_substrates(all_substrates)
        def reaction(_t, x, _):
            C = x[k]
            R = params['k']/(1+ (params['K']/C)**params['n'])
            f = jnp.zeros(N)
            if "r" in params and params["r"]:
                R = R * x[i]
                f = f.at[i].set(-R)
            if "p" in params and params["p"]:
                f = f.at[j].set(R)
            return f
    
        return reaction
    
    def linear_catalysis_reaction(all_substrates, params):
        """
        Args:
            all_substrates (string): string with each character being a substrate (e.g., "GI")
            params (dict): dictionary of rate constant k and identification of reactants
        Returns:
            function: A function that returns contribution to d[all concentrations]/dt,
                from a reaction which converts of r to p with rate k c r
        """
        assert "k" in params
        assert ("r" in params) or ("c" in params)
        assert ("r" in params) or ("p" in params)
    
        if "r" in params and params["r"]:
            i = find_index(all_substrates, params["r"])
        if "p" in params and params["p"]:
            j = find_index(all_substrates, params["p"])
        if "c" in params and params["c"]:
            k = find_index(all_substrates, params["c"])
        N = N_substrates(all_substrates)
        def reaction(_t, x, _):
            R = params['k']
            f = jnp.zeros(N)
            if 'c' in params and params['c']:
                R = R *x[k]
            if 'r' in params and params["r"]:
                R = R * x[i]
                f = f.at[i].set(-R)
            if 'p' in params and params["p"]:
                f = f.at[j].set(R)
            return f
    
        return reaction
    
    
    
    def add_reactions(reactions):
        """
        Args:
            reactions (list): list of reaction functions which compute terms in dX/dt
        Returns:
            function: A function that computes the sum dX/dt of these terms.
        """
        @jit
        def f(t, x, _):
            return jnp.sum(jnp.array([reaction(t,x,_) for reaction in reactions]), axis=0)
    
        return f
    
    def insulin_secretion_reaction(all_substrates, params):
        """
        Args:
            all_substrates (string): string with each character being a substrate (e.g., "GIBb")
            params (dict): dictionary of constants k, n, K
        Returns:
            function: A function that returns contribution to d[all concentrations]/dt,
                from insulin secretion at rate k B G^n / (G^n + K^n)
        """
        i = find_index(all_substrates, "G")
        j = find_index(all_substrates, "I")
        k = find_index(all_substrates, "B")
        l = find_index(all_substrates, "M")
    
        N = N_substrates(all_substrates)
        def reaction(_t, x, _):
            C = x[i]
            R = params['k']*(C**params['n'])/(C**params['n'] + params['K']**params['n'])
            R = R*x[k]*x[l]
            f = jnp.zeros(N)
            f = f.at[j].set(R)
            return f
    
        return reaction
    
    def flux(all_substrates, params):
        i = find_index(all_substrates, params['s'])
        N = N_substrates(all_substrates)
    
        def reaction(t, _x, _):
            R = jnp.where(jnp.logical_and( t >= params['t0'], t <=params['t1']), params['f'], 0.0)
    
            f = jnp.zeros(N)
            f = f.at[i].set(R)
            return f
    
        return reaction
    
    def integrator(f, tol=1e-3, t1=50, dt=1, max_steps=10000):
        term = diffrax.ODETerm(f)
        solver = diffrax.Kvaerno5()
        sc = diffrax.PIDController(rtol=tol, atol=tol)
    
        def g(x0):
            t = jnp.arange(0, t1+dt/2, dt)
            dt0 = dt
            saveat = diffrax.SaveAt(ts=t)
            return diffrax.diffeqsolve(term, solver, t[0], t[-1], dt0, x0, saveat=saveat,
                                       stepsize_controller=sc, max_steps=max_steps).ys
    
        h = jit(g)
        h: callable #jit confuses pylint, it seems
        return h
    
    
    def model(reactions, max_steps=10000,t1=50,tol=1e-3, dt=1):
        f = add_reactions(reactions)
        return integrator(f, max_steps=max_steps,t1=t1,tol=tol, dt=dt)
    
    def map_substrates(substrates):
        all_substrates = onp.unique(list(substrates))
        x = jnp.zeros(N_substrates(all_substrates))
    
        for i,s in enumerate(all_substrates):
            x = x.at[i].set(substrates[s])
    
        return x
    
    def hill(G_0, n,k):
        return [Polynomial((0,)*n + (k,)), Polynomial((G_0**n,) + (0,)*(n-1) + (1,))]
    
    def hill_plus_one(G_0, n,k):
        return [Polynomial((0,)*n + (k,)) + Polynomial((G_0**n,) + (0,)*(n-1) + (1,)),
                Polynomial((G_0**n,) + (0,)*(n-1) + (1,))]
    
    def evaluate(f, X):
        return f[0](X) / f[1](X)
    
    def get_f(params):
        return hill(params["f_K"], params["f_n"],params["β"])
    
    def get_r(params):
        return hill_plus_one(params["g_K"], params["g_n"],params["g_k"])
    
    
    def final_poly(params):
        #f and r should both be [p_numerator, p_denominator]
        #1/r(G) = fraction of active beta cells. f(G) = secretion function x number of beta cells
        m_0 = params["m_0"]
        M = params["M"]
        S_E = params["S_E"]
        S_I = params["S_I"]
        I_0 = params["I_0"]
        F_I = params["F_I"] if "F_I" in params else 0.0
        G = Polynomial([0,1])
        f = get_f(params)
        r = get_r(params)
    
        return (m_0*f[1]*r[0]**2 + M*(f[0]*r[1] + I_0*f[1]*r[0]+F_I*r[0]*f[1])*r[0])*f[1] - \
            G*(S_I*(f[0]*r[1]+F_I*r[0]*f[1]) + S_E*f[1]*r[0])*((I_0+F_I)*r[0]*f[1] + r[1]*f[0])
    
    def g_poly(params, frac):
        #fixed active beta cell fraction
        m_0 = params["m_0"]
        M = params["M"]
        S_E = params["S_E"]
        S_I = params["S_I"]
        I_0 = params["I_0"]
        F_I = params["F_I"] if "F_I" in params else 0.0
        G = Polynomial([0,1])
        f = get_f(params)
        r = [1, frac]
    
        return (m_0*f[1]*r[0]**2 + M*(f[0]*r[1] + I_0*f[1]*r[0]+F_I*r[0]*f[1])*r[0])*f[1] - \
            G*(S_I*(f[0]*r[1]+F_I*r[0]*f[1]) + S_E*f[1]*r[0])*((I_0+F_I)*r[0]*f[1] + r[1]*f[0])
    
    def eq_beta_frac(params, G):
        r = get_r(params)
        return 1/evaluate(r, G)
    
    def positive_roots(p):
        r = p.roots()
        arg = onp.where(onp.logical_and(onp.abs(r.imag) < 1e-14, r.real > 0))
        return r.real[arg]
    
    
    def adaptive_beta(params):
        m_0 = params["m_0"]
        S_E = params["S_E"]
        S_I = params["S_I"]
        I_0 = params["I_0"]
        G_0 = params["G_0"]
    
        f = get_f(dict(params,β=1))
        r = get_r(params)
    
        f0 = evaluate(f, G_0) / evaluate(r, G_0)
    
    
        a = G_0*S_I*f0**2
        b = (G_0*S_E + G_0*S_I*I_0)*f0
        c = G_0*S_E*I_0 - m_0
    
        return (-b + onp.sqrt(b**2 - 4*a*c))/(2*a)
    
    def adaptive_m0(params):
        I = params["I_targ"]
        S_E = params["S_E"]
        S_I = params["S_I"]
        I_0 = params["I_0"]
        G_0 = params["G_0"]
    
        return G_0*(S_E+S_I*I)*(I_0+I)
    
    def low_eq(params):
        G = positive_roots(final_poly(params))[0]
        r = get_r(params)
    
        return G, 1.0/evaluate(r, G)
    
    def low_eq2(params):
        G = positive_roots(final_poly(params))[0]
        r = get_r(params)
        f = get_f(params)
    
        F = params['F_I'] if "F_I" in params else 0.0
    
        return onp.array([G, F + evaluate(f,G)/evaluate(r, G)])
    
    def all_eq(params):
        G = positive_roots(final_poly(params))
        r = get_r(params)
        frac = [1.0/evaluate(r, g) for g in G]
    
        return G, frac
    
    
    
    def fast_model(params):
        s = "GI"
        g=params['γ'] # we will come to this later
    
    
        #everything but meals
        HGP = hill_catalysis_reaction(s, {"k": params["m_0"] / params["I_0"],
                                          "p": "G", "n": -1, "K": params["I_0"], "c": "I"})
    
        glucose_effectiveness = linear_catalysis_reaction(s, {"k": params["S_E"], "r": "G"})
    
        insulin_disposal = linear_catalysis_reaction(s, {"k": g, "r": "I"})
    
        @jit
        def fast(a, S_I, b, G_0, I_0):
            I = params['F_I']*params['γ']
    
            t0 = 0
    
            I_secretion = hill_catalysis_reaction(s,{"k": b, "n": params['f_n'],
                                                     "K": params['f_K'], "c": "G", "p": "I"})
            insulin_glucose_disposal = linear_catalysis_reaction(s, {"k": S_I, "r": "G", "c": "I"})
    
            base_reactions = [HGP, insulin_glucose_disposal,
                              glucose_effectiveness, I_secretion, insulin_disposal]
    
            x = jnp.array((G_0, I_0) ).reshape((1,-1))
            meal_insulin_schedule = [(12/48,0.0, I),(13/48,a, I),
                                     (24/48,0.0, I),(25/48,a, I),
                                     (36/48,0.0, I), (37/48, a, I), (1.0,0, I)]
    
            for (t,  M, I) in meal_insulin_schedule:
                reactions = base_reactions + [flux(s, {"s": "G", "t0": 0, "t1": t, "f": M}),
                                  flux(s, {"s": "I", "t0": 0, "t1": t, "f": I})]
                m = model(reactions, t1 = t-t0, dt=1/(60*24),tol=1e-8,max_steps=1000)
                X = m(x[-1])
                x = jnp.concatenate((x[:-1],X))
                t0 = t
    
            all_t = jnp.arange(0, t0+1/(2*60*24), 1/(60*24))
    
            return all_t, x
    
        return fast
    
    def r_table(params, a_list=None,F_I=0.0,beta_range=None,
                SI_range=None,SI_samples=None, beta_samples=1000):
    
        simulate_fast = fast_model(dict(params, F_I=F_I/params['γ']))
        output = []
        r = get_r(params)
        for _beta in onp.linspace(beta_range[0],beta_range[1], beta_samples):
            for _S_I in onp.linspace(SI_range[0],SI_range[1], SI_samples):
                if _beta == 0.0 and F_I == 0.0: #trick to avoid issues with error control when I=0.0
                    beta = 1.0
                    S_I = 0.0
                else:
                    beta = _beta
                    S_I = _S_I
    
                G_0, I_0 = low_eq2(params | {"β": beta / params['γ'],
                        "S_I": S_I,
                        "g_k": 0.0,
                        "F_I": F_I / params['γ']})
    
                for a in a_list:
                    t,x = simulate_fast(a, S_I, beta, G_0, I_0)
    
                    G = onp.array(x[:,0])
                    out = onp.trapz(evaluate(r, G), x=t) -1
                    output.append(
                        pd.DataFrame({"a": a, "S_I": _S_I,
                                      "beta": _beta, "r": out, "G_0": G_0}, index=[0])
                    )
    
    
        return pd.concat(output, axis=0, ignore_index=True)
    
    
    
    def resample(B, r, B0):
        arg = onp.where(B < B0)
        return B[arg]/B0, r[arg]
    
    
    def numerical_roots(beta, r, B0, params, S_I):
        B, r = resample(beta, r, B0)
    
        BB = onp.linspace(B[0], B[-1], 10000)
        X = -BB*onp.interp(BB, B, r)  + (1-BB)
        b_roots = B0*BB[onp.where(X[:-1]*X[1:] < 0)]
        if len(b_roots) < 1:
            b_roots = [B0]
        return [low_eq2(dict(params, β=b/params['γ'], S_I=S_I, g_k=0.0))[0] for b in b_roots][::-1]
    
    def SI_beta_scan(params, SI_range=None, beta_range=None,
                     SI_samples=10,beta_samples=1000, G_0=80, a=None, table_beta_samples=3000):
        r = get_r(params)
        G = onp.zeros((SI_samples, beta_samples))
    
        #0 = 1 low min, 1 = 2 min, 2 = 1 high min
        classification = onp.zeros((SI_samples, beta_samples), dtype=int)
        def c(g, roots):
            if g <= roots[0]:
                return 0
            if g <= roots[1]:
                return 1
            assert g > roots[-1]
            return 2
    
        SI = onp.linspace(SI_range[0], SI_range[1], SI_samples)
        if beta_range[0] > 0:
            betas = onp.linspace(beta_range[0], beta_range[1], beta_samples)
        else:
            betas = onp.linspace(beta_range[0], beta_range[1], beta_samples+1)[1:]
        stuff = r_table(params, a_list = [a],
                        SI_range=SI_range, SI_samples=SI_samples,
                        beta_range=[beta_range[0]/100,beta_range[1]], beta_samples=table_beta_samples)
    
        for i, S_I in enumerate(SI):
            s = stuff[stuff['S_I']==S_I]
            r = s['r'].values
            beta = s['beta'].values
            roots = onp.zeros((len(betas),3))
            boundary_roots = onp.full(4,-onp.inf)
            boundary_roots[0] = onp.inf
    
            for j, B0 in enumerate(betas):
                R = numerical_roots(beta, r, B0, params, S_I)
                try:
                    roots[j] = R
                except Exception as e:
                    print(e)
                    print(R)
                    return {"beta": beta, "r": r, "B0": B0, "params": params, "S_I": S_I}
                if len(R) == 3:
                    boundary_roots[0] = onp.min((boundary_roots[0], roots[j][0]))
                    boundary_roots[1] = onp.max((boundary_roots[1], roots[j][0]))
                    boundary_roots[2] = onp.max((boundary_roots[2], roots[j][1]))
                    boundary_roots[3] = onp.max((boundary_roots[3], roots[j][2]))
    
            roots = onp.where(roots>G_0, roots, G_0)
            for j, beta in enumerate(betas):
                g = roots[j][0]
                G[i,j] = g
                classification[i,j] = c(g, boundary_roots)
    
    
        return SI, betas, G, classification
    
    def full_eq(params):
        G, frac = low_eq(params)
        B = params['beta_tot']*frac/params['γ']
        b = params['beta_tot']/params['γ']-B
        f = get_f(params)
    
        I = evaluate(f, G)*frac
    
        return map_substrates({"B":B, "b":b, "G":G, "I":I, "g":G, "M":1})
    
    def full_model_day(params, a, _I):
    
        
        s = "BbGIgM"
        #everything but meals
        HGP = hill_catalysis_reaction(s, {"k":params["m_0"]/params["I_0"],
                                          "p":"G", "n":-1, "K":params["I_0"], "c":"I"})
        glucose_effectiveness = linear_catalysis_reaction(s, {"k":params["S_E"], "r":"G"})
        insulin_glucose_disposal = linear_catalysis_reaction(s, {"k":params["S_I"],
                                                                 "r":"G", "n":-1, "c":"I"})
        insulin_secretion = insulin_secretion_reaction(s, {"k":params['γ'],
                                                           "n":params['f_n'],"K":params['f_K']})
        insulin_disposal = linear_catalysis_reaction(s, {"k":params['γ'], "r":"I"})
    
        dediff = hill_catalysis_reaction(s,
                                         {"k":params['k_r']*params['g_k'], "n":params['g_n'],
                                          "K":params['g_K'], "r":"B", "p":"b", "c":"G"})
        rediff = linear_catalysis_reaction(s, {"k": params['k_r'], "r": "b", "p": "B"})
        death1 = hill_catalysis_reaction(s, {"k": params['k_d'], "n": 2, "K": 4000, "r": "B", "c": "G"})
        death2 = hill_catalysis_reaction(s, {"k": params['k_d'], "n": 2, "K": 4000, "r": "b", "c": "G"})
    
        base_reactions = [HGP, insulin_glucose_disposal, glucose_effectiveness,
                          insulin_secretion, insulin_disposal, dediff, rediff, death1, death2]
        @jit
        def M(x):
            t0 = 0.0
            if callable(_I): #function for deciding insulin dose
                I = _I(params, s, x[-1])
            else: #hard-coded insulin dose
                I = _I
            meal_insulin_schedule = [(12/48,0.0, I),(13/48,a, I),
                             (24/48,0.0, I),(25/48,a, I),
                             (36/48,0.0, I), (37/48, a, I), (1.0,0, I)]
    
            for (t,  M, I) in meal_insulin_schedule:
                reactions = base_reactions + [ flux(s, {"s":"G", "t0":0, "t1":t, "f":M}),
                                              flux(s, {"s":"I", "t0":0, "t1":t, "f":I})]
                m = model(reactions, t1 = t-t0, dt=1/(60*24),tol=1e-6,max_steps=1000)
                X = m(x[-1])
                x = jnp.concatenate((x[:-1],X))
                t0 = t
    
            all_t = jnp.arange(0, t0+1/(2*60*24), 1/(60*24))
    
            return all_t, x
    
        return M
    
    
    def full_simulation(params, pre_sugar_days=0, sugar_days=0, post_sugar_days=0,
                        insulin_days=0, post_insulin_days=0, low_sugar=0, high_sugar=0, insulin=0):
        normal_model = full_model_day(params, low_sugar, 0.0)
        sugar_model = full_model_day(params, high_sugar, 0.0)
        insulin_model = full_model_day(params, low_sugar, insulin)
        s = "BbGIgM"
        x = full_eq(params).reshape((1,-1))
        g = [x[-1,1], ]
        B = [x[-1, 0], ]
        b = [x[-1,-1], ]
        g = [x[-1,1], ]
        I = [x[-1,find_index(s, "I")], ]
    
        def F_I(x):
            if callable(insulin):
                return insulin(params, s, x)
            else:
                return insulin
    
        F = [0.0, ]
    
        for _ in range(pre_sugar_days):
            _, x = normal_model(x[-1,:].reshape((1,-1)))
            g.append(onp.mean(x[:,1]))
            B.append(onp.mean(x[:,0]))
            b.append(onp.mean(x[:,-1]))
            I.append(onp.mean(x[:,find_index(s, "I")]))
            F.append(0.0)
    
        for _ in range(sugar_days):
            _, x = sugar_model(x[-1,:].reshape((1,-1)))
            g.append(onp.mean(x[:,1]))
            B.append(onp.mean(x[:,0]))
            b.append(onp.mean(x[:,-1]))
            I.append(onp.mean(x[:,find_index(s, "I")]))
            F.append(0.0)
    
        for _ in range(post_sugar_days):
            _, x = normal_model(x[-1,:].reshape((1,-1)))
            g.append(onp.mean(x[:,1]))
            B.append(onp.mean(x[:,0]))
            b.append(onp.mean(x[:,-1]))
            I.append(onp.mean(x[:,find_index(s, "I")]))
            F.append(0.0)
    
        for _ in range(insulin_days):
            _, x = insulin_model(x[-1,:].reshape((1,-1)))
            g.append(onp.mean(x[:,1]))
            B.append(onp.mean(x[:,0]))
            b.append(onp.mean(x[:,-1]))
            I.append(onp.mean(x[:,find_index(s, "I")]))
            F.append(F_I(x[0,:]))
    
        for _ in range(post_insulin_days):
            _, x = normal_model(x[-1,:].reshape((1,-1)))
            g.append(onp.mean(x[:,1]))
            B.append(onp.mean(x[:,0]))
            b.append(onp.mean(x[:,-1]))
            I.append(onp.mean(x[:,find_index(s, "I")]))
            F.append(0.0)
    
    
        return onp.array(g), onp.array(I), onp.array(B), onp.array(b), onp.array(F), onp.arange(len(g))
    
    def fasting_maximum_insulin(G_min):
        """
        Returns a function that decides the maximum insulin flux which will produce fasting glucose at least G_min.
        """
    
        def FI_func(params, s, x):
            β = x[find_index(s, "B")]
    
            I_0 = params["I_0"]
            m_0 = params["m_0"]
            S_I = params["S_I"]
            S_E = params["S_E"]
    
            fasting_I = (-I_0 - S_E/S_I + onp.sqrt((I_0 + S_E/S_I)**2 + 4*(m_0 / (G_min*S_I)- S_E*I_0/S_I)))/2.0
            γ = params['γ']
    
            f = get_f(params | {"β": 1})
            f_min = evaluate(f, G_min)
    
            return jnp.where(γ*fasting_I - γ*β*f_min >0, γ*fasting_I - γ*β*f_min, 0.0)
    
        return FI_func
    
    
    def fasting_max_insulin_timescale(params):
        """
        Recovery timescale under insulin treatment which fixes G at G_min.
        """
        G_roots = positive_roots(final_poly(params))
        assert len(G_roots) == 3 #if no bistablity, what are you doing?
        #check the order is right while we're at it
        assert G_roots[2] >= G_roots[1]
        assert G_roots[1] >= G_roots[0]
    
        frac_f = eq_beta_frac(params, G_roots[0])
        frac_0 = eq_beta_frac(params, G_roots[2])
    
        G_min = onp.arange(85, G_roots[0],0.5)
    
        
        h = params["k_r"]
        r = params["k_r"]*evaluate(get_r(params), G_min)
    
        τ = (1/r)*onp.log((h-r*frac_0)/(h-r*frac_f))
    
        return G_min, τ 
    
    def fasting_approx_frac_schedule(params, G_min):
        G_roots = positive_roots(final_poly(params))
        frac_f = eq_beta_frac(params, G_roots[0])
        frac_0 = eq_beta_frac(params, G_roots[2])
        
        h = params["k_r"] #in the paper's notation, this is K_RE
        r = params["k_r"]*evaluate(get_r(params), G_min) #in the paper's notation, this is k_IN g(G) + k_RE
    
        τ = (1/r)*onp.log((h-r*frac_0)/(h-r*frac_f))
    
        t = onp.arange(0, τ+1)
    
        beta_frac = (1/r)*(onp.exp(-r*t)*(r*frac_0 - h) + h)
    
        return t, beta_frac
    
    def fasting_approx_insulin_schedule(params, G_min):
        t, β = fasting_approx_frac_schedule(params, G_min)
        γ = params["γ"]
    
        f = evaluate(get_f(params), G_min)
    
    
        I_0 = params["I_0"]
        m_0 = params["m_0"]
        S_I = params["S_I"]
        S_E = params["S_E"]
    
        fasting_I = (-I_0 - S_E/S_I + onp.sqrt((I_0 + S_E/S_I)**2 + 4*(m_0 / (G_min*S_I)- S_E*I_0/S_I)))/2.0
        
        return t, γ*fasting_I - γ*β*f
        
    
        

    back to top

    Software Heritage — Copyright (C) 2015–2026, The Software Heritage developers. License: GNU AGPLv3+.
    The source code of Software Heritage itself is available on our development forge.
    The source code files archived by Software Heritage are available under their own copyright and licenses.
    Terms of use: Archive access, API— Content policy— Contact— JavaScript license information— Web API