James McCool commited on
Commit
044be09
·
1 Parent(s): 0935af1

INITIAL COMMIT and modernization

Browse files
.streamlit/secrets.toml ADDED
@@ -0,0 +1,2 @@
 
 
 
1
+ NBAGetGameData = 'https://docs.google.com/spreadsheets/d/1tRQrF_I5rS7Q0g9vE8NrENDZ2P3_DvtbBZzKEakwOI0/edit#gid=1373653837'
2
+ NBABettingModel = 'https://docs.google.com/spreadsheets/d/1WBnvOHQi_zVTGF63efejK5ho02AY00HiYrMHnMJXY1E/edit#gid=1157978351'
Dockerfile CHANGED
@@ -5,11 +5,24 @@ WORKDIR /app
5
  RUN apt-get update && apt-get install -y \
6
  build-essential \
7
  curl \
 
8
  git \
9
  && rm -rf /var/lib/apt/lists/*
10
 
11
  COPY requirements.txt ./
12
  COPY src/ ./src/
 
 
 
 
 
 
 
 
 
 
 
 
13
 
14
  RUN pip3 install -r requirements.txt
15
 
 
5
  RUN apt-get update && apt-get install -y \
6
  build-essential \
7
  curl \
8
+ software-properties-common \
9
  git \
10
  && rm -rf /var/lib/apt/lists/*
11
 
12
  COPY requirements.txt ./
13
  COPY src/ ./src/
14
+ COPY .streamlit/ ./.streamlit/
15
+
16
+ ENV MONGO_URI="mongodb+srv://multichem:Xr1q5wZdXPbxdUmJ@testcluster.lgwtp5i.mongodb.net/?retryWrites=true&w=majority&appName=TestCluster"
17
+ ENV NBAGetGameData="https://docs.google.com/spreadsheets/d/1tRQrF_I5rS7Q0g9vE8NrENDZ2P3_DvtbBZzKEakwOI0/edit#gid=1373653837"
18
+ ENV NBABettingModel="https://docs.google.com/spreadsheets/d/1WBnvOHQi_zVTGF63efejK5ho02AY00HiYrMHnMJXY1E/edit#gid=1157978351"
19
+ RUN useradd -m -u 1000 user
20
+ USER user
21
+ ENV HOME=/home/user\
22
+ PATH=/home/user/.local/bin:$PATH
23
+ WORKDIR $HOME/app
24
+ RUN pip install --no-cache-dir --upgrade pip
25
+ COPY --chown=user . $HOME/app
26
 
27
  RUN pip3 install -r requirements.txt
28
 
requirements.txt CHANGED
@@ -1,3 +1,8 @@
1
- altair
2
- pandas
3
- streamlit
 
 
 
 
 
 
1
+ streamlit
2
+ gspread
3
+ openpyxl
4
+ matplotlib
5
+ pulp
6
+ docker
7
+ plotly
8
+ scipy
src/database.py ADDED
@@ -0,0 +1,35 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import streamlit as st
2
+ import gspread
3
+
4
+ @st.cache_resource
5
+ def init_conn():
6
+ scope = ['https://spreadsheets.google.com/feeds', 'https://www.googleapis.com/auth/drive']
7
+
8
+ credentials = {
9
+ "type": "service_account",
10
+ "project_id": "dfsnew",
11
+ "private_key_id": "2432f6c3771f70a410c5c878d1359869fc9dddc8",
12
+ "private_key": "-----BEGIN PRIVATE KEY-----\nMIIEvQIBADANBgkqhkiG9w0BAQEFAASCBKcwggSjAgEAAoIBAQDBNBDU2aJuEr6n\ne0o7pDY8gjg1+g1e3oHlpyY/CHMByZuEwfXewsZYP/TApfr8zxXDNG9X31CloWXH\n6ef8H0h6TjhRppE/2YCUZlbgtvpwlDg+1aKTKY5Lc/L937I6V512mgMDhDmTwX+p\noV0vhPuJnyFy+Fuo+xu8D9A46lhTTIK4EZhHc04SUBxUI3pDdfvuMbjciD/Pskn2\nMwBSEG/FQoe4GYrSmm7jzYdSHItVBakr26xl117m8BrIuceU7IEWrnJGDza8TtTZ\n+4Wp7PY9v6DgVt2+rnnDaF/g7kocLqoj2xWp1eS7OALwmqaIPFljIUkL5AJJiLC1\n+/ve6iwVAgMBAAECggEADTFsPdCvwBL9HGw1nT2BK6AbzQnKfHI2zhMcMD04N0TI\nXygsjT3hM/kIElizOyy7+HS97rLz65+KFvzwx71uIlXxkBfO/txwJJIZeCZeky33\n6kiF3cU+b4YXL4FlRwkhGk55irWuhdm2iUOY3KwYziTE8LgncDJXij/NMPnFtshZ\n/2Dc/7sKLi1tna5tfXr5v4N7LhyFOfHme8ZSZIhnpV+WnFM/VAVghwi+3vfzeV+a\nVgvv+QwRUBF+MYpoW8aDw3Y1jKuKKxcG0qHR1mQQTDK6eAymy28lJ9LfgKkZBLS3\nVEGH8O+gLQj2l8VR8koRxA1FETJ9BnIiV4OF+uLQQQKBgQDyYkeBnpPKnw3MXKgy\nxtpt7hLdrrQiR69PHEvHj9z6b60KTH9jDMKcbCU/ouwbTtLQnvtwta2RoWD/1xk+\n3uaeQv/jOtgKGE+Sa0FvJuDWZwBfUORnyqb+s5G9MpVlqNLLkUmE5myyrDbFdxei\nwzisIjvQxtJDLB3pucTRyd6a1QKBgQDMDoWUfNpQI/up3r0RWVCl3odpwOMnpN0S\nhf8uLyvEvtbcMnpxCQCl+4KWnOiX4GH4N9sZGF8YTPazO2Kd85/GioUoNo5u6vJo\ncxD0BTvg5meyUjfZsmuU620/eVQBa88TRdo3isLmBqUp7SAC+g4vTHpgxn00dRYv\neSfZN0dsQQKBgQDkxR34mVOkyrqbSFj4k/dWCn6D/YDHWiF86ZgcowxO01jff5Q8\nSK7mNKxzg7KVk7Amd+eaWd+YtFh5IOwTCw9gEJy0O7Xs0UVJTTJVVryfoFgZnp/1\n1rAHdjT3/eZELTPILzjU1yeA/Eo11lHYramvzh/mzcFm5RzWnR/HYmFYgQKBgFOy\nbSX/pAgVCkedvc0c5lBymvZMkJ+VJrxPS+Ckpn43jKea6M/uUl7Cb8jZKSoKdgS6\n3FpJvc+Y2eOgKw4AfHuSG5Xn8roaEj23XK/KacoQl130DUZ0wV2+xvuvBz7h+ni8\nQQphFxoEhcBRq7ys1h6ebt+86mQW1ne4aRjWbKxBAoGARA+rBNIC9Z1vyRzMAXfj\nnQ9/wShd/NGpVRNrm7sdUastfoyK8Ip3HkJac3xE1ARpQTvxAz742mdeDxPWI8wZ\nHDsjIrRqGLKMN7tSIoM720y6PY/Tsg89SdY4y0h6M75rrEi4Lv5b7s4EmqAZdfKT\nbEyuT7sCPCLeOX/RLy/lCpA=\n-----END PRIVATE KEY-----\n",
13
+ "client_email": "dfsapps@dfsnew.iam.gserviceaccount.com",
14
+ "client_id": "105107448378741046480",
15
+ "auth_uri": "https://accounts.google.com/o/oauth2/auth",
16
+ "token_uri": "https://oauth2.googleapis.com/token",
17
+ "auth_provider_x509_cert_url": "https://www.googleapis.com/oauth2/v1/certs",
18
+ "client_x509_cert_url": "https://www.googleapis.com/robot/v1/metadata/x509/dfsapps%40dfsnew.iam.gserviceaccount.com",
19
+ "universe_domain": "googleapis.com"
20
+ }
21
+
22
+ header= {'User-Agent': 'Mozilla/5.0 (X11; Linux x86_64) '
23
+ 'AppleWebKit/537.11 (KHTML, like Gecko) '
24
+ 'Chrome/23.0.1271.64 Safari/537.11',
25
+ 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8',
26
+ 'Accept-Charset': 'ISO-8859-1,utf-8;q=0.7,*;q=0.3',
27
+ 'Accept-Encoding': 'none',
28
+ 'Accept-Language': 'en-US,en;q=0.8',
29
+ 'Connection': 'keep-alive'}
30
+
31
+ gc_con = gspread.service_account_from_dict(credentials, scope)
32
+
33
+ return gc_con
34
+
35
+ gcservice_account = init_conn()
src/streamlit_app.py CHANGED
@@ -1,40 +1,88 @@
1
- import altair as alt
 
 
2
  import numpy as np
3
  import pandas as pd
4
- import streamlit as st
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
5
 
6
- """
7
- # Welcome to Streamlit!
8
-
9
- Edit `/streamlit_app.py` to customize this app to your heart's desire :heart:.
10
- If you have any questions, checkout our [documentation](https://docs.streamlit.io) and [community
11
- forums](https://discuss.streamlit.io).
12
-
13
- In the meantime, below is an example of what you can do with just a few lines of code:
14
- """
15
-
16
- num_points = st.slider("Number of points in spiral", 1, 10000, 1100)
17
- num_turns = st.slider("Number of turns in spiral", 1, 300, 31)
18
-
19
- indices = np.linspace(0, 1, num_points)
20
- theta = 2 * np.pi * num_turns * indices
21
- radius = indices
22
-
23
- x = radius * np.cos(theta)
24
- y = radius * np.sin(theta)
25
-
26
- df = pd.DataFrame({
27
- "x": x,
28
- "y": y,
29
- "idx": indices,
30
- "rand": np.random.randn(num_points),
31
- })
32
-
33
- st.altair_chart(alt.Chart(df, height=700, width=700)
34
- .mark_point(filled=True)
35
- .encode(
36
- x=alt.X("x", axis=None),
37
- y=alt.Y("y", axis=None),
38
- color=alt.Color("idx", legend=None, scale=alt.Scale()),
39
- size=alt.Size("rand", legend=None, scale=alt.Scale(range=[1, 150])),
40
- ))
 
1
+ import streamlit as st
2
+ st.set_page_config(layout="wide")
3
+
4
  import numpy as np
5
  import pandas as pd
6
+ from database import gcservice_account, NBAGetGameData, NBABettingModel
7
+ import os
8
+
9
+ NBAGetGameData = os.getenv('NBAGetGameData')
10
+ NBABettingModel = os.getenv('NBABettingModel')
11
+
12
+ @st.cache_resource(ttl = 300)
13
+ def init_baselines():
14
+ sh = gcservice_account.open_by_url(NBAGetGameData)
15
+
16
+ worksheet = sh.worksheet('MinPublic')
17
+ raw_display = pd.DataFrame(worksheet.get_values())
18
+ raw_display.columns = raw_display.iloc[0]
19
+ raw_display = raw_display[1:]
20
+ raw_display = raw_display.reset_index(drop=True)
21
+ raw_display.replace('', np.nan, inplace=True)
22
+ raw_display = raw_display[['NBAID', 'PID', 'Player', 'TC', 'MP (Today)', 'Next Game', 'H/R', 'Injury Notes', 'Player Impact per 48', 'Player Impact',
23
+ 'Team PM', 'Last Updated']]
24
+ raw_display = raw_display.apply(pd.to_numeric, errors='coerce').fillna(raw_display)
25
+ public_minutes = raw_display[raw_display['NBAID'] != ""]
26
+
27
+ sh = gcservice_account.open_by_url(NBABettingModel)
28
+
29
+ worksheet = sh.worksheet('PlayerImpactByTeam')
30
+ raw_display = pd.DataFrame(worksheet.get_values())
31
+ raw_display.columns = raw_display.iloc[0]
32
+ raw_display = raw_display[1:]
33
+ raw_display = raw_display.reset_index(drop=True)
34
+ raw_display.replace('', 0, inplace=True)
35
+ raw_display = raw_display[['PID', 'Player', 'Team', 'Avg Minutes last 30 days for team', 'Minutes Projection', 'Rotation Impact (versus last 30 days)',
36
+ 'Injury Notes', 'Minute Change', 'Baseline Team PM', 'Net Rotation PM +/- for Team', 'Projected PM for Game', 'Offset', 'Rank']]
37
+ raw_display['Minute Change'].replace('+', '', inplace=True)
38
+ raw_display = raw_display.apply(pd.to_numeric, errors='coerce').fillna(raw_display)
39
+ player_impact = raw_display[raw_display['PID'] != ""]
40
+
41
+ return public_minutes, player_impact
42
+
43
+ def convert_df_to_csv(df):
44
+ return df.to_csv().encode('utf-8')
45
+
46
+ public_minutes, player_impact = init_baselines()
47
+
48
+ tab1, tab2 = st.tabs(["Minutes Baselines", "Player Impacts"])
49
+
50
+ with tab1:
51
+ if st.button("Reset Data", key='reset1'):
52
+ st.cache_data.clear()
53
+ public_minutes, player_impact = init_baselines()
54
+ split_var1 = st.radio("Would you like to view all teams or specific ones?", ('All', 'Specific Teams'), key='split_var1')
55
+ if split_var1 == 'Specific Teams':
56
+ team_var1 = st.multiselect('Which teams would you like to include in the tables?', options = public_minutes['TC'].unique(), key='team_var1')
57
+ elif split_var1 == 'All':
58
+ team_var1 = public_minutes.TC.values.tolist()
59
+ public_minutes = public_minutes[public_minutes['TC'].isin(team_var1)]
60
+ player_min_disp = public_minutes.set_index('Player')
61
+ player_min_disp = player_min_disp.sort_values(by=['TC', 'MP (Today)'], ascending=[True, False])
62
+ st.dataframe(player_min_disp.style.background_gradient(axis=0).background_gradient(cmap='RdYlGn').format(precision=2), use_container_width = True)
63
+ st.download_button(
64
+ label="Export Minutes Baselines",
65
+ data=convert_df_to_csv(public_minutes),
66
+ file_name='AmericanNumbers_Min_Baseline_export.csv',
67
+ mime='text/csv',
68
+ )
69
 
70
+ with tab2:
71
+ if st.button("Reset Data", key='reset2'):
72
+ st.cache_data.clear()
73
+ public_minutes, player_impact = init_baselines()
74
+ split_var2 = st.radio("Would you like to view all teams or specific ones?", ('All', 'Specific Teams'), key='split_var2')
75
+ if split_var2 == 'Specific Teams':
76
+ team_var2 = st.multiselect('Which teams would you like to include in the tables?', options = player_impact['Team'].unique(), key='team_var2')
77
+ elif split_var2 == 'All':
78
+ team_var2 = player_impact.Team.values.tolist()
79
+ player_impact = player_impact[player_impact['Team'].isin(team_var2)]
80
+ player_impact_disp = player_impact.set_index('Player')
81
+ player_impact_disp = player_impact_disp.sort_values(by=['Team', 'Rotation Impact (versus last 30 days)'], ascending=[True, False])
82
+ st.dataframe(player_impact_disp.style.background_gradient(axis=0).background_gradient(cmap='RdYlGn').format(precision=2), use_container_width = True)
83
+ st.download_button(
84
+ label="Export Player Impacts",
85
+ data=convert_df_to_csv(player_impact),
86
+ file_name='AmericanNumbers_Impact_export.csv',
87
+ mime='text/csv',
88
+ )