Skip to content

Commit e17515d

Browse files
Escape xml special characters (#278) Release 2.2.2
1 parent 9468b41 commit e17515d

21 files changed

+64
-61
lines changed

examples/directory/clear_directory.py

-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
import adal
2-
32
from settings import settings
43

54
from office365.graph_client import GraphClient

examples/onedrive/export_files.py

-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@
22
import tempfile
33

44
import adal
5-
65
from settings import settings
76

87
from office365.graph_client import GraphClient

examples/onedrive/import_files.py

-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@
22
from os.path import isfile, join
33

44
import adal
5-
65
from settings import settings
76

87
from office365.graph_client import GraphClient

examples/onedrive/print_folders_and_files.py

-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
import adal
2-
32
from settings import settings
43

54
from office365.graph_client import GraphClient

examples/outlook/send_message.py

-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
import adal
2-
32
from settings import settings
43

54
from office365.graph_client import GraphClient

examples/sharepoint/connect_with_app.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,8 @@
33
from office365.runtime.auth.client_credential import ClientCredential
44
from office365.sharepoint.client_context import ClientContext
55

6-
credentials = ClientCredential(settings['client_credentials']['client_id'],
7-
settings['client_credentials']['client_secret'])
6+
credentials = ClientCredential(settings.get('client_credentials').get('client_id'),
7+
settings.get('client_credentials').get('client_secret'))
88
ctx = ClientContext(settings['url']).with_credentials(credentials)
99
if not ctx.authentication_context.acquire_token_func():
1010
print("Acquire token failed")

examples/sharepoint/connect_with_user_creds.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,8 @@
44
from office365.sharepoint.client_context import ClientContext
55

66
ctx = ClientContext(settings["url"]).with_credentials(
7-
UserCredential(settings['user_credentials']['username'],
8-
settings['user_credentials']['password']))
7+
UserCredential(settings.get('user_credentials').get('username'),
8+
settings.get('user_credentials').get('password')))
99

1010
web = ctx.web
1111
ctx.load(web)

examples/sharepoint/files/download_file_large.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,10 @@
11
import os
22
import tempfile
33

4-
from office365.runtime.http.request_options import RequestOptions
54
from settings import settings
65

76
from office365.runtime.auth.client_credential import ClientCredential
7+
from office365.runtime.http.request_options import RequestOptions
88
from office365.sharepoint.client_context import ClientContext
99

1010

@@ -40,7 +40,7 @@ def print_download_progress(offset):
4040
settings.get('client_credentials').get('client_secret'))
4141
ctx = ClientContext(site_url).with_credentials(credentials)
4242

43-
file_url = f'/sites/team/Shared Documents/big_buck_bunny.mp4'
43+
file_url = '/sites/team/Shared Documents/big_buck_bunny.mp4'
4444
local_file_name = os.path.join(tempfile.mkdtemp(), os.path.basename(file_url))
4545
with open(local_file_name, "wb") as local_file:
4646
download_file(ctx, file_url, local_file, print_download_progress)

examples/sharepoint/lists_and_items/read_attachments.py

+9-10
Original file line numberDiff line numberDiff line change
@@ -1,23 +1,22 @@
11
import os
22
import tempfile
33

4-
54
from settings import settings
65

76
from office365.runtime.auth.client_credential import ClientCredential
87
from office365.sharepoint.client_context import ClientContext
9-
from office365.sharepoint.listitems.caml.caml_query import CamlQuery
108

11-
download_path = tempfile.mkdtemp()
9+
creds = ClientCredential(settings.get('client_credentials').get('client_id'),
10+
settings.get('client_credentials').get('client_secret'))
11+
ctx = ClientContext(settings['url']).with_credentials(creds)
1212

13-
client_creds = ClientCredential(settings['client_credentials']['client_id'],
14-
settings['client_credentials']['client_secret'])
15-
ctx = ClientContext(settings['url']).with_credentials(client_creds)
13+
download_path = tempfile.mkdtemp()
1614

