Skip to content

Commit

Permalink
Merge In Master
Browse files Browse the repository at this point in the history
  • Loading branch information
shiva-menta committed Nov 11, 2024
2 parents b559803 + 323f536 commit 26db6e3
Show file tree
Hide file tree
Showing 26 changed files with 1,273 additions and 651 deletions.
2 changes: 1 addition & 1 deletion .devcontainer/Dockerfile
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
FROM --platform=linux/amd64 ubuntu:22.04
ARG IMAGE_NAME=pennlabs/courses-devcontainer
RUN apt-get update && apt-get install -y wget curl gcc python3-dev libpq-dev
RUN apt-get update && apt-get install -y wget curl gcc python3-dev libpq-dev postgresql-client
14 changes: 0 additions & 14 deletions backend/Dockerfile.dev

This file was deleted.

129 changes: 129 additions & 0 deletions backend/PennCourses/docs_settings.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,12 +6,15 @@
from textwrap import dedent

import jsonref
from django.db import models
from django.urls import get_resolver
from rest_framework import serializers
from rest_framework.fields import _UnvalidatedField
from rest_framework.permissions import IsAuthenticated
from rest_framework.renderers import JSONOpenAPIRenderer
from rest_framework.schemas.openapi import AutoSchema
from rest_framework.schemas.utils import is_list_view
from rest_framework.settings import api_settings


"""
Expand Down Expand Up @@ -353,6 +356,7 @@ def get_url_by_name(name):
"review": "PCR",
"base": "PCx",
"accounts": "Accounts",
"degree": "PDP",
}
assert all(
[isinstance(key, str) and isinstance(val, str) for key, val in subpath_abbreviations.items()]
Expand Down Expand Up @@ -991,6 +995,7 @@ def map_serializer(self, serializer):
"""

result = super().map_serializer(serializer)

properties = result["properties"]
model = None
if hasattr(serializer, "Meta") and hasattr(serializer.Meta, "model"):
Expand All @@ -1011,6 +1016,130 @@ def map_serializer(self, serializer):

return result

# Overrides, uses overridden method
# (https://www.django-rest-framework.org/api-guide/schemas/#map_field)
def map_field(self, field):

# Nested Serializers, `many` or not.
if isinstance(field, serializers.ListSerializer):
return {"type": "array", "items": []}
if isinstance(field, serializers.Serializer):
data = self.map_serializer(field)
data["type"] = "object"
return data

# Related fields.
if isinstance(field, serializers.ManyRelatedField):
return {"type": "array", "items": self.map_field(field.child_relation)}
if isinstance(field, serializers.PrimaryKeyRelatedField):
if getattr(field, "pk_field", False):
return self.map_field(field=field.pk_field)
model = getattr(field.queryset, "model", None)
if model is not None:
model_field = model._meta.pk
if isinstance(model_field, models.AutoField):
return {"type": "integer"}

# ChoiceFields (single and multiple).
# Q:
# - Is 'type' required?
# - can we determine the TYPE of a choicefield?
if isinstance(field, serializers.MultipleChoiceField):
return {"type": "array", "items": self.map_choicefield(field)}

if isinstance(field, serializers.ChoiceField):
return self.map_choicefield(field)

# ListField.
if isinstance(field, serializers.ListField):
mapping = {
"type": "array",
"items": {},
}
if not isinstance(field.child, _UnvalidatedField):
mapping["items"] = self.map_field(field.child)
return mapping

# DateField and DateTimeField type is string
if isinstance(field, serializers.DateField):
return {
"type": "string",
"format": "date",
}

if isinstance(field, serializers.DateTimeField):
return {
"type": "string",
"format": "date-time",
}

# "Formats such as "email", "uuid", and so on, MAY be used even though undefined by this
# specification."
# see: https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#data-types
# see also: https://swagger.io/docs/specification/data-models/data-types/#string
if isinstance(field, serializers.EmailField):
return {"type": "string", "format": "email"}

if isinstance(field, serializers.URLField):
return {"type": "string", "format": "uri"}

if isinstance(field, serializers.UUIDField):
return {"type": "string", "format": "uuid"}

if isinstance(field, serializers.IPAddressField):
content = {
"type": "string",
}
if field.protocol != "both":
content["format"] = field.protocol
return content

if isinstance(field, serializers.DecimalField):
if getattr(field, "coerce_to_string", api_settings.COERCE_DECIMAL_TO_STRING):
content = {
"type": "string",
"format": "decimal",
}
else:
content = {"type": "number"}

if field.decimal_places:
content["multipleOf"] = float("." + (field.decimal_places - 1) * "0" + "1")
if field.max_whole_digits:
content["maximum"] = int(field.max_whole_digits * "9") + 1
content["minimum"] = -content["maximum"]
self._map_min_max(field, content)
return content

if isinstance(field, serializers.FloatField):
content = {
"type": "number",
}
self._map_min_max(field, content)
return content

if isinstance(field, serializers.IntegerField):
content = {"type": "integer"}
self._map_min_max(field, content)
# 2147483647 is max for int32_size, so we use int64 for format
if int(content.get("maximum", 0)) > 2147483647:
content["format"] = "int64"
if int(content.get("minimum", 0)) > 2147483647:
content["format"] = "int64"
return content

if isinstance(field, serializers.FileField):
return {"type": "string", "format": "binary"}

# Simplest cases, default to 'string' type:
FIELD_CLASS_SCHEMA_TYPE = {
serializers.BooleanField: "boolean",
serializers.JSONField: "object",
serializers.DictField: "object",
serializers.HStoreField: "object",
}
return {"type": FIELD_CLASS_SCHEMA_TYPE.get(field.__class__, "string")}

# Helper method
def get_action(self, path, method):
"""
Expand Down
3 changes: 3 additions & 0 deletions backend/PennCourses/settings/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -238,3 +238,6 @@
# The name of the schedule that is created/verified by Penn Mobile,
# containing the user's active course registrations from Path.
PATH_REGISTRATION_SCHEDULE_NAME = "Path Registration"

# Manually Set Cache Prefix
CACHE_PREFIX = "MANUAL_CACHE_"
Loading

0 comments on commit 26db6e3

Please sign in to comment.