Skip to content

Commit eabe72f

Browse files
committed
Update challenge
1 parent 8670edd commit eabe72f

22 files changed

+155
-87
lines changed

README.md

+73-29
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,21 @@
1-
# API Team Challenge
1+
# Python Data Challenge
22

3-
Thank you for your interest in joining G Adventures, and specifically being part
4-
of the API Team! We're really excited to see what you can do.
3+
Thank you for your interest in joining G Adventures. This challenge will allow
4+
us to further evaluate where you may potentially fit from a technical
5+
perspective. We're really excited to see what you can do.
56

67
## Submission
78

89
* Fork this project on Github. You will need to create an account if you don't already have one.
9-
* Complete the project as described within your fork.
10+
* Complete the project as described, within your fork.
1011
* Push all of your changes to your fork on github and submit a pull request.
1112
* We will be notified of your pull request (or feel free to email us as well), and we'll review!
1213

1314
If you don't wish to publicize your work on this challenge, you may simply send
14-
the completed project in an archived file to [[email protected]](mailto:[email protected])
15+
the completed project in an archived file to [[email protected]](mailto:[email protected])
16+
17+
We recommend using *Python 3* for all work you produce, but we're also happy if
18+
you use *Go*, or *Javascript*
1519

1620
## Project
1721

@@ -20,32 +24,77 @@ get familiar with documentation, please take your time. In the end, we don't
2024
want you to feel rushed, but we also don't want you spending too much time on
2125
this.
2226

