Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
E
edx-platform
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
edx
edx-platform
Commits
1d5fe0ed
Commit
1d5fe0ed
authored
May 26, 2016
by
wajeeha-khalid
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
MA-2419: make thread 'read' mutual exlusive to update fields
parent
a07a762f
Hide whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
48 additions
and
41 deletions
+48
-41
lms/djangoapps/discussion_api/api.py
+11
-0
lms/djangoapps/discussion_api/forms.py
+1
-0
lms/djangoapps/discussion_api/serializers.py
+1
-1
lms/djangoapps/discussion_api/tests/test_api.py
+0
-1
lms/djangoapps/discussion_api/tests/test_serializers.py
+0
-2
lms/djangoapps/discussion_api/tests/test_views.py
+3
-37
lms/djangoapps/discussion_api/tests/utils.py
+12
-0
lms/lib/comment_client/user.py
+20
-0
No files found.
lms/djangoapps/discussion_api/api.py
View file @
1d5fe0ed
...
@@ -514,6 +514,8 @@ def _do_extra_actions(api_content, cc_content, request_fields, actions_form, con
...
@@ -514,6 +514,8 @@ def _do_extra_actions(api_content, cc_content, request_fields, actions_form, con
_handle_abuse_flagged_field
(
form_value
,
context
[
"cc_requester"
],
cc_content
)
_handle_abuse_flagged_field
(
form_value
,
context
[
"cc_requester"
],
cc_content
)
elif
field
==
"voted"
:
elif
field
==
"voted"
:
_handle_voted_field
(
form_value
,
cc_content
,
api_content
,
request
,
context
)
_handle_voted_field
(
form_value
,
cc_content
,
api_content
,
request
,
context
)
elif
field
==
"read"
:
_handle_read_field
(
api_content
,
form_value
,
context
[
"cc_requester"
],
cc_content
)
else
:
else
:
raise
ValidationError
({
field
:
[
"Invalid Key"
]})
raise
ValidationError
({
field
:
[
"Invalid Key"
]})
...
@@ -549,6 +551,15 @@ def _handle_voted_field(form_value, cc_content, api_content, request, context):
...
@@ -549,6 +551,15 @@ def _handle_voted_field(form_value, cc_content, api_content, request, context):
)
)
def
_handle_read_field
(
api_content
,
form_value
,
user
,
cc_content
):
"""
Marks thread as read for the user
"""
if
form_value
and
not
cc_content
[
'read'
]:
user
.
read
(
cc_content
)
api_content
[
"unread_comment_count"
]
-=
1
def
create_thread
(
request
,
thread_data
):
def
create_thread
(
request
,
thread_data
):
"""
"""
Create a thread.
Create a thread.
...
...
lms/djangoapps/discussion_api/forms.py
View file @
1d5fe0ed
...
@@ -98,6 +98,7 @@ class ThreadActionsForm(Form):
...
@@ -98,6 +98,7 @@ class ThreadActionsForm(Form):
following
=
BooleanField
(
required
=
False
)
following
=
BooleanField
(
required
=
False
)
voted
=
BooleanField
(
required
=
False
)
voted
=
BooleanField
(
required
=
False
)
abuse_flagged
=
BooleanField
(
required
=
False
)
abuse_flagged
=
BooleanField
(
required
=
False
)
read
=
BooleanField
(
required
=
False
)
class
CommentListGetForm
(
_PaginationForm
):
class
CommentListGetForm
(
_PaginationForm
):
...
...
lms/djangoapps/discussion_api/serializers.py
View file @
1d5fe0ed
...
@@ -272,7 +272,7 @@ class ThreadSerializer(_ContentSerializer):
...
@@ -272,7 +272,7 @@ class ThreadSerializer(_ContentSerializer):
def
update
(
self
,
instance
,
validated_data
):
def
update
(
self
,
instance
,
validated_data
):
for
key
,
val
in
validated_data
.
items
():
for
key
,
val
in
validated_data
.
items
():
instance
[
key
]
=
val
instance
[
key
]
=
val
instance
.
save
(
params
=
{
"requested_user_id"
:
self
.
context
[
"cc_requester"
][
"id"
]}
)
instance
.
save
()
return
instance
return
instance
...
...
lms/djangoapps/discussion_api/tests/test_api.py
View file @
1d5fe0ed
...
@@ -1992,7 +1992,6 @@ class UpdateThreadTest(
...
@@ -1992,7 +1992,6 @@ class UpdateThreadTest(
"closed"
:
[
"False"
],
"closed"
:
[
"False"
],
"pinned"
:
[
"False"
],
"pinned"
:
[
"False"
],
"read"
:
[
"False"
],
"read"
:
[
"False"
],
"requested_user_id"
:
[
str
(
self
.
user
.
id
)],
}
}
)
)
...
...
lms/djangoapps/discussion_api/tests/test_serializers.py
View file @
1d5fe0ed
...
@@ -566,7 +566,6 @@ class ThreadSerializerDeserializationTest(CommentsServiceMockMixin, UrlResetMixi
...
@@ -566,7 +566,6 @@ class ThreadSerializerDeserializationTest(CommentsServiceMockMixin, UrlResetMixi
"pinned"
:
[
"False"
],
"pinned"
:
[
"False"
],
"user_id"
:
[
str
(
self
.
user
.
id
)],
"user_id"
:
[
str
(
self
.
user
.
id
)],
"read"
:
[
"False"
],
"read"
:
[
"False"
],
"requested_user_id"
:
[
str
(
self
.
user
.
id
)],
}
}
)
)
...
@@ -595,7 +594,6 @@ class ThreadSerializerDeserializationTest(CommentsServiceMockMixin, UrlResetMixi
...
@@ -595,7 +594,6 @@ class ThreadSerializerDeserializationTest(CommentsServiceMockMixin, UrlResetMixi
"pinned"
:
[
"False"
],
"pinned"
:
[
"False"
],
"user_id"
:
[
str
(
self
.
user
.
id
)],
"user_id"
:
[
str
(
self
.
user
.
id
)],
"read"
:
[
str
(
read
)],
"read"
:
[
str
(
read
)],
"requested_user_id"
:
[
str
(
self
.
user
.
id
)],
}
}
)
)
for
key
in
data
:
for
key
in
data
:
...
...
lms/djangoapps/discussion_api/tests/test_views.py
View file @
1d5fe0ed
...
@@ -730,7 +730,6 @@ class ThreadViewSetPartialUpdateTest(DiscussionAPIViewTestMixin, ModuleStoreTest
...
@@ -730,7 +730,6 @@ class ThreadViewSetPartialUpdateTest(DiscussionAPIViewTestMixin, ModuleStoreTest
"closed"
:
[
"False"
],
"closed"
:
[
"False"
],
"pinned"
:
[
"False"
],
"pinned"
:
[
"False"
],
"read"
:
[
"False"
],
"read"
:
[
"False"
],
"requested_user_id"
:
[
str
(
self
.
user
.
id
)],
}
}
)
)
...
@@ -787,7 +786,9 @@ class ThreadViewSetPartialUpdateTest(DiscussionAPIViewTestMixin, ModuleStoreTest
...
@@ -787,7 +786,9 @@ class ThreadViewSetPartialUpdateTest(DiscussionAPIViewTestMixin, ModuleStoreTest
def
test_patch_read_owner_user
(
self
):
def
test_patch_read_owner_user
(
self
):
self
.
register_get_user_response
(
self
.
user
)
self
.
register_get_user_response
(
self
.
user
)
self
.
register_thread
()
self
.
register_thread
()
self
.
register_read_response
(
self
.
user
,
"thread"
,
"test_thread"
)
request_data
=
{
"read"
:
True
}
request_data
=
{
"read"
:
True
}
response
=
self
.
request_patch
(
request_data
)
response
=
self
.
request_patch
(
request_data
)
self
.
assertEqual
(
response
.
status_code
,
200
)
self
.
assertEqual
(
response
.
status_code
,
200
)
response_data
=
json
.
loads
(
response
.
content
)
response_data
=
json
.
loads
(
response
.
content
)
...
@@ -802,30 +803,13 @@ class ThreadViewSetPartialUpdateTest(DiscussionAPIViewTestMixin, ModuleStoreTest
...
@@ -802,30 +803,13 @@ class ThreadViewSetPartialUpdateTest(DiscussionAPIViewTestMixin, ModuleStoreTest
})
})
)
)
self
.
assertEqual
(
httpretty
.
last_request
()
.
parsed_body
,
{
"course_id"
:
[
unicode
(
self
.
course
.
id
)],
"commentable_id"
:
[
"original_topic"
],
"thread_type"
:
[
"discussion"
],
"title"
:
[
"Original Title"
],
"body"
:
[
"Original body"
],
"user_id"
:
[
str
(
self
.
user
.
id
)],
"anonymous"
:
[
"False"
],
"anonymous_to_peers"
:
[
"False"
],
"closed"
:
[
"False"
],
"pinned"
:
[
"False"
],
"read"
:
[
"True"
],
"requested_user_id"
:
[
str
(
self
.
user
.
id
)],
}
)
def
test_patch_read_non_owner_user
(
self
):
def
test_patch_read_non_owner_user
(
self
):
self
.
register_get_user_response
(
self
.
user
)
self
.
register_get_user_response
(
self
.
user
)
thread_owner_user
=
UserFactory
.
create
(
password
=
self
.
password
)
thread_owner_user
=
UserFactory
.
create
(
password
=
self
.
password
)
CourseEnrollmentFactory
.
create
(
user
=
thread_owner_user
,
course_id
=
self
.
course
.
id
)
CourseEnrollmentFactory
.
create
(
user
=
thread_owner_user
,
course_id
=
self
.
course
.
id
)
self
.
register_get_user_response
(
thread_owner_user
)
self
.
register_get_user_response
(
thread_owner_user
)
self
.
register_thread
({
"username"
:
thread_owner_user
.
username
,
"user_id"
:
str
(
thread_owner_user
.
id
)})
self
.
register_thread
({
"username"
:
thread_owner_user
.
username
,
"user_id"
:
str
(
thread_owner_user
.
id
)})
self
.
register_read_response
(
self
.
user
,
"thread"
,
"test_thread"
)
request_data
=
{
"read"
:
True
}
request_data
=
{
"read"
:
True
}
response
=
self
.
request_patch
(
request_data
)
response
=
self
.
request_patch
(
request_data
)
...
@@ -843,24 +827,6 @@ class ThreadViewSetPartialUpdateTest(DiscussionAPIViewTestMixin, ModuleStoreTest
...
@@ -843,24 +827,6 @@ class ThreadViewSetPartialUpdateTest(DiscussionAPIViewTestMixin, ModuleStoreTest
})
})
)
)
self
.
assertEqual
(
httpretty
.
last_request
()
.
parsed_body
,
{
"course_id"
:
[
unicode
(
self
.
course
.
id
)],
"commentable_id"
:
[
"original_topic"
],
"thread_type"
:
[
"discussion"
],
"title"
:
[
"Original Title"
],
"body"
:
[
"Original body"
],
"user_id"
:
[
str
(
thread_owner_user
.
id
)],
"anonymous"
:
[
"False"
],
"anonymous_to_peers"
:
[
"False"
],
"closed"
:
[
"False"
],
"pinned"
:
[
"False"
],
"read"
:
[
"True"
],
"requested_user_id"
:
[
str
(
self
.
user
.
id
)],
}
)
@httpretty.activate
@httpretty.activate
@disable_signal
(
api
,
'thread_deleted'
)
@disable_signal
(
api
,
'thread_deleted'
)
...
...
lms/djangoapps/discussion_api/tests/utils.py
View file @
1d5fe0ed
...
@@ -260,6 +260,18 @@ class CommentsServiceMockMixin(object):
...
@@ -260,6 +260,18 @@ class CommentsServiceMockMixin(object):
status
=
200
status
=
200
)
)
def
register_read_response
(
self
,
user
,
content_type
,
content_id
):
"""
Register a mock response for POST on the CS 'read' endpoint
"""
httpretty
.
register_uri
(
httpretty
.
POST
,
"http://localhost:4567/api/v1/users/{id}/read"
.
format
(
id
=
user
.
id
),
params
=
{
'source_type'
:
content_type
,
'source_id'
:
content_id
},
body
=
json
.
dumps
({}),
# body is unused
status
=
200
)
def
register_thread_flag_response
(
self
,
thread_id
):
def
register_thread_flag_response
(
self
,
thread_id
):
"""Register a mock response for PUT on the CS thread flag endpoints"""
"""Register a mock response for PUT on the CS thread flag endpoints"""
self
.
register_flag_response
(
"thread"
,
thread_id
)
self
.
register_flag_response
(
"thread"
,
thread_id
)
...
...
lms/lib/comment_client/user.py
View file @
1d5fe0ed
...
@@ -30,6 +30,19 @@ class User(models.Model):
...
@@ -30,6 +30,19 @@ class User(models.Model):
external_id
=
str
(
user
.
id
),
external_id
=
str
(
user
.
id
),
username
=
user
.
username
)
username
=
user
.
username
)
def
read
(
self
,
source
):
"""
Calls cs_comments_service to mark thread as read for the user
"""
params
=
{
'source_type'
:
source
.
type
,
'source_id'
:
source
.
id
}
perform_request
(
'post'
,
_url_for_read
(
self
.
id
),
params
,
metric_action
=
'user.read'
,
metric_tags
=
self
.
_metric_tags
+
[
'target.type:{}'
.
format
(
source
.
type
)],
)
def
follow
(
self
,
source
):
def
follow
(
self
,
source
):
params
=
{
'source_type'
:
source
.
type
,
'source_id'
:
source
.
id
}
params
=
{
'source_type'
:
source
.
type
,
'source_id'
:
source
.
id
}
response
=
perform_request
(
response
=
perform_request
(
...
@@ -172,3 +185,10 @@ def _url_for_user_active_threads(user_id):
...
@@ -172,3 +185,10 @@ def _url_for_user_active_threads(user_id):
def
_url_for_user_subscribed_threads
(
user_id
):
def
_url_for_user_subscribed_threads
(
user_id
):
return
"{prefix}/users/{user_id}/subscribed_threads"
.
format
(
prefix
=
settings
.
PREFIX
,
user_id
=
user_id
)
return
"{prefix}/users/{user_id}/subscribed_threads"
.
format
(
prefix
=
settings
.
PREFIX
,
user_id
=
user_id
)
def
_url_for_read
(
user_id
):
"""
Returns cs_comments_service url endpoint to mark thread as read for given user_id
"""
return
"{prefix}/users/{user_id}/read"
.
format
(
prefix
=
settings
.
PREFIX
,
user_id
=
user_id
)
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment