Ex-74 Registration Code redemption fix the translation issues added a check if a user is already registered in a course. Changed the messages added course depth=0 and removed pep8 violations Ex-72-added additional billing information Added a new CSV file in the instructor dashboard sales tab to download all the order sales separated from the invoice sales fix path to image updated the failed unit tests and add some minor tweaks in the Shoppingcart.scss Ex-78 New UI In receipt page EX-73 Purchasers can buy a course on behalf of a different student WL-78 updated the receipt page UI. Wl-72 updated Billing Information UI and removed the Order Address fields WL-71 Remove Purchase Type Buttons from UI WL-71 Updated minor UI issues and updated test cases WL-78 updated the enrollment links in the receipt page made changes in Order generated sales csv in Instructor Dashboard. The total_registration_codes and total_used_codes were not correctly stored in the csv file. 1) The total_registration_codes were not filtered with course_id. 2) The total_used_codes that a user had redeemed were not correctly included in the CSV. added a fix in the courseware view to let the users visit the courseware if they have enrolled in the course by clicking on the enrollment link rebase and resolved conflicts with master WL-97 Bulk Registration Email Confirmation Below is the commit summary. - Make email text bold as per requirement. - Improve email template quality and reorder points. - Add text in billing details page : "if no additional billing details are populated the payment confirmation will be sent to the user making the purchase" - Update text on receipt page "You have successfully purchase 3 course registration codes" WL-100 fixed the bug on the edit/add coupon and set course price. Ajax requests were duplicating in each callback. fixed this issue by creating the manual ajax request rather than the Lean Modal Ajax requests allow for better White Label branding in shopping cart purchase emails fix up typos and text fix goof fix fix incorporated model changes as suggested by Jason. updated order sales csv updated test cases for CourseRegCodeItem model and csv for the order generated sales updated the migrations history fixed the lms acceptance tests Be sure to check for multiple types address PR feedback PR feedback PR feedback pep8 fix
709 lines
28 KiB
Python
709 lines
28 KiB
Python
import logging
|
|
import datetime
|
|
import decimal
|
|
import pytz
|
|
from django.conf import settings
|
|
from django.contrib.auth.models import Group
|
|
from django.http import (
|
|
HttpResponse, HttpResponseRedirect, HttpResponseNotFound,
|
|
HttpResponseBadRequest, HttpResponseForbidden, Http404
|
|
)
|
|
from django.utils.translation import ugettext as _
|
|
from util.json_request import JsonResponse
|
|
from django.views.decorators.http import require_POST, require_http_methods
|
|
from django.core.urlresolvers import reverse
|
|
from django.views.decorators.csrf import csrf_exempt
|
|
from util.bad_request_rate_limiter import BadRequestRateLimiter
|
|
from django.contrib.auth.decorators import login_required
|
|
from microsite_configuration import microsite
|
|
from edxmako.shortcuts import render_to_response
|
|
from opaque_keys.edx.locations import SlashSeparatedCourseKey
|
|
from opaque_keys.edx.locator import CourseLocator
|
|
from opaque_keys import InvalidKeyError
|
|
from courseware.courses import get_course_by_id
|
|
from courseware.views import registered_for_course
|
|
from config_models.decorators import require_config
|
|
from shoppingcart.reports import RefundReport, ItemizedPurchaseReport, UniversityRevenueShareReport, CertificateStatusReport
|
|
from student.models import CourseEnrollment
|
|
from .exceptions import (
|
|
ItemAlreadyInCartException, AlreadyEnrolledInCourseException,
|
|
CourseDoesNotExistException, ReportTypeDoesNotExistException,
|
|
RegCodeAlreadyExistException, ItemDoesNotExistAgainstRegCodeException,
|
|
MultipleCouponsNotAllowedException, InvalidCartItem
|
|
)
|
|
from .models import (
|
|
Order, OrderTypes,
|
|
PaidCourseRegistration, OrderItem, Coupon, CourseRegCodeItem,
|
|
CouponRedemption, CourseRegistrationCode, RegistrationCodeRedemption,
|
|
Donation, DonationConfiguration
|
|
)
|
|
from .processors import (
|
|
process_postpay_callback, render_purchase_form_html,
|
|
get_signed_purchase_params, get_purchase_endpoint
|
|
)
|
|
|
|
import json
|
|
from xmodule_django.models import CourseKeyField
|
|
|
|
log = logging.getLogger("shoppingcart")
|
|
AUDIT_LOG = logging.getLogger("audit")
|
|
|
|
EVENT_NAME_USER_UPGRADED = 'edx.course.enrollment.upgrade.succeeded'
|
|
|
|
REPORT_TYPES = [
|
|
("refund_report", RefundReport),
|
|
("itemized_purchase_report", ItemizedPurchaseReport),
|
|
("university_revenue_share", UniversityRevenueShareReport),
|
|
("certificate_status", CertificateStatusReport),
|
|
]
|
|
|
|
|
|
def initialize_report(report_type, start_date, end_date, start_letter=None, end_letter=None):
|
|
"""
|
|
Creates the appropriate type of Report object based on the string report_type.
|
|
"""
|
|
for item in REPORT_TYPES:
|
|
if report_type in item:
|
|
return item[1](start_date, end_date, start_letter, end_letter)
|
|
raise ReportTypeDoesNotExistException
|
|
|
|
|
|
@require_POST
|
|
def add_course_to_cart(request, course_id):
|
|
"""
|
|
Adds course specified by course_id to the cart. The model function add_to_order does all the
|
|
heavy lifting (logging, error checking, etc)
|
|
"""
|
|
|
|
assert isinstance(course_id, basestring)
|
|
if not request.user.is_authenticated():
|
|
log.info("Anon user trying to add course {} to cart".format(course_id))
|
|
return HttpResponseForbidden(_('You must be logged-in to add to a shopping cart'))
|
|
cart = Order.get_cart_for_user(request.user)
|
|
course_key = SlashSeparatedCourseKey.from_deprecated_string(course_id)
|
|
# All logging from here handled by the model
|
|
try:
|
|
PaidCourseRegistration.add_to_order(cart, course_key)
|
|
except CourseDoesNotExistException:
|
|
return HttpResponseNotFound(_('The course you requested does not exist.'))
|
|
except ItemAlreadyInCartException:
|
|
return HttpResponseBadRequest(_('The course {0} is already in your cart.'.format(course_id)))
|
|
except AlreadyEnrolledInCourseException:
|
|
return HttpResponseBadRequest(_('You are already registered in course {0}.'.format(course_id)))
|
|
return HttpResponse(_("Course added to cart."))
|
|
|
|
|
|
@login_required
|
|
def update_user_cart(request):
|
|
"""
|
|
when user change the number-of-students from the UI then
|
|
this method Update the corresponding qty field in OrderItem model and update the order_type in order model.
|
|
"""
|
|
try:
|
|
qty = int(request.POST.get('qty', -1))
|
|
except ValueError:
|
|
log.exception('Quantity must be an integer.')
|
|
return HttpResponseBadRequest('Quantity must be an integer.')
|
|
|
|
if not 1 <= qty <= 1000:
|
|
log.warning('Quantity must be between 1 and 1000.')
|
|
return HttpResponseBadRequest('Quantity must be between 1 and 1000.')
|
|
|
|
item_id = request.POST.get('ItemId', None)
|
|
if item_id:
|
|
try:
|
|
item = OrderItem.objects.get(id=item_id, status='cart')
|
|
except OrderItem.DoesNotExist:
|
|
log.exception('Cart OrderItem id={0} DoesNotExist'.format(item_id))
|
|
return HttpResponseNotFound('Order item does not exist.')
|
|
|
|
item.qty = qty
|
|
item.save()
|
|
item.order.update_order_type()
|
|
total_cost = item.order.total_cost
|
|
return JsonResponse({"total_cost": total_cost}, 200)
|
|
|
|
return HttpResponseBadRequest('Order item not found in request.')
|
|
|
|
|
|
@login_required
|
|
def show_cart(request):
|
|
"""
|
|
This view shows cart items.
|
|
"""
|
|
cart = Order.get_cart_for_user(request.user)
|
|
total_cost = cart.total_cost
|
|
cart_items = cart.orderitem_set.all().select_subclasses()
|
|
shoppingcart_items = []
|
|
for cart_item in cart_items:
|
|
course_key = getattr(cart_item, 'course_id')
|
|
if course_key:
|
|
course = get_course_by_id(course_key, depth=0)
|
|
shoppingcart_items.append((cart_item, course))
|
|
|
|
site_name = microsite.get_value('SITE_NAME', settings.SITE_NAME)
|
|
|
|
callback_url = request.build_absolute_uri(
|
|
reverse("shoppingcart.views.postpay_callback")
|
|
)
|
|
form_html = render_purchase_form_html(cart, callback_url=callback_url)
|
|
context = {
|
|
'order': cart,
|
|
'shoppingcart_items': shoppingcart_items,
|
|
'amount': total_cost,
|
|
'site_name': site_name,
|
|
'form_html': form_html,
|
|
}
|
|
return render_to_response("shoppingcart/shopping_cart.html", context)
|
|
|
|
|
|
@login_required
|
|
def clear_cart(request):
|
|
cart = Order.get_cart_for_user(request.user)
|
|
cart.clear()
|
|
coupon_redemption = CouponRedemption.objects.filter(user=request.user, order=cart.id)
|
|
if coupon_redemption:
|
|
coupon_redemption.delete()
|
|
log.info('Coupon redemption entry removed for user {0} for order {1}'.format(request.user, cart.id))
|
|
|
|
reg_code_redemption = RegistrationCodeRedemption.objects.filter(redeemed_by=request.user, order=cart.id)
|
|
if reg_code_redemption:
|
|
reg_code_redemption.delete()
|
|
log.info('Registration code redemption entry removed for user {0} for order {1}'.format(request.user, cart.id))
|
|
|
|
return HttpResponse('Cleared')
|
|
|
|
|
|
@login_required
|
|
def remove_item(request):
|
|
"""
|
|
This will remove an item from the user cart and also delete the corresponding coupon codes redemption.
|
|
"""
|
|
item_id = request.REQUEST.get('id', '-1')
|
|
|
|
items = OrderItem.objects.filter(id=item_id, status='cart').select_subclasses()
|
|
|
|
if not len(items):
|
|
log.exception('Cannot remove cart OrderItem id={0}. DoesNotExist or item is already purchased'.format(item_id))
|
|
else:
|
|
item = items[0]
|
|
if item.user == request.user:
|
|
order_item_course_id = getattr(item, 'course_id')
|
|
item.delete()
|
|
log.info('order item {0} removed for user {1}'.format(item_id, request.user))
|
|
remove_code_redemption(order_item_course_id, item_id, item, request.user)
|
|
item.order.update_order_type()
|
|
|
|
return HttpResponse('OK')
|
|
|
|
def remove_code_redemption(order_item_course_id, item_id, item, user):
|
|
"""
|
|
If an item removed from shopping cart then we will remove
|
|
the corresponding redemption info of coupon/registration code.
|
|
"""
|
|
try:
|
|
# Try to remove redemption information of coupon code, If exist.
|
|
coupon_redemption = CouponRedemption.objects.get(
|
|
user=user,
|
|
coupon__course_id=order_item_course_id if order_item_course_id else CourseKeyField.Empty,
|
|
order=item.order_id
|
|
)
|
|
coupon_redemption.delete()
|
|
log.info('Coupon "{0}" redemption entry removed for user "{1}" for order item "{2}"'
|
|
.format(coupon_redemption.coupon.code, user, item_id))
|
|
except CouponRedemption.DoesNotExist:
|
|
pass
|
|
|
|
try:
|
|
# Try to remove redemption information of registration code, If exist.
|
|
reg_code_redemption = RegistrationCodeRedemption.objects.get(redeemed_by=user, order=item.order_id)
|
|
except RegistrationCodeRedemption.DoesNotExist:
|
|
log.debug('Code redemption does not exist for order item id={0}.'.format(item_id))
|
|
else:
|
|
if order_item_course_id == reg_code_redemption.registration_code.course_id:
|
|
reg_code_redemption.delete()
|
|
log.info('Registration code "{0}" redemption entry removed for user "{1}" for order item "{2}"'
|
|
.format(reg_code_redemption.registration_code.code, user, item_id))
|
|
|
|
|
|
@login_required
|
|
def reset_code_redemption(request):
|
|
"""
|
|
This method reset the code redemption from user cart items.
|
|
"""
|
|
cart = Order.get_cart_for_user(request.user)
|
|
cart.reset_cart_items_prices()
|
|
CouponRedemption.delete_coupon_redemption(request.user, cart)
|
|
RegistrationCodeRedemption.delete_registration_redemption(request.user, cart)
|
|
return HttpResponse('reset')
|
|
|
|
|
|
@login_required
|
|
def use_code(request):
|
|
"""
|
|
This method may generate the discount against valid coupon code
|
|
and save its entry into coupon redemption table
|
|
OR
|
|
Make the cart item free of cost against valid registration code.
|
|
Valid Code can be either coupon or registration code.
|
|
"""
|
|
code = request.POST["code"]
|
|
coupons = Coupon.objects.filter(code=code, is_active=True)
|
|
if not coupons:
|
|
# If not coupon code then we check that code against course registration code
|
|
try:
|
|
course_reg = CourseRegistrationCode.objects.get(code=code)
|
|
except CourseRegistrationCode.DoesNotExist:
|
|
return HttpResponseNotFound(_("Discount does not exist against code '{0}'.".format(code)))
|
|
|
|
return use_registration_code(course_reg, request.user)
|
|
|
|
return use_coupon_code(coupons, request.user)
|
|
|
|
|
|
def get_reg_code_validity(registration_code, request, limiter):
|
|
"""
|
|
This function checks if the registration code is valid, and then checks if it was already redeemed.
|
|
"""
|
|
reg_code_already_redeemed = False
|
|
course_registration = None
|
|
try:
|
|
course_registration = CourseRegistrationCode.objects.get(code=registration_code)
|
|
except CourseRegistrationCode.DoesNotExist:
|
|
reg_code_is_valid = False
|
|
else:
|
|
reg_code_is_valid = True
|
|
try:
|
|
RegistrationCodeRedemption.objects.get(registration_code__code=registration_code)
|
|
except RegistrationCodeRedemption.DoesNotExist:
|
|
reg_code_already_redeemed = False
|
|
else:
|
|
reg_code_already_redeemed = True
|
|
|
|
if not reg_code_is_valid:
|
|
#tick the rate limiter counter
|
|
AUDIT_LOG.info("Redemption of a non existing RegistrationCode {code}".format(code=registration_code))
|
|
limiter.tick_bad_request_counter(request)
|
|
raise Http404()
|
|
|
|
return reg_code_is_valid, reg_code_already_redeemed, course_registration
|
|
|
|
|
|
@require_http_methods(["GET", "POST"])
|
|
@login_required
|
|
def register_code_redemption(request, registration_code):
|
|
"""
|
|
This view allows the student to redeem the registration code
|
|
and enroll in the course.
|
|
"""
|
|
|
|
# Add some rate limiting here by re-using the RateLimitMixin as a helper class
|
|
site_name = microsite.get_value('SITE_NAME', settings.SITE_NAME)
|
|
limiter = BadRequestRateLimiter()
|
|
if limiter.is_rate_limit_exceeded(request):
|
|
AUDIT_LOG.warning("Rate limit exceeded in registration code redemption.")
|
|
return HttpResponseForbidden()
|
|
|
|
template_to_render = 'shoppingcart/registration_code_receipt.html'
|
|
if request.method == "GET":
|
|
reg_code_is_valid, reg_code_already_redeemed, course_registration = get_reg_code_validity(registration_code,
|
|
request, limiter)
|
|
course = get_course_by_id(getattr(course_registration, 'course_id'), depth=0)
|
|
context = {
|
|
'reg_code_already_redeemed': reg_code_already_redeemed,
|
|
'reg_code_is_valid': reg_code_is_valid,
|
|
'reg_code': registration_code,
|
|
'site_name': site_name,
|
|
'course': course,
|
|
'registered_for_course': registered_for_course(course, request.user)
|
|
}
|
|
return render_to_response(template_to_render, context)
|
|
elif request.method == "POST":
|
|
reg_code_is_valid, reg_code_already_redeemed, course_registration = get_reg_code_validity(registration_code,
|
|
request, limiter)
|
|
|
|
course = get_course_by_id(getattr(course_registration, 'course_id'), depth=0)
|
|
if reg_code_is_valid and not reg_code_already_redeemed:
|
|
#now redeem the reg code.
|
|
RegistrationCodeRedemption.create_invoice_generated_registration_redemption(course_registration, request.user)
|
|
CourseEnrollment.enroll(request.user, course.id)
|
|
context = {
|
|
'redemption_success': True,
|
|
'reg_code': registration_code,
|
|
'site_name': site_name,
|
|
'course': course,
|
|
}
|
|
else:
|
|
context = {
|
|
'reg_code_is_valid': reg_code_is_valid,
|
|
'reg_code_already_redeemed': reg_code_already_redeemed,
|
|
'redemption_success': False,
|
|
'reg_code': registration_code,
|
|
'site_name': site_name,
|
|
'course': course,
|
|
}
|
|
return render_to_response(template_to_render, context)
|
|
|
|
|
|
def use_registration_code(course_reg, user):
|
|
"""
|
|
This method utilize course registration code
|
|
"""
|
|
try:
|
|
cart = Order.get_cart_for_user(user)
|
|
RegistrationCodeRedemption.add_reg_code_redemption(course_reg, cart)
|
|
except RegCodeAlreadyExistException:
|
|
return HttpResponseBadRequest(_("Oops! The code '{0}' you entered is either invalid or expired".format(course_reg.code)))
|
|
except ItemDoesNotExistAgainstRegCodeException:
|
|
return HttpResponseNotFound(_("Code '{0}' is not valid for any course in the shopping cart.".format(course_reg.code)))
|
|
|
|
return HttpResponse(json.dumps({'response': 'success'}), content_type="application/json")
|
|
|
|
|
|
def use_coupon_code(coupons, user):
|
|
"""
|
|
This method utilize course coupon code
|
|
"""
|
|
cart = Order.get_cart_for_user(user)
|
|
cart_items = cart.orderitem_set.all().select_subclasses()
|
|
is_redemption_applied = False
|
|
for coupon in coupons:
|
|
try:
|
|
if CouponRedemption.add_coupon_redemption(coupon, cart, cart_items):
|
|
is_redemption_applied = True
|
|
except MultipleCouponsNotAllowedException:
|
|
return HttpResponseBadRequest(_("Only one coupon redemption is allowed against an order"))
|
|
|
|
if not is_redemption_applied:
|
|
log.warning("Course item does not exist for coupon '{0}'".format(coupons[0].code))
|
|
return HttpResponseNotFound(_("Coupon '{0}' is not valid for any course in the shopping cart.".format(coupons[0].code)))
|
|
|
|
return HttpResponse(json.dumps({'response': 'success'}), content_type="application/json")
|
|
|
|
|
|
@login_required
|
|
def register_courses(request):
|
|
"""
|
|
This method enroll the user for available course(s)
|
|
in cart on which valid registration code is applied
|
|
"""
|
|
cart = Order.get_cart_for_user(request.user)
|
|
CourseRegistrationCode.free_user_enrollment(cart)
|
|
return HttpResponse(json.dumps({'response': 'success'}), content_type="application/json")
|
|
|
|
|
|
@require_config(DonationConfiguration)
|
|
@require_POST
|
|
@login_required
|
|
def donate(request):
|
|
"""Add a single donation item to the cart and proceed to payment.
|
|
|
|
Warning: this call will clear all the items in the user's cart
|
|
before adding the new item!
|
|
|
|
Arguments:
|
|
request (Request): The Django request object. This should contain
|
|
a JSON-serialized dictionary with "amount" (string, required),
|
|
and "course_id" (slash-separated course ID string, optional).
|
|
|
|
Returns:
|
|
HttpResponse: 200 on success with JSON-encoded dictionary that has keys
|
|
"payment_url" (string) and "payment_params" (dictionary). The client
|
|
should POST the payment params to the payment URL.
|
|
HttpResponse: 400 invalid amount or course ID.
|
|
HttpResponse: 404 donations are disabled.
|
|
HttpResponse: 405 invalid request method.
|
|
|
|
Example usage:
|
|
|
|
POST /shoppingcart/donation/
|
|
with params {'amount': '12.34', course_id': 'edX/DemoX/Demo_Course'}
|
|
will respond with the signed purchase params
|
|
that the client can send to the payment processor.
|
|
|
|
"""
|
|
amount = request.POST.get('amount')
|
|
course_id = request.POST.get('course_id')
|
|
|
|
# Check that required parameters are present and valid
|
|
if amount is None:
|
|
msg = u"Request is missing required param 'amount'"
|
|
log.error(msg)
|
|
return HttpResponseBadRequest(msg)
|
|
try:
|
|
amount = (
|
|
decimal.Decimal(amount)
|
|
).quantize(
|
|
decimal.Decimal('.01'),
|
|
rounding=decimal.ROUND_DOWN
|
|
)
|
|
except decimal.InvalidOperation:
|
|
return HttpResponseBadRequest("Could not parse 'amount' as a decimal")
|
|
|
|
# Any amount is okay as long as it's greater than 0
|
|
# Since we've already quantized the amount to 0.01
|
|
# and rounded down, we can check if it's less than 0.01
|
|
if amount < decimal.Decimal('0.01'):
|
|
return HttpResponseBadRequest("Amount must be greater than 0")
|
|
|
|
if course_id is not None:
|
|
try:
|
|
course_id = CourseLocator.from_string(course_id)
|
|
except InvalidKeyError:
|
|
msg = u"Request included an invalid course key: {course_key}".format(course_key=course_id)
|
|
log.error(msg)
|
|
return HttpResponseBadRequest(msg)
|
|
|
|
# Add the donation to the user's cart
|
|
cart = Order.get_cart_for_user(request.user)
|
|
cart.clear()
|
|
|
|
try:
|
|
# Course ID may be None if this is a donation to the entire organization
|
|
Donation.add_to_order(cart, amount, course_id=course_id)
|
|
except InvalidCartItem as ex:
|
|
log.exception((
|
|
u"Could not create donation item for "
|
|
u"amount '{amount}' and course ID '{course_id}'"
|
|
).format(amount=amount, course_id=course_id))
|
|
return HttpResponseBadRequest(unicode(ex))
|
|
|
|
# Start the purchase.
|
|
# This will "lock" the purchase so the user can't change
|
|
# the amount after we send the information to the payment processor.
|
|
# If the user tries to make another donation, it will be added
|
|
# to a new cart.
|
|
cart.start_purchase()
|
|
|
|
# Construct the response params (JSON-encoded)
|
|
callback_url = request.build_absolute_uri(
|
|
reverse("shoppingcart.views.postpay_callback")
|
|
)
|
|
|
|
response_params = json.dumps({
|
|
# The HTTP end-point for the payment processor.
|
|
"payment_url": get_purchase_endpoint(),
|
|
|
|
# Parameters the client should send to the payment processor
|
|
"payment_params": get_signed_purchase_params(
|
|
cart,
|
|
callback_url=callback_url,
|
|
extra_data=([unicode(course_id)] if course_id else None)
|
|
),
|
|
})
|
|
|
|
return HttpResponse(response_params, content_type="text/json")
|
|
|
|
|
|
@csrf_exempt
|
|
@require_POST
|
|
def postpay_callback(request):
|
|
"""
|
|
Receives the POST-back from processor.
|
|
Mainly this calls the processor-specific code to check if the payment was accepted, and to record the order
|
|
if it was, and to generate an error page.
|
|
If successful this function should have the side effect of changing the "cart" into a full "order" in the DB.
|
|
The cart can then render a success page which links to receipt pages.
|
|
If unsuccessful the order will be left untouched and HTML messages giving more detailed error info will be
|
|
returned.
|
|
"""
|
|
params = request.POST.dict()
|
|
result = process_postpay_callback(params)
|
|
if result['success']:
|
|
return HttpResponseRedirect(reverse('shoppingcart.views.show_receipt', args=[result['order'].id]))
|
|
else:
|
|
return render_to_response('shoppingcart/error.html', {'order': result['order'],
|
|
'error_html': result['error_html']})
|
|
|
|
|
|
@require_http_methods(["GET", "POST"])
|
|
@login_required
|
|
def billing_details(request):
|
|
"""
|
|
This is the view for capturing additional billing details
|
|
in case of the business purchase workflow.
|
|
"""
|
|
|
|
cart = Order.get_cart_for_user(request.user)
|
|
cart_items = cart.orderitem_set.all()
|
|
|
|
if getattr(cart, 'order_type') != OrderTypes.BUSINESS:
|
|
raise Http404('Page not found!')
|
|
|
|
if request.method == "GET":
|
|
callback_url = request.build_absolute_uri(
|
|
reverse("shoppingcart.views.postpay_callback")
|
|
)
|
|
form_html = render_purchase_form_html(cart, callback_url=callback_url)
|
|
total_cost = cart.total_cost
|
|
context = {
|
|
'shoppingcart_items': cart_items,
|
|
'amount': total_cost,
|
|
'form_html': form_html,
|
|
'site_name': microsite.get_value('SITE_NAME', settings.SITE_NAME),
|
|
}
|
|
return render_to_response("shoppingcart/billing_details.html", context)
|
|
elif request.method == "POST":
|
|
company_name = request.POST.get("company_name", "")
|
|
company_contact_name = request.POST.get("company_contact_name", "")
|
|
company_contact_email = request.POST.get("company_contact_email", "")
|
|
recipient_name = request.POST.get("recipient_name", "")
|
|
recipient_email = request.POST.get("recipient_email", "")
|
|
customer_reference_number = request.POST.get("customer_reference_number", "")
|
|
|
|
cart.add_billing_details(company_name, company_contact_name, company_contact_email, recipient_name,
|
|
recipient_email, customer_reference_number)
|
|
return JsonResponse({
|
|
'response': _('success')
|
|
}) # status code 200: OK by default
|
|
|
|
|
|
@login_required
|
|
def show_receipt(request, ordernum):
|
|
"""
|
|
Displays a receipt for a particular order.
|
|
404 if order is not yet purchased or request.user != order.user
|
|
"""
|
|
|
|
try:
|
|
order = Order.objects.get(id=ordernum)
|
|
except Order.DoesNotExist:
|
|
raise Http404('Order not found!')
|
|
|
|
if order.user != request.user or order.status != 'purchased':
|
|
raise Http404('Order not found!')
|
|
|
|
order_items = OrderItem.objects.filter(order=order).select_subclasses()
|
|
shoppingcart_items = []
|
|
course_names_list = []
|
|
for order_item in order_items:
|
|
course_key = getattr(order_item, 'course_id')
|
|
if course_key:
|
|
course = get_course_by_id(course_key, depth=0)
|
|
shoppingcart_items.append((order_item, course))
|
|
course_names_list.append(course.display_name)
|
|
|
|
appended_course_names = ", ".join(course_names_list)
|
|
any_refunds = any(i.status == "refunded" for i in order_items)
|
|
receipt_template = 'shoppingcart/receipt.html'
|
|
__, instructions = order.generate_receipt_instructions()
|
|
order_type = getattr(order, 'order_type')
|
|
|
|
# Only orders where order_items.count() == 1 might be attempting to upgrade
|
|
attempting_upgrade = request.session.get('attempting_upgrade', False)
|
|
if attempting_upgrade:
|
|
course_enrollment = CourseEnrollment.get_or_create_enrollment(request.user, order_items[0].course_id)
|
|
course_enrollment.emit_event(EVENT_NAME_USER_UPGRADED)
|
|
request.session['attempting_upgrade'] = False
|
|
|
|
recipient_list = []
|
|
registration_codes = None
|
|
total_registration_codes = None
|
|
recipient_list.append(getattr(order.user, 'email'))
|
|
if order_type == OrderTypes.BUSINESS:
|
|
registration_codes = CourseRegistrationCode.objects.filter(order=order)
|
|
total_registration_codes = registration_codes.count()
|
|
if order.company_contact_email:
|
|
recipient_list.append(order.company_contact_email)
|
|
if order.recipient_email:
|
|
recipient_list.append(order.recipient_email)
|
|
|
|
appended_recipient_emails = ", ".join(recipient_list)
|
|
|
|
context = {
|
|
'order': order,
|
|
'shoppingcart_items': shoppingcart_items,
|
|
'any_refunds': any_refunds,
|
|
'instructions': instructions,
|
|
'site_name': microsite.get_value('SITE_NAME', settings.SITE_NAME),
|
|
'order_type': order_type,
|
|
'appended_course_names': appended_course_names,
|
|
'appended_recipient_emails': appended_recipient_emails,
|
|
'total_registration_codes': total_registration_codes,
|
|
'registration_codes': registration_codes,
|
|
'order_purchase_date': order.purchase_time.strftime("%B %d, %Y"),
|
|
}
|
|
# we want to have the ability to override the default receipt page when
|
|
# there is only one item in the order
|
|
if order_items.count() == 1:
|
|
receipt_template = order_items[0].single_item_receipt_template
|
|
context.update(order_items[0].single_item_receipt_context)
|
|
|
|
return render_to_response(receipt_template, context)
|
|
|
|
|
|
def _can_download_report(user):
|
|
"""
|
|
Tests if the user can download the payments report, based on membership in a group whose name is determined
|
|
in settings. If the group does not exist, denies all access
|
|
"""
|
|
try:
|
|
access_group = Group.objects.get(name=settings.PAYMENT_REPORT_GENERATOR_GROUP)
|
|
except Group.DoesNotExist:
|
|
return False
|
|
return access_group in user.groups.all()
|
|
|
|
|
|
def _get_date_from_str(date_input):
|
|
"""
|
|
Gets date from the date input string. Lets the ValueError raised by invalid strings be processed by the caller
|
|
"""
|
|
return datetime.datetime.strptime(date_input.strip(), "%Y-%m-%d").replace(tzinfo=pytz.UTC)
|
|
|
|
|
|
def _render_report_form(start_str, end_str, start_letter, end_letter, report_type, total_count_error=False, date_fmt_error=False):
|
|
"""
|
|
Helper function that renders the purchase form. Reduces repetition
|
|
"""
|
|
context = {
|
|
'total_count_error': total_count_error,
|
|
'date_fmt_error': date_fmt_error,
|
|
'start_date': start_str,
|
|
'end_date': end_str,
|
|
'start_letter': start_letter,
|
|
'end_letter': end_letter,
|
|
'requested_report': report_type,
|
|
}
|
|
return render_to_response('shoppingcart/download_report.html', context)
|
|
|
|
|
|
@login_required
|
|
def csv_report(request):
|
|
"""
|
|
Downloads csv reporting of orderitems
|
|
"""
|
|
if not _can_download_report(request.user):
|
|
return HttpResponseForbidden(_('You do not have permission to view this page.'))
|
|
|
|
if request.method == 'POST':
|
|
start_date = request.POST.get('start_date', '')
|
|
end_date = request.POST.get('end_date', '')
|
|
start_letter = request.POST.get('start_letter', '')
|
|
end_letter = request.POST.get('end_letter', '')
|
|
report_type = request.POST.get('requested_report', '')
|
|
try:
|
|
start_date = _get_date_from_str(start_date) + datetime.timedelta(days=0)
|
|
end_date = _get_date_from_str(end_date) + datetime.timedelta(days=1)
|
|
except ValueError:
|
|
# Error case: there was a badly formatted user-input date string
|
|
return _render_report_form(start_date, end_date, start_letter, end_letter, report_type, date_fmt_error=True)
|
|
|
|
report = initialize_report(report_type, start_date, end_date, start_letter, end_letter)
|
|
items = report.rows()
|
|
|
|
response = HttpResponse(mimetype='text/csv')
|
|
filename = "purchases_report_{}.csv".format(datetime.datetime.now(pytz.UTC).strftime("%Y-%m-%d-%H-%M-%S"))
|
|
response['Content-Disposition'] = 'attachment; filename="{}"'.format(filename)
|
|
report.write_csv(response)
|
|
return response
|
|
|
|
elif request.method == 'GET':
|
|
end_date = datetime.datetime.now(pytz.UTC)
|
|
start_date = end_date - datetime.timedelta(days=30)
|
|
start_letter = ""
|
|
end_letter = ""
|
|
return _render_report_form(start_date.strftime("%Y-%m-%d"), end_date.strftime("%Y-%m-%d"), start_letter, end_letter, report_type="")
|
|
|
|
else:
|
|
return HttpResponseBadRequest("HTTP Method Not Supported")
|