17-
list_obj = ctx.web.lists.get_by_title("Tasks123")
18-
#items = list_obj.get_items(CamlQuery.create_all_items_query())
19-
#items = list_obj.get_items()
20-
items = list_obj.items
15+
list_title = "Tasks"
16+
source_list = ctx.web.lists.get_by_title(list_title)
17+
# items = list_obj.get_items(CamlQuery.create_all_items_query())
18+
# items = list_obj.get_items()
19+
items = source_list.items
2120
ctx.load(items, ["ID", "UniqueId", "FileRef", "LinkFilename", "Title", "Attachments"])
2221
ctx.execute_query()
2322
for item in items:

examples/teams/connect_with_msal.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import msal
2-
32
from settings import settings
3+
44
from office365.graph_client import GraphClient
55

66

examples/teams/provision_team.py

-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
import uuid
22

33
import adal
4-
54
from settings import settings
65

76
from office365.directory.groupProfile import GroupProfile

examples/teams/send_message.py

+4-5
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
import msal
2-
3-
from office365.teams.itemBody import ItemBody
42
from settings import settings
3+
54
from office365.graph_client import GraphClient
65

76

@@ -29,6 +28,6 @@ def acquire_token_by_username_password():
2928
messages = target_team.primaryChannel.messages.get().execute_query()
3029
print(messages)
3130

32-
#item_body = ItemBody("Hello world!")
33-
#message = target_team.primaryChannel.messages.add(item_body).execute_query()
34-
#print(message.web_url)
31+
# item_body = ItemBody("Hello world!")
32+
# message = target_team.primaryChannel.messages.add(item_body).execute_query()
33+
# print(message.web_url)

office365/runtime/auth/authentication_context.py

