1
+ import datetime
1
2
import json
2
- import math
3
3
import os
4
4
5
5
import cartopy
6
+ import matplotlib
6
7
import matplotlib .cm as cm
7
8
import matplotlib .colors as colors
8
9
import matplotlib .pyplot as plt
@@ -56,13 +57,55 @@ def _run_total_users_report(property_id):
56
57
57
58
def get_total_users (PORTAL_ID , FOUNDATIONS_ID , COOKBOOKS_ID ):
58
59
metrics_dict = {}
60
+ metrics_dict ['Now' ] = str (datetime .datetime .now ())
59
61
metrics_dict ['Portal' ] = _run_total_users_report (PORTAL_ID )
60
62
metrics_dict ['Foundations' ] = _run_total_users_report (FOUNDATIONS_ID )
61
63
metrics_dict ['Cookbooks' ] = _run_total_users_report (COOKBOOKS_ID )
62
64
with open ('portal/metrics/user_metrics.json' , 'w' ) as outfile :
63
65
json .dump (metrics_dict , outfile )
64
66
65
67
68
+ def _run_active_users_this_year (property_id ):
69
+ current_year = datetime .datetime .now ().year
70
+ start_date = f'{ current_year } -01-01'
71
+
72
+ request = RunReportRequest (
73
+ property = f'properties/{ property_id } ' ,
74
+ dimensions = [Dimension (name = 'date' )],
75
+ metrics = [Metric (name = 'activeUsers' )],
76
+ date_ranges = [DateRange (start_date = start_date , end_date = 'today' )],
77
+ )
78
+ response = client .run_report (request )
79
+
80
+ dates = []
81
+ user_counts = []
82
+ for row in response .rows :
83
+ date_str = row .dimension_values [0 ].value
84
+ date = datetime .datetime .strptime (date_str , '%Y%m%d' )
85
+ dates .append (date )
86
+ user_counts .append (int (row .metric_values [0 ].value ))
87
+
88
+ return zip (* sorted (zip (dates , user_counts ), key = lambda x : x [0 ]))
89
+
90
+
91
+ def plot_projects_this_year (PORTAL_ID , FOUNDATIONS_ID , COOKBOOKS_ID ):
92
+ portal_dates , portal_users = _run_active_users_this_year (PORTAL_ID )
93
+ foundations_dates , foundations_users = _run_active_users_this_year (FOUNDATIONS_ID )
94
+ cookbooks_dates , cookbooks_users = _run_active_users_this_year (COOKBOOKS_ID )
95
+
96
+ plt .figure (figsize = (10 , 5.5 ))
97
+ plt .title ('Year-to-Date Pythia Active Users' , fontsize = 15 )
98
+
99
+ plt .plot (portal_dates , portal_users , color = 'purple' , label = 'Portal' )
100
+ plt .plot (foundations_dates , foundations_users , color = 'royalblue' , label = 'Foundations' )
101
+ plt .plot (cookbooks_dates , cookbooks_users , color = 'indianred' , label = 'Cookbooks' )
102
+
103
+ plt .legend (fontsize = 12 , loc = 'upper right' )
104
+
105
+ plt .xlabel ('Date' , fontsize = 12 )
106
+ plt .savefig ('portal/metrics/thisyear.png' , bbox_inches = 'tight' )
107
+
108
+
66
109
def _run_top_pages_report (property_id ):
67
110
request = RunReportRequest (
68
111
property = f'properties/{ property_id } ' ,
@@ -72,72 +115,56 @@ def _run_top_pages_report(property_id):
72
115
)
73
116
response = client .run_report (request )
74
117
75
- page_views = {}
118
+ views_dict = {}
76
119
for row in response .rows :
77
120
page = row .dimension_values [0 ].value
78
121
views = int (row .metric_values [0 ].value )
79
- page_views [page ] = views
80
-
81
- top_10_pages = sorted (page_views .items (), key = lambda item : item [1 ], reverse = True )[:10 ]
82
- return {page : views for page , views in top_10_pages }
83
-
84
-
85
- def plot_top_pages (portal_id , foundations_id , cookbooks_id ):
86
- portal_page_views = _run_top_pages_report (portal_id )
87
- portal_pages = []
88
- portal_sorted = {k : v for k , v in sorted (portal_page_views .items (), key = lambda item : item [1 ])}
89
- portal_views = portal_sorted .values ()
90
- for key in portal_sorted :
91
- newkey = key .split ('—' )[0 ]
92
- portal_pages .append (newkey )
93
-
94
- foundations_page_views = _run_top_pages_report (foundations_id )
95
- foundations_pages = []
96
- foundations_sorted = {k : v for k , v in sorted (foundations_page_views .items (), key = lambda item : item [1 ])}
97
- foundations_views = foundations_sorted .values ()
98
- for key in foundations_sorted :
99
- newkey = key .split ('—' )[0 ]
100
- foundations_pages .append (newkey )
101
-
102
- cookbooks_page_views = _run_top_pages_report (cookbooks_id )
103
- cookbooks_pages = []
104
- cookbooks_sorted = {k : v for k , v in sorted (cookbooks_page_views .items (), key = lambda item : item [1 ])}
105
- cookbooks_views = cookbooks_sorted .values ()
106
- for key in cookbooks_page_views :
107
- newkey = key .split ('—' )[0 ]
108
- cookbooks_pages .insert (0 , newkey )
122
+ views_dict [page ] = views
109
123
110
- pages = cookbooks_pages + foundations_pages + portal_pages
124
+ top_pages = sorted (views_dict .items (), key = lambda item : item [1 ], reverse = True )[:5 ]
125
+ pages = [page .split ('—' )[0 ] for page , _ in top_pages ]
126
+ views = [views for _ , views in top_pages ]
127
+
128
+ return pages [::- 1 ], views [::- 1 ]
111
129
112
- fig , ax = plt .subplots (figsize = (10 , 8 ))
113
- plt .title ('All-Time Top Pages' )
114
130
115
- views_max = int (math .ceil (max (portal_views ) / 10000.0 )) * 10000
116
- ax .set_xlim ([0 , views_max ])
131
+ def plot_top_pages (PORTAL_ID , FOUNDATIONS_ID , COOKBOOKS_ID ):
132
+ portal_pages , portal_views = _run_top_pages_report (PORTAL_ID )
133
+ foundations_pages , foundations_views = _run_top_pages_report (FOUNDATIONS_ID )
134
+ cookbooks_pages , cookbooks_views = _run_top_pages_report (COOKBOOKS_ID )
117
135
118
- y = np .arange (10 )
119
- y2 = np .arange (11 , 21 )
120
- y3 = np .arange (22 , 32 )
136
+ pages = cookbooks_pages + foundations_pages + portal_pages
137
+
138
+ fig , ax = plt .subplots (figsize = (10 , 5.5 ))
139
+ plt .title ('All-Time Top Pages' , fontsize = 15 )
140
+
141
+ y = np .arange (5 )
142
+ y2 = np .arange (6 , 11 )
143
+ y3 = np .arange (12 , 17 )
121
144
y4 = np .append (y , y2 )
122
145
y4 = np .append (y4 , y3 )
123
146
124
147
bar1 = ax .barh (y3 , portal_views , align = 'center' , label = 'Portal' , color = 'purple' )
125
148
bar2 = ax .barh (y2 , foundations_views , align = 'center' , label = 'Foundations' , color = 'royalblue' )
126
149
bar3 = ax .barh (y , cookbooks_views , align = 'center' , label = 'Cookbooks' , color = 'indianred' )
127
150
128
- ax .set_yticks (y4 , labels = pages )
151
+ ax .set_yticks (y4 , labels = pages , fontsize = 12 )
152
+
153
+ ax .bar_label (bar1 , fmt = _format_rounding , padding = 5 , fontsize = 10 )
154
+ ax .bar_label (bar2 , fmt = _format_rounding , padding = 5 , fontsize = 10 )
155
+ ax .bar_label (bar3 , fmt = _format_rounding , padding = 5 , fontsize = 10 )
129
156
130
- ax .bar_label ( bar1 , fmt = _format_rounding )
131
- ax .bar_label ( bar2 , fmt = _format_rounding )
132
- ax .bar_label ( bar3 , fmt = _format_rounding )
157
+ ax .set_xscale ( 'log' )
158
+ ax .set_xlim ([ 10 , 10 ** 5 ] )
159
+ ax .set_xlabel ( 'Page Views' , fontsize = 12 )
133
160
134
- plt .legend ()
161
+ plt .legend (fontsize = 12 , loc = 'lower right' )
135
162
plt .savefig ('portal/metrics/toppages.png' , bbox_inches = 'tight' )
136
163
137
164
138
- def _run_usersXcountry_report (foundations_id ):
165
+ def _run_usersXcountry_report (property_id ):
139
166
request = RunReportRequest (
140
- property = f'properties/{ foundations_id } ' ,
167
+ property = f'properties/{ property_id } ' ,
141
168
dimensions = [Dimension (name = 'country' )],
142
169
metrics = [Metric (name = 'activeUsers' )],
143
170
date_ranges = [DateRange (start_date = '2020-03-31' , end_date = 'today' )],
@@ -153,8 +180,8 @@ def _run_usersXcountry_report(foundations_id):
153
180
return user_by_country
154
181
155
182
156
- def plot_usersXcountry (foundations_id ):
157
- users_by_country = _run_usersXcountry_report (foundations_id )
183
+ def plot_usersXcountry (FOUNDATIONS_ID ):
184
+ users_by_country = _run_usersXcountry_report (FOUNDATIONS_ID )
158
185
159
186
dict_api2cartopy = {
160
187
'Tanzania' : 'United Republic of Tanzania' ,
@@ -178,42 +205,49 @@ def plot_usersXcountry(foundations_id):
178
205
179
206
fig = plt .figure (figsize = (10 , 4 ))
180
207
ax = plt .axes (projection = cartopy .crs .PlateCarree (), frameon = False )
181
- ax .set_title ('Pythia Foundations Unique Users by Country' )
208
+ ax .set_title ('Pythia Foundations Users by Country' , fontsize = 15 )
182
209
183
210
shapefile = cartopy .io .shapereader .natural_earth (category = 'cultural' , resolution = '110m' , name = 'admin_0_countries' )
184
211
reader = cartopy .io .shapereader .Reader (shapefile )
185
212
countries = reader .records ()
186
213
187
214
colormap = plt .get_cmap ('Blues' )
188
- colormap .set_extremes (under = 'grey' )
189
- vmax = int (math .ceil (max (users_by_country .values ()) / 100.0 )) * 100
190
- norm = colors .LogNorm (vmin = 1 , vmax = vmax )
191
- mappable = cm .ScalarMappable (norm = norm , cmap = colormap )
215
+ newcmp = colors .ListedColormap (colormap (np .linspace (0.2 , 1 , 128 )))
216
+ newcmp .set_extremes (under = 'grey' )
217
+
218
+ norm = colors .LogNorm (vmin = 1 , vmax = max (users_by_country .values ()))
219
+ mappable = cm .ScalarMappable (norm = norm , cmap = newcmp )
192
220
193
221
for country in countries :
194
222
country_name = country .attributes ['SOVEREIGNT' ]
195
223
if country_name in users_by_country .keys ():
196
- facecolor = colormap ((users_by_country [country_name ] / 105 ))
197
-
224
+ facecolor = newcmp (norm (users_by_country [country_name ]))
198
225
ax .add_geometries (
199
- [country .geometry ], cartopy .crs .PlateCarree (), facecolor = facecolor , edgecolor = 'white' , linewidth = 0.7
226
+ [country .geometry ],
227
+ cartopy .crs .PlateCarree (),
228
+ facecolor = facecolor ,
229
+ edgecolor = 'white' ,
230
+ linewidth = 0.7 ,
231
+ norm = matplotlib .colors .LogNorm (),
200
232
)
201
233
else :
202
234
ax .add_geometries (
203
235
[country .geometry ], cartopy .crs .PlateCarree (), facecolor = 'grey' , edgecolor = 'white' , linewidth = 0.7
204
236
)
205
237
206
238
cax = fig .add_axes ([0.1 , - 0.015 , 0.67 , 0.03 ])
207
- fig .colorbar (mappable = mappable , cax = cax , spacing = 'uniform' , orientation = 'horizontal' , extend = 'min' )
239
+ cbar = fig .colorbar (mappable = mappable , cax = cax , spacing = 'uniform' , orientation = 'horizontal' , extend = 'min' )
240
+ cbar .set_label ('Unique Users' )
208
241
209
242
props = dict (boxstyle = 'round' , facecolor = 'white' , edgecolor = 'white' )
210
- ax .text (1.01 , 0.5 , top_10_text , transform = ax .transAxes , fontsize = 9 , verticalalignment = 'center' , bbox = props )
243
+ ax .text (1.01 , 0.5 , top_10_text , transform = ax .transAxes , fontsize = 12 , verticalalignment = 'center' , bbox = props )
211
244
212
245
plt .tight_layout ()
213
246
plt .savefig ('portal/metrics/bycountry.png' , bbox_inches = 'tight' )
214
247
215
248
216
249
if __name__ == '__main__' :
217
250
get_total_users (PORTAL_ID , FOUNDATIONS_ID , COOKBOOKS_ID )
251
+ plot_projects_this_year (PORTAL_ID , FOUNDATIONS_ID , COOKBOOKS_ID )
218
252
plot_top_pages (PORTAL_ID , FOUNDATIONS_ID , COOKBOOKS_ID )
219
- plot_usersXcountry (str ( FOUNDATIONS_ID ) )
253
+ plot_usersXcountry (FOUNDATIONS_ID )
0 commit comments