Skip to content

Commit 935e1d3

Browse files
authored
String manipulation on GH secret in metrics automation AND contributor's instructions (#414)
* Merge branch 'main' of https://github.com/jukent/projectpythia.github.io * add metrics steps to contributing guide * add note about security * pre-commit * sp
1 parent 8bc81bb commit 935e1d3

File tree

2 files changed

+87
-2
lines changed

2 files changed

+87
-2
lines changed

.github/workflows/get-metrics.py

+5-2
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,8 @@
2020

2121
# Access Secrets
2222
PRIVATE_KEY_ID = os.environ.get('PRIVATE_KEY_ID')
23-
PRIVATE_KEY = os.environ.get('PRIVATE_KEY')
23+
# Ensure GH secrets doesn't intrudce extra '\' new line characters (related to '\' being an escape character)
24+
PRIVATE_KEY = os.environ.get('PRIVATE_KEY').replace('\\n', '\n')
2425

2526
credentials_dict = {
2627
'type': 'service_account',
@@ -41,7 +42,9 @@
4142
except google.auth.exceptions.MalformedError as e:
4243
print('Malformed Error:', repr(e))
4344
# Insight into reason for failure without exposing secret key
44-
print('Length of PRIVATE_KEY:', len(PRIVATE_KEY)) # 0: Secret not found, ~734: Secret malformed
45+
# 0: Secret not found, else malformed
46+
# 706: extra quote, 732: extra '\', 734: both
47+
print('Length of PRIVATE_KEY:', len(PRIVATE_KEY))
4548

4649
pre_project_date = '2020-03-31' # Random date before project start
4750

CONTRIBUTING.md

+82
Original file line numberDiff line numberDiff line change
@@ -118,3 +118,85 @@ Once a pull request has passed all tests, including the `preview-site` check, th
118118
![CI-check](/portal/_static/images/deploy-site-CI-check.png)
119119
120120
![Netlify Preview](/portal/_static/images/netlify-preview.png)
121+
122+
123+
## Instructions for intacting with the Google Analytics API
124+
125+
### Setting up the Virtual Environment
126+
127+
Analytics must be run on a virtual environment. To create and activate this environment, with the necessary `google-analytics-data` package, the terminal commands are:
128+
129+
```
130+
python -m venv analytics-api-env
131+
source analytics-api-env/bin/activate
132+
pip install google-analytics-data
133+
```
134+
135+
Replace 'analytics-api-env' with any new environment name. Also, `pip install` any other packages you may want for your analytics work.
136+
137+
### Setting up Credentials
138+
139+
To interact with the Google Analytics API locally you need to download the credentials file. This file has been uploaded to the ProjectPythia Google Drive and lives in the Analytics_API folder.
140+
141+
**This credentials file needs to be kept secure**, especially the `private_key` field. **Do NOT share this file.** If you do not have access to our Google Drive and need access to this file, please reach out to the team on discourse or in a community meeting.
142+
143+
The credentials file will have a name similar to `cisl-vast-pythia-{letters and numbers}.json`. This file may be replaced intermittently with a slightly different alphanumeric string for additional security.
144+
145+
One way to ensure that your Python script is using the correct credentials file is to read it as a dictionary and pass that into your API client at the begging of your script.
146+
147+
```
148+
from google.analytics.data_v1beta import BetaAnalyticsDataClient
149+
from google.analytics.data_v1beta.types import DateRange, Dimension, Metric, RunReportRequest
150+
151+
with open('{credentials-file-path}') as json_file:
152+
credentials_dict = json.load(json_file)
153+
154+
client = BetaAnalyticsDataClient.from_service_account_info(credentials_dict)
155+
```
156+
157+
Recommended and commonly needed import statements are also shown at the script beginning.
158+
159+
### Making a request
160+
161+
Below is a sample function for calling an Analytics API request.
162+
163+
```
164+
def _run_analytics_request(property_id):
165+
request = RunReportRequest(
166+
property=f'properties/{property_id}',
167+
dimensions=[Dimension(name='date')],
168+
metrics=[Metric(name='activeUsers')],
169+
date_ranges=[DateRange(start_date='2024-01-01', end_date='today')],
170+
)
171+
response = client.run_report(request)
172+
return response
173+
```
174+
175+
This function demonstrates how to format your `RunReportRequest()` arguments, notably the `dimensions` and `metrics` fields, as well as the expected date formatting in `date_ranges`.
176+
177+
This [Google Analytics API Schema](https://developers.google.com/analytics/devguides/reporting/data/v1/api-schema) documentation lists all of the available dimension and metric keys that can be passed into your request.
178+
179+
`property_id` is a 9-digit number associated with the project you are interested in. This number can be found on the Analytics project page. For Project Pythia, our three different property IDs are:
180+
```
181+
PORTAL_ID = '266784902'
182+
FOUNDATIONS_ID = '281776420'
183+
COOKBOOKS_ID = '324070631'
184+
```
185+
186+
### Working with your request output
187+
188+
Your Google Analytics response is formatted in a series of rows that each have the key `dimension_value` and `metric_value`. You may find it easier to work with your data in a dictionary or tuple. For the single dimension of "date" and metric of "activeUsers" as specified in our example function, here is what your data manipulation may look like before you can carry out additional analysis.
189+
190+
```
191+
dates=[]
192+
users=[]
193+
for row in response.rows:
194+
date_str = row.dimension_values[0].value
195+
date = datetime.datetime.strptime(date_str, '%Y%m%d')
196+
dates.append(date)
197+
users.append(int(row.metric_values[0].value))
198+
199+
dates, users = zip(*sorted(zip(dates, user_counts), key=lambda x: x[0]))
200+
```
201+
202+
One thing to note is that your Analytics response rows are not automatically chronological, so in this example we zipped our sorted tuple to ensure that the dates are in the expected order.

0 commit comments

Comments
 (0)