+4-4
Original file line numberDiff line numberDiff line change
@@ -33,15 +33,15 @@ def acquire_token_for_user(self, username, password):
3333
:type password: str
3434
:type username: str
3535
"""
36-
self.provider = SamlTokenProvider(self.url, username, password)
37-
if not self.provider.acquire_token():
36+
self.provider = SamlTokenProvider(self.url)
37+
if not self.provider.acquire_token(username=username, password=password):
3838
raise ValueError('Acquire token failed: {0}'.format(self.provider.error))
3939
return True
4040

4141
def acquire_token_for_app(self, client_id, client_secret):
4242
"""Acquire token via client credentials (SharePoint App Principal)"""
43-
self.provider = ACSTokenProvider(self.url, client_id, client_secret)
44-
if not self.provider.acquire_token():
43+
self.provider = ACSTokenProvider(self.url)
44+
if not self.provider.acquire_token(client_id=client_id, client_secret=client_secret):
4545
raise ValueError('Acquire token failed: {0}'.format(self.provider.error))
4646
return True
4747

office365/runtime/auth/providers/acs_token_provider.py

+14-9
Original file line numberDiff line numberDiff line change
@@ -7,18 +7,14 @@
77

88
class ACSTokenProvider(BaseTokenProvider, office365.logger.LoggerContext):
99

10-
def __init__(self, url, client_id, client_secret):
10+
def __init__(self, url):
1111
"""
1212
Provider to acquire the access token from a Microsoft Azure Access Control Service (ACS)
1313
14-
:type client_secret: str
15-
:type client_id: str
1614
:type url: str
1715
"""
1816
self.token = TokenResponse()
1917
self.url = url
20-
self.client_id = client_id
21-
self.client_secret = client_secret
2218
self.redirect_url = None
2319
self.error = None
2420
self.SharePointPrincipal = "00000003-0000-0ff1-ce00-000000000000"
@@ -28,23 +24,32 @@ def is_authenticated(self):
2824

2925
def acquire_token(self, **kwargs):
3026
try:
27+
client_id = kwargs.get("client_id")
28+
client_secret = kwargs.get("client_secret")
3129
realm = self._get_realm_from_target_url()
3230
try:
3331
from urlparse import urlparse # Python 2.X
3432
except ImportError:
3533
from urllib.parse import urlparse # Python 3+
3634
url_info = urlparse(self.url)
37-
self.token = self.get_app_only_access_token(url_info.hostname, realm)
35+
self.token = self.get_app_only_access_token(client_id, client_secret, url_info.hostname, realm)
3836
return self.token.is_valid
3937
except requests.exceptions.RequestException as e:
4038
self.error = "Error: {}".format(e)
4139
return False
4240

43-
def get_app_only_access_token(self, target_host, target_realm):
41+
def get_app_only_access_token(self, client_id, client_secret, target_host, target_realm):
42+
"""
43+
44+
:type client_id: str
45+
:type client_secret: str
46+
:type target_host: str
47+
:type target_realm: str
48+
"""
4449
resource = self.get_formatted_principal(self.SharePointPrincipal, target_host, target_realm)
45-
principal_id = self.get_formatted_principal(self.client_id, None, target_realm)
50+
principal_id = self.get_formatted_principal(client_id, None, target_realm)
4651
sts_url = self.get_security_token_service_url(target_realm)
47-
oauth2_request = self.create_access_token_request(principal_id, self.client_secret, resource)
52+
oauth2_request = self.create_access_token_request(principal_id, client_secret, resource)
4853
response = requests.post(url=sts_url, headers={'Content-Type': 'application/x-www-form-urlencoded'},
4954
data=oauth2_request)
5055
return TokenResponse.from_json(response.json())

office365/runtime/auth/providers/saml_token_provider.py

+15-9
Original file line numberDiff line numberDiff line change
@@ -19,17 +19,21 @@ def resolve_base_url(url):
1919
return parts[0] + '://' + host_name
2020

2121

22+
def xml_escape(s_val):
23+
s_val = s_val.replace("&", "&")
24+
s_val = s_val.replace("<", "&lt;")
25+
s_val = s_val.replace(">", "&gt;")
26+
s_val = s_val.replace("\"", "&quot;")
27+
return s_val
28+
29+
2230
class SamlTokenProvider(BaseTokenProvider, office365.logger.LoggerContext):
2331

24-
def __init__(self, url, username, password):
32+
def __init__(self, url):
2533
"""SAML Security Token Service provider
2634
27-
:type password: str
28-
:type username: str
2935
:type url: str
3036
"""
31-
self.__username = username
32-
self.__password = password
3337
# Security Token Service info
3438
self.__sts_profile = STSProfile(resolve_base_url(url))
3539
# Last occurred error
@@ -62,11 +66,13 @@ def acquire_token(self, **kwargs):
6266

6367
try:
6468
logger.debug("Acquiring Access Token..")
65-
user_realm = self._get_user_realm(self.__username)
69+
username = kwargs.get("username")
70+
password = xml_escape(kwargs.get("password"))
71+
user_realm = self._get_user_realm(username)
6672
if user_realm.IsFederated:
67-
token = self.acquire_service_token_from_adfs(user_realm.STSAuthUrl, self.__username, self.__password)
73+
token = self.acquire_service_token_from_adfs(user_realm.STSAuthUrl, username, password)
6874
else:
69-
token = self._acquire_service_token(self.__username, self.__password)
75+
token = self._acquire_service_token(username, password)
7076
return self._acquire_authentication_cookie(token, user_realm.IsFederated)
7177
except requests.exceptions.RequestException as e:
7278
self.error = "Error: {}".format(e)
@@ -118,7 +124,7 @@ def acquire_service_token_from_adfs(self, adfs_url, username, password):
118124
'{0}Body/{1}RequestSecurityTokenResponse/{1}RequestedSecurityToken/{2}Assertion'.format(
119125
self.__ns_prefixes['s'], self.__ns_prefixes['wst'], self.__ns_prefixes['saml']))
120126
if assertion_node is None:
121-
self.error = 'Cannot get security assertion for user {0} from {1}'.format(self.__username, adfs_url)
127+
self.error = 'Cannot get security assertion for user {0} from {1}'.format(username, adfs_url)
122128
logger.error(self.error)
123129
return None
124130
# 2. prepare & submit token request

office365/sharepoint/userprofiles/peopleManager.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,9 @@
22
from office365.runtime.queries.service_operation_query import ServiceOperationQuery
33
from office365.runtime.resource_path import ResourcePath
44
from office365.sharepoint.base_entity import BaseEntity
5+
from office365.sharepoint.userprofiles.personalSiteCreationPriority import PersonalSiteCreationPriority
56
from office365.sharepoint.userprofiles.personProperties import PersonProperties
67
from office365.sharepoint.userprofiles.personPropertiesCollection import PersonPropertiesCollection
7-
from office365.sharepoint.userprofiles.personalSiteCreationPriority import PersonalSiteCreationPriority
88

99

1010
class PeopleManager(BaseEntity):

requirements.txt

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
requests~=2.24.0
22
requests_ntlm [NTLMAuthentication]
33
setuptools~=50.3.2
4-
msal~=1.5.1
4+
msal~=1.6.0
55

setup.py

+4-3
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010

1111
setup(
1212
name="Office365-REST-Python-Client",
13-
version="2.2.1.1",
13+
version="2.2.2",
1414
author="Vadim Gremyachev",
1515
author_email="[email protected]",
1616
maintainer="Konrad Gądek",
@@ -19,7 +19,7 @@
1919
long_description=long_description,
2020
long_description_content_type="text/markdown",
2121
url="https://github.com/vgrem/Office365-REST-Python-Client",
22-
install_requires=['requests', 'adal'],
22+
install_requires=['requests', 'msal'],
2323
extras_require={
2424
'NTLMAuthentication': ["requests_ntlm"]
2525
},
@@ -44,6 +44,7 @@
4444
],
4545
packages=setuptools.find_packages(),
4646
package_data={
47-
'office365': ["runtime/auth/providers/templates/SAML.xml", "runtime/auth/providers/templates/RST2.xml", "runtime/auth/providers/templates/FederatedSAML.xml"]
47+
'office365': ["runtime/auth/providers/templates/SAML.xml", "runtime/auth/providers/templates/RST2.xml",
48+
"runtime/auth/providers/templates/FederatedSAML.xml"]
4849
}
4950
)

tests/sharepoint/test_folder.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,8 @@ def test1_enum_folders_and_files(self):
4646
self.assertIsNotNone(file_in_folder.resource_path)
4747

4848
def test2_create_folder(self):
49-
folder_new = self.__class__.target_list.rootFolder.folders.add(self.__class__.target_folder_name).execute_query()
49+
folder_new = self.__class__.target_list.rootFolder.folders.add(self.__class__.target_folder_name)\
50+
.execute_query()
5051
self.assertTrue(folder_new.properties["Exists"])
5152
self.__class__.target_folder = folder_new
5253

tests/sharepoint/test_taxonomy.py

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
1-
import uuid
1+
from tests.sharepoint.sharepoint_case import SPTestCase
22

33
from office365.sharepoint.fields.field import Field
4-
from tests.sharepoint.sharepoint_case import SPTestCase
54

65

76
class TestSPTaxonomy(SPTestCase):

tests/sharepoint/test_user_profile.py

+2-3
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,10 @@
11
from unittest import TestCase
22

3+
from settings import settings
4+
35
from office365.runtime.auth.user_credential import UserCredential
46
from office365.sharepoint.client_context import ClientContext
57
from office365.sharepoint.userprofiles.peopleManager import PeopleManager
6-
from settings import settings
7-
8-
98
from office365.sharepoint.userprofiles.profileLoader import ProfileLoader
109

1110

0 commit comments

Comments
 (0)