awacke1 commited on
Commit
d5816db
Β·
verified Β·
1 Parent(s): 2616314

Create app.py

Browse files
Files changed (1) hide show
  1. app.py +394 -0
app.py ADDED
@@ -0,0 +1,394 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ import os
2
+ import streamlit as st
3
+ import requests
4
+ import msal
5
+
6
+ # πŸ€“ Load environment variables (Ensure these are set!)
7
+ APPLICATION_ID_KEY = os.getenv('APPLICATION_ID_KEY')
8
+ CLIENT_SECRET_KEY = os.getenv('CLIENT_SECRET_KEY')
9
+ AUTHORITY_URL = 'https://login.microsoftonline.com/common' # Use 'common' for multi-tenant apps
10
+ REDIRECT_URI = 'http://localhost:8501' # πŸ‘ˆ Make sure this matches your app's redirect URI
11
+
12
+ # 🎯 Define the scopes your app will need
13
+ SCOPES = ['User.Read', 'Calendars.ReadWrite', 'Mail.ReadWrite', 'Notes.ReadWrite.All', 'Files.ReadWrite.All']
14
+
15
+ # πŸ› οΈ Initialize the MSAL client
16
+ def get_msal_app():
17
+ return msal.ConfidentialClientApplication(
18
+ client_id=APPLICATION_ID_KEY,
19
+ client_credential=CLIENT_SECRET_KEY,
20
+ authority=AUTHORITY_URL
21
+ )
22
+
23
+ # πŸ” Acquire access token using authorization code
24
+ def get_access_token(code):
25
+ client_instance = get_msal_app()
26
+ result = client_instance.acquire_token_by_authorization_code(
27
+ code=code,
28
+ scopes=SCOPES,
29
+ redirect_uri=REDIRECT_URI
30
+ )
31
+ if 'access_token' in result:
32
+ return result['access_token']
33
+ else:
34
+ st.error('Error acquiring token: ' + str(result.get('error_description')))
35
+ st.stop()
36
+
37
+ # πŸƒβ€β™‚οΈ Main application function
38
+ def main():
39
+ st.title("πŸ¦„ Simple Streamlit App with MS Graph API")
40
+
41
+ # πŸš€ Sidebar navigation
42
+ st.sidebar.title("Navigation")
43
+ menu = st.sidebar.radio("Go to", [
44
+ "1️⃣ Dashboard with Widgets",
45
+ "🏠 Landing Page",
46
+ "πŸ“… Upcoming Events",
47
+ "πŸ“† Schedule",
48
+ "πŸ“ Agenda",
49
+ "πŸ” Event Details",
50
+ "βž• Add Event",
51
+ "πŸ”Ž Filter By"
52
+ ])
53
+
54
+ # πŸ“ M365 Products with MS Graph Integration & AI Features
55
+ st.sidebar.title("πŸ“ M365 Products")
56
+ st.sidebar.write("Toggle integration for each product:")
57
+
58
+ # πŸŽ›οΈ Product Integration Toggles
59
+ products = {
60
+ "πŸ“§ Outlook": {
61
+ "ai_capabilities": "Copilot for enhanced email writing, calendar management, and scheduling.",
62
+ "graph_api": "Access to mail, calendar, contacts, and events."
63
+ },
64
+ "πŸ“’ OneNote": {
65
+ "ai_capabilities": "Content suggestion, note organization, and OCR for extracting text from images.",
66
+ "graph_api": "Manage notebooks, sections, and pages."
67
+ },
68
+ "πŸ“Š Excel": {
69
+ "ai_capabilities": "Copilot for advanced data analysis, data insights, and formula generation.",
70
+ "graph_api": "Create and manage worksheets, tables, charts, and workbooks."
71
+ },
72
+ "πŸ“„ Word": {
73
+ "ai_capabilities": "Copilot for document drafting, summarization, and grammar improvements.",
74
+ "graph_api": "Document content access, templates, and file management."
75
+ },
76
+ "πŸ—ƒοΈ SharePoint": {
77
+ "ai_capabilities": "Intelligent search, document tagging, and metadata extraction.",
78
+ "graph_api": "Access to sites, lists, files, and document libraries."
79
+ },
80
+ "πŸ“… Teams": {
81
+ "ai_capabilities": "Copilot for meeting summaries, transcription, and chat suggestions.",
82
+ "graph_api": "Manage chats, teams, channels, and meetings."
83
+ },
84
+ "πŸ’¬ Viva": {
85
+ "ai_capabilities": "Personalized learning insights, well-being insights, and productivity suggestions.",
86
+ "graph_api": "Access to user analytics and learning modules."
87
+ },
88
+ "πŸš€ Power Platform": {
89
+ "ai_capabilities": "Automation with AI Builder, data insights, and custom AI models.",
90
+ "graph_api": "Automation workflows, app creation, and data visualization."
91
+ },
92
+ "🧠 Copilot": {
93
+ "ai_capabilities": "Embedded across Word, Excel, Outlook, Teams, and more for AI-driven productivity.",
94
+ "graph_api": "Underpins Copilot's access to data and integrations."
95
+ },
96
+ "πŸ—‚οΈ OneDrive": {
97
+ "ai_capabilities": "Intelligent file organization and search.",
98
+ "graph_api": "File and folder access, sharing, and metadata."
99
+ },
100
+ "πŸ’‘ PowerPoint": {
101
+ "ai_capabilities": "Design suggestions, presentation summarization, and speaker coaching.",
102
+ "graph_api": "Presentation creation, slide management, and templates."
103
+ },
104
+ "πŸ“š Microsoft Bookings": {
105
+ "ai_capabilities": "Automated scheduling and reminders.",
106
+ "graph_api": "Booking calendars, services, and appointment details."
107
+ },
108
+ "πŸ““ Loop": {
109
+ "ai_capabilities": "Real-time collaboration and content suggestions.",
110
+ "graph_api": "Access to shared workspaces and collaborative pages."
111
+ },
112
+ "πŸ—£οΈ Translator": {
113
+ "ai_capabilities": "Real-time language translation and text-to-speech.",
114
+ "graph_api": "Integrated into communication and translation services."
115
+ },
116
+ "πŸ“‹ To Do & Planner": {
117
+ "ai_capabilities": "Task prioritization and smart reminders.",
118
+ "graph_api": "Task creation, management, and synchronization."
119
+ },
120
+ "πŸ”— Azure OpenAI Service": {
121
+ "ai_capabilities": "Access to GPT models for custom AI implementations.",
122
+ "graph_api": "Used indirectly for building custom AI models into workflows."
123
+ }
124
+ }
125
+
126
+ # πŸ—³οΈ Create toggles for each product
127
+ selected_products = {}
128
+ for product, info in products.items():
129
+ selected = st.sidebar.checkbox(product)
130
+ if selected:
131
+ st.sidebar.write(f"**AI Capabilities:** {info['ai_capabilities']}")
132
+ st.sidebar.write(f"**Graph API:** {info['graph_api']}")
133
+ selected_products[product] = True
134
+
135
+ # πŸ”‘ Authentication
136
+ if 'access_token' not in st.session_state:
137
+ # πŸ•΅οΈβ€β™‚οΈ Check for authorization code in query parameters
138
+ query_params = st.experimental_get_query_params()
139
+ if 'code' in query_params:
140
+ code = query_params['code'][0]
141
+ st.write('πŸ”‘ Acquiring access token...')
142
+ access_token = get_access_token(code)
143
+ st.session_state['access_token'] = access_token
144
+ st.experimental_rerun() # Reload the app to clear the code from URL
145
+ else:
146
+ # πŸ“’ Prompt user to log in
147
+ client_instance = get_msal_app()
148
+ authorization_url = client_instance.get_authorization_request_url(
149
+ scopes=SCOPES,
150
+ redirect_uri=REDIRECT_URI
151
+ )
152
+ st.write('πŸ‘‹ Please [click here]({}) to log in and authorize the app.'.format(authorization_url))
153
+ st.stop()
154
+ else:
155
+ # πŸ₯³ User is authenticated, proceed with the app
156
+ access_token = st.session_state['access_token']
157
+ headers = {'Authorization': 'Bearer ' + access_token}
158
+
159
+ # πŸ€— Greet the user
160
+ response = requests.get('https://graph.microsoft.com/v1.0/me', headers=headers)
161
+ if response.status_code == 200:
162
+ user_info = response.json()
163
+ st.sidebar.write(f"πŸ‘‹ Hello, {user_info['displayName']}!")
164
+ else:
165
+ st.error('Failed to fetch user info.')
166
+ st.write(response.text)
167
+
168
+ # πŸŽ›οΈ Handle menu options
169
+ if menu == "1️⃣ Dashboard with Widgets":
170
+ st.header("1️⃣ Dashboard with Widgets")
171
+ st.write("Widgets will be displayed here. πŸŽ›οΈ")
172
+ # Add your widgets here
173
+
174
+ elif menu == "🏠 Landing Page":
175
+ st.header("🏠 Landing Page")
176
+ st.write("Welcome to the app! πŸ₯³")
177
+ # Add landing page content here
178
+
179
+ elif menu == "πŸ“… Upcoming Events":
180
+ st.header("πŸ“… Upcoming Events")
181
+ events = get_upcoming_events(access_token)
182
+ for event in events:
183
+ st.write(f"πŸ“† {event['subject']} on {event['start']['dateTime']}")
184
+ # Display upcoming events
185
+
186
+ elif menu == "πŸ“† Schedule":
187
+ st.header("πŸ“† Schedule")
188
+ schedule = get_schedule(access_token)
189
+ st.write(schedule)
190
+ # Display schedule
191
+
192
+ elif menu == "πŸ“ Agenda":
193
+ st.header("πŸ“ Agenda")
194
+ st.write("Your agenda for today. πŸ“‹")
195
+ # Display agenda
196
+
197
+ elif menu == "πŸ” Event Details":
198
+ st.header("πŸ” Event Details")
199
+ event_id = st.text_input("Enter Event ID")
200
+ if event_id:
201
+ event_details = get_event_details(access_token, event_id)
202
+ st.write(event_details)
203
+ # Display event details based on ID
204
+
205
+ elif menu == "βž• Add Event":
206
+ st.header("βž• Add Event")
207
+ event_subject = st.text_input("Event Subject")
208
+ event_start = st.date_input("Event Start Date")
209
+ event_start_time = st.time_input("Event Start Time")
210
+ event_end = st.date_input("Event End Date")
211
+ event_end_time = st.time_input("Event End Time")
212
+ if st.button("Add Event"):
213
+ event_details = {
214
+ "subject": event_subject,
215
+ "start": {
216
+ "dateTime": f"{event_start}T{event_start_time}",
217
+ "timeZone": "UTC"
218
+ },
219
+ "end": {
220
+ "dateTime": f"{event_end}T{event_end_time}",
221
+ "timeZone": "UTC"
222
+ }
223
+ }
224
+ add_event(access_token, event_details)
225
+ st.success("Event added successfully! πŸŽ‰")
226
+ # Form to add new event
227
+
228
+ elif menu == "πŸ”Ž Filter By":
229
+ st.header("πŸ”Ž Filter Events")
230
+ filter_criteria = st.text_input("Enter filter criteria")
231
+ if filter_criteria:
232
+ filtered_events = filter_events(access_token, filter_criteria)
233
+ for event in filtered_events:
234
+ st.write(f"πŸ“… {event['subject']} on {event['start']['dateTime']}")
235
+ # Filter events based on criteria
236
+
237
+ # 🧩 Handle selected products
238
+ if selected_products:
239
+ st.header("🧩 M365 Product Integrations")
240
+ for product in selected_products:
241
+ st.subheader(f"{product}")
242
+ # Call the function corresponding to the product
243
+ handle_product_integration(access_token, product)
244
+ else:
245
+ st.write("No products selected.")
246
+
247
+ # 🧩 Function to handle product integration
248
+ def handle_product_integration(access_token, product):
249
+ if product == "πŸ“§ Outlook":
250
+ st.write("Accessing Outlook data...")
251
+ # Implement Outlook integration
252
+ inbox_messages = get_outlook_messages(access_token)
253
+ st.write("Inbox Messages:")
254
+ for msg in inbox_messages:
255
+ st.write(f"βœ‰οΈ {msg['subject']}")
256
+ elif product == "πŸ“’ OneNote":
257
+ st.write("Accessing OneNote data...")
258
+ # Implement OneNote integration
259
+ notebooks = get_onenote_notebooks(access_token)
260
+ st.write("Notebooks:")
261
+ for notebook in notebooks:
262
+ st.write(f"πŸ“” {notebook['displayName']}")
263
+ elif product == "πŸ“Š Excel":
264
+ st.write("Accessing Excel data...")
265
+ # Implement Excel integration
266
+ st.write("Excel integration is a placeholder.")
267
+ elif product == "πŸ“„ Word":
268
+ st.write("Accessing Word documents...")
269
+ # Implement Word integration
270
+ st.write("Word integration is a placeholder.")
271
+ elif product == "πŸ—ƒοΈ SharePoint":
272
+ st.write("Accessing SharePoint sites...")
273
+ # Implement SharePoint integration
274
+ sites = get_sharepoint_sites(access_token)
275
+ st.write("Sites:")
276
+ for site in sites:
277
+ st.write(f"🌐 {site['displayName']}")
278
+ elif product == "πŸ“… Teams":
279
+ st.write("Accessing Teams data...")
280
+ # Implement Teams integration
281
+ teams = get_user_teams(access_token)
282
+ st.write("Teams:")
283
+ for team in teams:
284
+ st.write(f"πŸ‘₯ {team['displayName']}")
285
+ # Add additional product integrations as needed
286
+ else:
287
+ st.write(f"No integration implemented for {product}.")
288
+
289
+ # πŸ“¨ Function to get Outlook messages
290
+ def get_outlook_messages(access_token):
291
+ headers = {'Authorization': 'Bearer ' + access_token}
292
+ response = requests.get('https://graph.microsoft.com/v1.0/me/messages?$top=5', headers=headers)
293
+ if response.status_code == 200:
294
+ messages = response.json().get('value', [])
295
+ return messages
296
+ else:
297
+ st.error('Failed to fetch messages.')
298
+ st.write(response.text)
299
+ return []
300
+
301
+ # πŸ“” Function to get OneNote notebooks
302
+ def get_onenote_notebooks(access_token):
303
+ headers = {'Authorization': 'Bearer ' + access_token}
304
+ response = requests.get('https://graph.microsoft.com/v1.0/me/onenote/notebooks', headers=headers)
305
+ if response.status_code == 200:
306
+ notebooks = response.json().get('value', [])
307
+ return notebooks
308
+ else:
309
+ st.error('Failed to fetch notebooks.')
310
+ st.write(response.text)
311
+ return []
312
+
313
+ # 🌐 Function to get SharePoint sites
314
+ def get_sharepoint_sites(access_token):
315
+ headers = {'Authorization': 'Bearer ' + access_token}
316
+ response = requests.get('https://graph.microsoft.com/v1.0/sites?search=*', headers=headers)
317
+ if response.status_code == 200:
318
+ sites = response.json().get('value', [])
319
+ return sites
320
+ else:
321
+ st.error('Failed to fetch sites.')
322
+ st.write(response.text)
323
+ return []
324
+
325
+ # πŸ‘₯ Function to get user's Teams
326
+ def get_user_teams(access_token):
327
+ headers = {'Authorization': 'Bearer ' + access_token}
328
+ response = requests.get('https://graph.microsoft.com/v1.0/me/joinedTeams', headers=headers)
329
+ if response.status_code == 200:
330
+ teams = response.json().get('value', [])
331
+ return teams
332
+ else:
333
+ st.error('Failed to fetch teams.')
334
+ st.write(response.text)
335
+ return []
336
+
337
+ # πŸ“… Function to get upcoming events
338
+ def get_upcoming_events(access_token):
339
+ headers = {'Authorization': 'Bearer ' + access_token}
340
+ response = requests.get('https://graph.microsoft.com/v1.0/me/events?$orderby=start/dateTime&$top=10', headers=headers)
341
+ if response.status_code == 200:
342
+ events = response.json().get('value', [])
343
+ return events
344
+ else:
345
+ st.error('Failed to fetch upcoming events.')
346
+ st.write(response.text)
347
+ return []
348
+
349
+ # πŸ“† Function to get schedule (Placeholder)
350
+ def get_schedule(access_token):
351
+ # Implement API call to get schedule
352
+ return "πŸ“† Your schedule goes here."
353
+
354
+ # βž• Function to add a new event
355
+ def add_event(access_token, event_details):
356
+ headers = {
357
+ 'Authorization': 'Bearer ' + access_token,
358
+ 'Content-Type': 'application/json'
359
+ }
360
+ response = requests.post('https://graph.microsoft.com/v1.0/me/events', headers=headers, json=event_details)
361
+ if response.status_code == 201:
362
+ st.success('Event created successfully! πŸŽ‰')
363
+ else:
364
+ st.error('Failed to create event.')
365
+ st.write(response.text)
366
+
367
+ # πŸ” Function to get event details
368
+ def get_event_details(access_token, event_id):
369
+ headers = {'Authorization': 'Bearer ' + access_token}
370
+ response = requests.get(f'https://graph.microsoft.com/v1.0/me/events/{event_id}', headers=headers)
371
+ if response.status_code == 200:
372
+ event = response.json()
373
+ return event
374
+ else:
375
+ st.error('Failed to fetch event details.')
376
+ st.write(response.text)
377
+ return {}
378
+
379
+ # πŸ”Ž Function to filter events
380
+ def filter_events(access_token, filter_criteria):
381
+ headers = {'Authorization': 'Bearer ' + access_token}
382
+ # Implement filtering logic based on criteria
383
+ response = requests.get(f"https://graph.microsoft.com/v1.0/me/events?$filter=startswith(subject,'{filter_criteria}')", headers=headers)
384
+ if response.status_code == 200:
385
+ events = response.json().get('value', [])
386
+ return events
387
+ else:
388
+ st.error('Failed to filter events.')
389
+ st.write(response.text)
390
+ return []
391
+
392
+ # πŸš€ Run the main function
393
+ if __name__ == "__main__":
394
+ main()