mirror of https://github.com/grpc/grpc.git
commit
201fd84753
5 changed files with 159 additions and 0 deletions
@ -0,0 +1,13 @@ |
||||
runtime: python27 |
||||
api_version: 1 |
||||
threadsafe: true |
||||
|
||||
service: github-stats-tracking |
||||
|
||||
handlers: |
||||
- url: /.* |
||||
script: main.app |
||||
|
||||
libraries: |
||||
- name: ssl |
||||
version: latest |
@ -0,0 +1,19 @@ |
||||
# Copyright 2019 gRPC authors. |
||||
# |
||||
# Licensed under the Apache License, Version 2.0 (the "License"); |
||||
# you may not use this file except in compliance with the License. |
||||
# You may obtain a copy of the License at |
||||
# |
||||
# http://www.apache.org/licenses/LICENSE-2.0 |
||||
# |
||||
# Unless required by applicable law or agreed to in writing, software |
||||
# distributed under the License is distributed on an "AS IS" BASIS, |
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||
# See the License for the specific language governing permissions and |
||||
# limitations under the License. |
||||
|
||||
# appengine_config.py |
||||
from google.appengine.ext import vendor |
||||
|
||||
# Add any libraries install in the "lib" folder. |
||||
vendor.add('lib') |
@ -0,0 +1,4 @@ |
||||
cron: |
||||
- description: "daily github stats tracking job" |
||||
url: /daily |
||||
schedule: every 24 hours |
@ -0,0 +1,94 @@ |
||||
# Copyright 2019 gRPC authors. |
||||
# |
||||
# Licensed under the Apache License, Version 2.0 (the "License"); |
||||
# you may not use this file except in compliance with the License. |
||||
# You may obtain a copy of the License at |
||||
# |
||||
# http://www.apache.org/licenses/LICENSE-2.0 |
||||
# |
||||
# Unless required by applicable law or agreed to in writing, software |
||||
# distributed under the License is distributed on an "AS IS" BASIS, |
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||
# See the License for the specific language governing permissions and |
||||
# limitations under the License. |
||||
|
||||
from github import Github, Label |
||||
from datetime import datetime, timedelta |
||||
from time import time |
||||
from google.cloud import bigquery |
||||
|
||||
ACCESS_TOKEN = "" |
||||
|
||||
|
||||
def get_stats_from_github(): |
||||
# Please set the access token properly before deploying. |
||||
assert ACCESS_TOKEN |
||||
g = Github(ACCESS_TOKEN) |
||||
print g.rate_limiting |
||||
repo = g.get_repo('grpc/grpc') |
||||
|
||||
LABEL_LANG = set(label for label in repo.get_labels() |
||||
if label.name.split('/')[0] == 'lang') |
||||
LABEL_KIND_BUG = repo.get_label('kind/bug') |
||||
LABEL_PRIORITY_P0 = repo.get_label('priority/P0') |
||||
LABEL_PRIORITY_P1 = repo.get_label('priority/P1') |
||||
LABEL_PRIORITY_P2 = repo.get_label('priority/P2') |
||||
|
||||
def is_untriaged(issue): |
||||
key_labels = set() |
||||
for label in issue.labels: |
||||
label_kind = label.name.split('/')[0] |
||||
if label_kind in ('lang', 'kind', 'priority'): |
||||
key_labels.add(label_kind) |
||||
return len(key_labels) < 3 |
||||
|
||||
untriaged_open_issues = [ |
||||
issue for issue in repo.get_issues(state='open') |
||||
if issue.pull_request is None and is_untriaged(issue) |
||||
] |
||||
total_bugs = [ |
||||
issue |
||||
for issue in repo.get_issues(state='all', labels=[LABEL_KIND_BUG]) |
||||
if issue.pull_request is None |
||||
] |
||||
|
||||
lang_to_stats = {} |
||||
for lang in LABEL_LANG: |
||||
lang_bugs = filter(lambda bug: lang in bug.labels, total_bugs) |
||||
closed_bugs = filter(lambda bug: bug.state == 'closed', lang_bugs) |
||||
open_bugs = filter(lambda bug: bug.state == 'open', lang_bugs) |
||||
open_p0_bugs = filter(lambda bug: LABEL_PRIORITY_P0 in bug.labels, |
||||
open_bugs) |
||||
open_p1_bugs = filter(lambda bug: LABEL_PRIORITY_P1 in bug.labels, |
||||
open_bugs) |
||||
open_p2_bugs = filter(lambda bug: LABEL_PRIORITY_P2 in bug.labels, |
||||
open_bugs) |
||||
lang_to_stats[lang] = [ |
||||
len(lang_bugs), |
||||
len(closed_bugs), |
||||
len(open_bugs), |
||||
len(open_p0_bugs), |
||||
len(open_p1_bugs), |
||||
len(open_p2_bugs) |
||||
] |
||||
return len(untriaged_open_issues), lang_to_stats |
||||
|
||||
|
||||
def insert_stats_to_db(untriaged_open_issues, lang_to_stats): |
||||
timestamp = time() |
||||
client = bigquery.Client() |
||||
dataset_ref = client.dataset('github_issues') |
||||
table_ref = dataset_ref.table('untriaged_issues') |
||||
table = client.get_table(table_ref) |
||||
errors = client.insert_rows(table, [(timestamp, untriaged_open_issues)]) |
||||
table_ref = dataset_ref.table('bug_stats') |
||||
table = client.get_table(table_ref) |
||||
rows = [] |
||||
for lang, stats in lang_to_stats.iteritems(): |
||||
rows.append((timestamp, lang.name[5:]) + tuple(stats)) |
||||
errors = client.insert_rows(table, rows) |
||||
|
||||
|
||||
def fetch(): |
||||
untriaged_open_issues, lang_to_stats = get_stats_from_github() |
||||
insert_stats_to_db(untriaged_open_issues, lang_to_stats) |
@ -0,0 +1,29 @@ |
||||
# Copyright 2019 gRPC authors. |
||||
# |
||||
# Licensed under the Apache License, Version 2.0 (the "License"); |
||||
# you may not use this file except in compliance with the License. |
||||
# You may obtain a copy of the License at |
||||
# |
||||
# http://www.apache.org/licenses/LICENSE-2.0 |
||||
# |
||||
# Unless required by applicable law or agreed to in writing, software |
||||
# distributed under the License is distributed on an "AS IS" BASIS, |
||||
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
||||
# See the License for the specific language governing permissions and |
||||
# limitations under the License. |
||||
|
||||
import webapp2 |
||||
from fetch_data import fetch |
||||
|
||||
|
||||
class DailyCron(webapp2.RequestHandler): |
||||
|
||||
def get(self): |
||||
fetch() |
||||
self.response.status = 204 |
||||
|
||||
|
||||
app = webapp2.WSGIApplication( |
||||
[ |
||||
('/daily', DailyCron), |
||||
], debug=True) |
Loading…
Reference in new issue