/app/soc/modules/gsoc/logic/project.py

https://code.google.com/ · Python · 180 lines · 66 code · 33 blank · 81 comment · 8 complexity · 04c85dff3a29316b50e410f11de71406 MD5 · raw file

  1. #!/usr/bin/env python2.5
  2. #
  3. # Copyright 2011 the Melange authors.
  4. #
  5. # Licensed under the Apache License, Version 2.0 (the "License");
  6. # you may not use this file except in compliance with the License.
  7. # You may obtain a copy of the License at
  8. #
  9. # http://www.apache.org/licenses/LICENSE-2.0
  10. #
  11. # Unless required by applicable law or agreed to in writing, software
  12. # distributed under the License is distributed on an "AS IS" BASIS,
  13. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. # See the License for the specific language governing permissions and
  15. # limitations under the License.
  16. """Logic for GSoC Project Model.
  17. """
  18. import datetime
  19. from google.appengine.api import memcache
  20. from google.appengine.ext import db
  21. from soc.modules.gsoc.models.project import GSoCProject
  22. NUMBER_OF_EVALUATIONS = 2
  23. def getFeaturedProject(current_timeline, program):
  24. """Return a featured project for a given program.
  25. Args:
  26. current_timeline: where we are currently on the program timeline
  27. program: entity representing the program from which the featured
  28. projects should be fetched
  29. """
  30. # expiry time to fetch the new featured project entity
  31. # the current expiry time is 2 hours.
  32. expiry_time = datetime.timedelta(seconds=7200)
  33. def queryForProject():
  34. query = GSoCProject.all()
  35. query.filter('is_featured', True)
  36. query.filter('program', program)
  37. if current_timeline == 'coding_period':
  38. project_status = 'accepted'
  39. else:
  40. project_status = 'completed'
  41. query.filter('status', project_status)
  42. return query
  43. q = queryForProject()
  44. # the cache stores a 3-tuple in the order student_project entity,
  45. # cursor and the last time the cache was updated
  46. fsp_cache = memcache.get('featured_gsoc_project' + program.key().name())
  47. if fsp_cache:
  48. cached_project, cached_cursor, cache_expiry_time = fsp_cache
  49. if not datetime.datetime.now() > cache_expiry_time + expiry_time:
  50. return cached_project
  51. else:
  52. q.with_cursor(cached_cursor)
  53. if q.count() == 0:
  54. q = queryForProject()
  55. new_project = q.get()
  56. new_cursor = q.cursor()
  57. memcache.set(
  58. key='featured_gsoc_project',
  59. value=(new_project, new_cursor, datetime.datetime.now()))
  60. return new_project
  61. def getProjectsQuery(keys_only=False, ancestor=None, **properties):
  62. """Returns the Appengine GSoCProject query object for the given set
  63. of properties.
  64. Args:
  65. ancestor: The student for which the accepted projects must be fetched.
  66. properties: keyword arguments containing the properties for which the
  67. query must be constructed.
  68. """
  69. q = db.Query(GSoCProject, keys_only=keys_only)
  70. if ancestor:
  71. q.ancestor(ancestor)
  72. for k, v in properties.items():
  73. q.filter(k, v)
  74. return q
  75. def getAcceptedProjectsQuery(keys_only=False, ancestor=None, **properties):
  76. """Returns the Appengine GSoCProject query object for the given
  77. set of properties for accepted projects.
  78. Args:
  79. ancestor: The student for which the accepted projects must be fetched.
  80. properties: keyword arguments containing the properties for which the
  81. query must be constructed.
  82. """
  83. q = getProjectsQuery(keys_only, ancestor, **properties)
  84. q.filter('status', 'accepted')
  85. return q
  86. def getAcceptedProjectsForOrg(org, limit=1000):
  87. """Returns all the accepted projects for a given organization.
  88. Args:
  89. org: The organization entity for which the accepted projects are accepted.
  90. """
  91. q = getAcceptedProjectsQuery(org=org)
  92. return q.fetch(limit)
  93. def getAcceptedProjectsForStudent(student, limit=1000):
  94. """Returns all the accepted projects for a given student.
  95. Args:
  96. student: The student for whom the projects should be retrieved.
  97. """
  98. q = getAcceptedProjectsQuery(ancestor=student)
  99. return q.fetch(limit)
  100. def getProjectsQueryForOrgs(orgs):
  101. """Returns the query corresponding to projects for the given organization(s).
  102. Args:
  103. orgs: The list of organization entities for which the projects
  104. should be queried.
  105. """
  106. q = getProjectsQuery()
  107. orgs = list(orgs)
  108. q.filter('org IN', orgs)
  109. return q
  110. def getProjectsQueryForEval(keys_only=False, ancestor=None, **properties):
  111. """Returns the query corresponding to projects to be evaluated.
  112. This is a special query needed to build evaluation lists.
  113. """
  114. q = getProjectsQuery(keys_only, ancestor, **properties)
  115. q.filter('status IN', ['accepted', 'failed', 'completed'])
  116. return q
  117. def getProjectsQueryForEvalForOrgs(orgs):
  118. """Returns the query corresponding to projects for the given organization(s).
  119. This is a special query needed to build evaluation lists.
  120. Args:
  121. orgs: The list of organization entities for which the projects
  122. should be queried.
  123. """
  124. q = getProjectsQueryForOrgs(orgs)
  125. q.filter('status IN', ['accepted', 'failed', 'completed'])
  126. return q
  127. def getProjectsForOrgs(orgs, limit=1000):
  128. """Returns all the projects for the given organization(s).
  129. Unlike getAcceptedProjectsForOrg function, this returns all the projects
  130. for all the orgs listed
  131. Args:
  132. orgs: The list of organization entities for which the projects
  133. should be queried.
  134. """
  135. q = getProjectsQueryForOrgs(orgs)
  136. return q.fetch(limit)