23-
For this project, we want you to use the [Django REST Framework](http://www.django-rest-framework.org/) library to construct an API that displays a list of Trips.
27+
For this project, you'll build a script which consumes data from a local API
28+
(provided). Your script will transform this data in various ways (described
29+
below), and be output as a CSV.
30+
31+
### Local API
32+
33+
Provided within this project is a local API, which uses [Django REST Framework](http://www.django-rest-framework.org/)
34+
to provide a simple, standards-compliant API. Below, within the `Getting
35+
Started` section, you'll find instructions on how to run the API server.
36+
37+
Once running, your only task for the server, is to load data into the API. We've provided a
38+
`departures.json` file, which you should load into the server using [Django Data
39+
Migrations](https://docs.djangoproject.com/en/2.0/topics/migrations/#data-migrations).
40+
You will need to make a new *empty* migration as per the documentation, and
41+
write the code to read the file and [insert it into the Departure
42+
Model](https://docs.djangoproject.com/en/2.0/ref/models/instances/#creating-objects).
43+
44+
After this data is loaded, you should be able to view it all at
45+
`http://localhost:8000/departures`, assuming you're running the local API
46+
server.
47+
48+
### Data Collection Script
49+
50+
Your primary task within this project is to build a script which consumes data
51+
from your local API. Within this API, you have a `/departures` URL, which is a
52+
multi-page listing of trips, operating on various dates.
53+
54+
The script will be expected to query for `/departures`, and iterate over every
55+
page, collecting all `departures` referenced from the API.
2456

25-
A Trip model has been supplied for you within the project. It has a `name`,
26-
`start_date`, and `finish_date`. If you wish to extend this model, feel free to
27-
do so. The model has no data, you'll want to understand how to interact with
28-
[Django Models](https://docs.djangoproject.com/en/1.9/topics/db/models/) to
29-
create some data. Feel free to include a script of creating such data in your
30-
submission.
57+
To iterate over a page, the script simply needs to follow the `next` links,
58+
which look like so in the API response:
3159

32-
Your API will provide this Trip data in a RESTful manner, ideally responding in
33-
JSON (Built-in to Django REST Framework)
60+
```
61+
{
62+
"count": 100,
63+
"next": "http://localhost:8000/departures/?limit=50&offset=50",
64+
"previous": null,
65+
"results": [`
66+
...
67+
]
3468
35-
Once you have constructed the API to display these trips, you'll create a page
36-
that displays the list of Trips, as per the response of the API.
69+
```
3770

38-
Stylistically, how you display these trips is up to you. Data-wise, as long as the data is coming from the API view you've built, we
39-
are agnostic as to what technology you use to query that view. Generally, it'd be in Javascript using an _AJAX_ call, or you can do something else. Feel like displaying it on an Android application instead? That's fine, as long as you built the API for it :)
71+
Once the script has collected all `departures`, from all pages, it will need to
72+
filter down the data to only include the following:
73+
74+
1. Filter down so that your local data only contains `departures` with a `start_date` after `June 1st, 2018`
75+
76+
2. Additionally, filter down to only `departures` of `category = Adventurous`
77+
78+
At this point, your script should have a subset of the data fetched from the
79+
API. You will now want to write this information into a CSV. For the output,
80+
ensure the following:
81+
82+
83+
1. Every attribute within `departures` is given its own column, and there's a
84+
header within the CSV with the attribute names, title-case.
85+
2. Every remaining `departures` instance is written to a row.
86+
87+
The CSV should be written to the root folder of your project, with whatever file
88+
name you believe is relevant.
4089

4190
... And that's it! Feel free to try new technologies, as long you're within the
4291
scope of the challenge requirements, we're happy.
4392

4493
## Goal
4594

4695
We'd like to get a sense of how you work, specifically within areas that are
47-
unexplored territory, and if you are able to fulfill the requirements scoped
48-
for a project.
96+
unexplored territory, and if you are able to fulfill all the requirements scoped
97+
for a task.
4998

5099
We're looking for code that is well structured, documented, and testable.
51100

@@ -73,22 +122,17 @@ Once installed, you can run the Django project like so:
73122

74123
python manage.py runserver
75124

76-
You should now be able to visit your project at `http://127.0.0.1:8000/trips`
77-
78-
We've included a basic Django project and a `Trip` model with some fields.
79-
You'll want to focus on understanding how to integrate Django REST Framework
80-
into an existing Django project. Their documentation is great!
125+
You should now be able to visit your project at `http://127.0.0.1:8000/departures`
81126

82-
We have also supplied a blank template, which lives in
83-
`trips/templates/trips/index.html`. This is the template you see when you visit
84-
`/trips` within your project.
127+
We've included a basic Django project and a `Departure` model with some fields,
128+
which you will consume.
85129

86130
Otherwise, the rest is up to you!
87131

88132
## Stuck?
89133

90134
If you get stuck -- Please don't hesitate to email
91-
[bartekc@gadventures.com](mailto:bartekc@gadventures.com). We are looking for
135+
[recruiting_tech@gadventures.com](mailto:recruiting_tech@gadventures.com). We are looking for
92136
candidates who are not afraid to ask questions, and explore new ideas. Asking
93137
questions will not hurt your chances.
94138

apichallenge/settings.py

+17-12
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,13 @@
11
"""
22
Django settings for apichallenge project.
33
4-
Generated by 'django-admin startproject' using Django 1.9.2.
4+
Generated by 'django-admin startproject' using Django 2.0.2.
55
66
For more information on this file, see
7-
https://docs.djangoproject.com/en/1.9/topics/settings/
7+
https://docs.djangoproject.com/en/2.0/topics/settings/
88
99
For the full list of settings and their values, see
10-
https://docs.djangoproject.com/en/1.9/ref/settings/
10+
https://docs.djangoproject.com/en/2.0/ref/settings/
1111
"""
1212

1313
import os
@@ -17,10 +17,10 @@
1717

1818

1919
# Quick-start development settings - unsuitable for production
20-
# See https://docs.djangoproject.com/en/1.9/howto/deployment/checklist/
20+
# See https://docs.djangoproject.com/en/2.0/howto/deployment/checklist/
2121

2222
# SECURITY WARNING: keep the secret key used in production secret!
23-
SECRET_KEY = 'd(_v6s%d0zcb6#u1y9@u4)pvo@3hsc49cupfyd)3%gmziq95k$'
23+
SECRET_KEY = '!daaozar3!$-e$z+=w3d2afsol(b(g!&z^(*$gn2=hj8sw9q25'
2424

2525
# SECURITY WARNING: don't run with debug turned on in production!
2626
DEBUG = True
@@ -37,16 +37,21 @@
3737
'django.contrib.sessions',
3838
'django.contrib.messages',
3939
'django.contrib.staticfiles',
40-
'trips',
40+
'rest_framework',
41+
'departures',
4142
]
4243

43-
MIDDLEWARE_CLASSES = [
44+
REST_FRAMEWORK = {
45+
'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.LimitOffsetPagination',
46+
'PAGE_SIZE': 50
47+
}
48+
49+
MIDDLEWARE = [
4450
'django.middleware.security.SecurityMiddleware',
4551
'django.contrib.sessions.middleware.SessionMiddleware',
4652
'django.middleware.common.CommonMiddleware',
4753
'django.middleware.csrf.CsrfViewMiddleware',
4854
'django.contrib.auth.middleware.AuthenticationMiddleware',
49-
'django.contrib.auth.middleware.SessionAuthenticationMiddleware',
5055
'django.contrib.messages.middleware.MessageMiddleware',
5156
'django.middleware.clickjacking.XFrameOptionsMiddleware',
5257
]
@@ -73,7 +78,7 @@
7378

7479

7580
# Database
76-
# https://docs.djangoproject.com/en/1.9/ref/settings/#databases
81+
# https://docs.djangoproject.com/en/2.0/ref/settings/#databases
7782

7883
DATABASES = {
7984
'default': {
@@ -84,7 +89,7 @@
8489

8590

8691
# Password validation
87-
# https://docs.djangoproject.com/en/1.9/ref/settings/#auth-password-validators
92+
# https://docs.djangoproject.com/en/2.0/ref/settings/#auth-password-validators
8893

8994
AUTH_PASSWORD_VALIDATORS = [
9095
{
@@ -103,7 +108,7 @@
103108

104109

105110
# Internationalization
106-
# https://docs.djangoproject.com/en/1.9/topics/i18n/
111+
# https://docs.djangoproject.com/en/2.0/topics/i18n/
107112

108113
LANGUAGE_CODE = 'en-us'
109114

@@ -117,6 +122,6 @@
117122

118123

119124
# Static files (CSS, JavaScript, Images)
120-
# https://docs.djangoproject.com/en/1.9/howto/static-files/
125+
# https://docs.djangoproject.com/en/2.0/howto/static-files/
121126

122127
STATIC_URL = '/static/'

apichallenge/urls.py

+8-8
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,22 @@
11
"""apichallenge URL Configuration
22
33
The `urlpatterns` list routes URLs to views. For more information please see:
4-
https://docs.djangoproject.com/en/1.9/topics/http/urls/
4+
https://docs.djangoproject.com/en/2.0/topics/http/urls/
55
Examples:
66
Function views
77
1. Add an import: from my_app import views
8-
2. Add a URL to urlpatterns: url(r'^$', views.home, name='home')
8+
2. Add a URL to urlpatterns: path('', views.home, name='home')
99
Class-based views
1010
1. Add an import: from other_app.views import Home
11-
2. Add a URL to urlpatterns: url(r'^$', Home.as_view(), name='home')
11+
2. Add a URL to urlpatterns: path('', Home.as_view(), name='home')
1212
Including another URLconf
13-
1. Import the include() function: from django.conf.urls import url, include
14-
2. Add a URL to urlpatterns: url(r'^blog/', include('blog.urls'))
13+
1. Import the include() function: from django.urls import include, path
14+
2. Add a URL to urlpatterns: path('blog/', include('blog.urls'))
1515
"""
16-
from django.conf.urls import url, include
1716
from django.contrib import admin
17+
from django.urls import path, include
1818

1919
urlpatterns = [
20-
url(r'^admin/', admin.site.urls),
21-
url(r'^trips/', include('trips.urls')),
20+
path('admin/', admin.site.urls),
21+
path('departures/', include('departures.urls')),
2222
]

apichallenge/wsgi.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
It exposes the WSGI callable as a module-level variable named ``application``.
55
66
For more information on this file, see
7-
https://docs.djangoproject.com/en/1.9/howto/deployment/wsgi/
7+
https://docs.djangoproject.com/en/2.0/howto/deployment/wsgi/
88
"""
99

1010
import os

db.sqlite3

16 KB
Binary file not shown.

departures.json

+1
Large diffs are not rendered by default.
File renamed without changes.

trips/admin.py departures/admin.py

File renamed without changes.

departures/apps.py

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
from django.apps import AppConfig
2+
3+
4+
class DeparturesConfig(AppConfig):
5+
name = 'departures'

trips/migrations/0001_initial.py departures/migrations/0001_initial.py

+3-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,4 @@
1-
# -*- coding: utf-8 -*-
2-
# Generated by Django 1.9.2 on 2016-02-03 03:57
3-
from __future__ import unicode_literals
1+
# Generated by Django 2.0.2 on 2018-02-12 02:23
42

53
from django.db import migrations, models
64

@@ -14,12 +12,13 @@ class Migration(migrations.Migration):
1412

1513
operations = [
1614
migrations.CreateModel(
17-
name='Trip',
15+
name='Departure',
1816
fields=[
1917
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
2018
('name', models.CharField(max_length=55)),
2119
('start_date', models.DateField()),
2220
('finish_date', models.DateField()),
21+
('category', models.CharField(choices=[('Adventurous', 'Adventurous'), ('Classic', 'Classic'), ('Marine', 'Marine')], max_length=25)),
2322
],
2423
),
2524
]
File renamed without changes.

departures/models.py

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
from __future__ import unicode_literals
2+
3+
from django.db import models
4+
5+
6+
class Departure(models.Model):
7+
name = models.CharField(max_length=55)
8+
start_date = models.DateField()
9+
finish_date = models.DateField()
10+
category = models.CharField(
11+
max_length=25,
12+
choices=(
13+
('Adventurous', 'Adventurous'),
14+
('Classic', 'Classic'),
15+
('Marine', 'Marine'),
16+
)
17+
)
File renamed without changes.

trips/tests.py departures/tests.py

File renamed without changes.

departures/urls.py

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
from django.conf.urls import url
2+
3+
from .views import DepartureView
4+
5+
urlpatterns = [
6+
url('^', DepartureView.as_view())
7+
]

departures/views.py

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
from rest_framework import generics
2+
from rest_framework import serializers
3+
4+
from .models import Departure
5+
6+
class DepartureSerializer(serializers.ModelSerializer):
7+
class Meta:
8+
model = Departure
9+
fields = ('name', 'start_date', 'finish_date', 'category')
10+
11+
class DepartureView(generics.ListAPIView):
12+
queryset = Departure.objects.all()
13+
serializer_class = DepartureSerializer

manage.py

+8-3
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,12 @@
44

55
if __name__ == "__main__":
66
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "apichallenge.settings")
7-
8-
from django.core.management import execute_from_command_line
9-
7+
try:
8+
from django.core.management import execute_from_command_line
9+
except ImportError as exc:
10+
raise ImportError(
11+
"Couldn't import Django. Are you sure it's installed and "
12+
"available on your PYTHONPATH environment variable? Did you "
13+
"forget to activate a virtual environment?"
14+
) from exc
1015
execute_from_command_line(sys.argv)

requirements.txt

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,2 @@
1-
Django==1.9.2
2-
djangorestframework==3.3.2
1+
Django==2.0.2
2+
djangorestframework==3.7.7

trips/apps.py

-7
This file was deleted.

trips/models.py

-9
This file was deleted.

trips/urls.py

-7
This file was deleted.

trips/views.py

-5
This file was deleted.

0 commit comments

Comments
 (0)