-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmiseryindex.py
155 lines (120 loc) · 4.8 KB
/
miseryindex.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
import argparse
import configparser
import json
import numpy as np
np.set_printoptions(precision=4)
import matplotlib.pyplot as plt
plt.style.use('seaborn-white')
plt.style.use('common.mplstyle')
parser = argparse.ArgumentParser(description="Calculate misery index in terms of slope.")
parser.add_argument("profile", metavar='PROFILE.INI', help="File containing our assumptions", default='bike.ini')
parser.add_argument("--graph", metavar='SLOPE', help="Make a plot for a given slope, write no output. Slope is given as a percentage", type=float)
parser.add_argument("--power", action='store_true', help="Plot power instead of required traction force")
parser.add_argument("-o", "--output", metavar='DATA.json', help="Set output file", default='misery-index.json')
args = parser.parse_args()
config = configparser.ConfigParser()
config.read(args.profile, encoding='utf-8')
cfg_main = config['main']
ride_v_table = []
ride_P_table = []
for k, v in config['ride-profile'].items():
ride_v_table.append(float(k))
ride_P_table.append(float(v))
# ensure v/P curve intersects 0
ride_v_table.append(ride_v_table[-1] + 0.1)
ride_P_table.append(-0.01)
M = float(cfg_main['m'])
Crr = float(cfg_main['Crr'])
half_rho_cd_a2 = float(cfg_main['half-rho-cd-a2'])
G = 9.81
def cycling_power(speed, slope, wind):
""" how much power do we need in this situation
speed: cycling speed
slope: positive number is uphill, eg. 0.02 for a 2% slope
wind: positive number for headwind
All speeds in km/h
Returns power in watts"""
v = speed / 3.6
vRel = (speed + wind) / 3.6
p = M * G * (Crr + slope) * v + \
half_rho_cd_a2 * vRel * abs(vRel) * v
return np.maximum(p, 0)
v = np.arange(0.1, ride_v_table[-1] + 0.101, 0.1)
# walking profile
v_walk = np.arange(2, 6, 0.1)
pwr_walk = np.interp(v_walk, [2, 6], [140, 10])
walk_penalty = float(cfg_main['walk-penalty'])
v_ride = np.arange(ride_v_table[0], ride_v_table[-1] + 0.101, 0.1)
pwr_ride = np.interp(v_ride, ride_v_table, ride_P_table)
pwr_intersect = np.maximum(
np.interp(v, v_ride, pwr_ride, left=0),
np.interp(v, v_walk, pwr_walk, right=0))
pwr_baseline = cycling_power(v, 0, 0)
def calc_power_use(slope, show_plot=False):
wind = np.arange(-20, 20.1, 10) if args.graph is not None else np.arange(-25, 25.1, 5)
p_corr = 1 if args.power else 1 / v
pwr = [ cycling_power(v, -slope, w) for w in wind ] +\
[ cycling_power(v, slope, w) for w in wind ]
# intersection points
p_speed = np.empty([2 * len(wind)])
p_power = np.empty([2 * len(wind)])
p_speed[:] = np.nan
p_power[:] = np.nan
for i, p in enumerate(pwr):
idx = np.nonzero(np.diff(np.sign(pwr_intersect - p)))[0]
j = idx[-1]
p_speed[i] = v[j]
p_power[i] = p[j] + walk_penalty * v[j] * (v[j] < ride_v_table[0])
energy = p_power / p_speed
avg = np.average(energy)
if show_plot:
fig, ax = plt.subplots(figsize=[6, 4])
# print table
for pv, pp, pe, w in zip(p_speed, p_power, energy, np.concatenate((wind, wind))):
print(f'wind: {w:3.0f}km/h | {pv:4.1f}km/h {pp:4.1f}W {pe:5.2f}Wh/km')
print()
print(f'Average: {avg:5.2f}Wh/km')
# reference: light gray
ax.plot(v, pwr_baseline * p_corr, '--', color=(.8, .8, .8))
for pp, w in zip(pwr, np.concatenate((wind, wind))):
# tailwind: blue
if w < 0:
c = (.2, .4, 1)
elif w > 0:
# headwind: red
c = (.8, .2, .3)
else:
# still: yellow
c = (.7, .6, 0)
ax.plot(v, pp * p_corr, color=c)
# ref: green
ax.plot(v_walk, pwr_walk * (1 if args.power else 1/v_walk), '--', color=(0.0, 0.6, 0.2))
ax.plot(v_ride, pwr_ride * (1 if args.power else 1/v_ride), '--', color=(0.0, 0.6, 0.2))
# intersections
i_corr = 1 if args.power else 1 / p_speed
ax.scatter(p_speed, p_power * i_corr, 9, color=(.1, .1, .1))
ax.grid(True)
ax.set_title(f'slope: {slope*100:.0f}%')
if args.power:
ax.set_ylim([-3, 145])
ax.set_ylabel('Power (W)')
else:
ax.set_ylim([-2, max(30, 1+np.max(energy))])
ax.set_ylabel('Energy use (Wh/km)')
ax.set_xlabel('Speed (km/h)')
plt.show()
return avg
mi_list = []
pwr_0 = None
if args.graph is not None:
calc_power_use(args.graph * .01, show_plot=True)
else:
for sl in np.arange(0, 0.25001, 0.005):
pwr = calc_power_use(sl, show_plot=False)
if sl == 0:
pwr_0 = pwr
mi = pwr/pwr_0 - 1
mi_list.append(dict(slope=sl, mi=mi, p=pwr))
print(f"misery index for {100*sl:4.1f}%: {mi:4.1f} ({pwr:.2f})")
f = open(args.output, 'wt')
json.dump(mi_list, f)