Create initial database schema and data ingest #48
22 changed files with 1396 additions and 30 deletions
89
README.md
89
README.md
|
@ -1,9 +1,7 @@
|
|||
# RightTree
|
||||
|
||||
Right Plant Right Place Right Time implementation using React and Django.
|
||||
|
||||
## Running application for development
|
||||
### Initial Setup
|
||||
## Initial Setup
|
||||
|
||||
Before running the applications please ensure the following prerequisites have been met.
|
||||
#### Software
|
||||
|
@ -15,20 +13,56 @@ $ sudo apt install git docker-compose
|
|||
|
||||
To install `docker`, follow the [official installation documentation](https://docs.docker.com/get-docker/). [Instructions are also available for `docker-compose`](https://docs.docker.com/compose/install/).
|
||||
|
||||
#### Initialise database
|
||||
You may also need to give the `dev` script executable permissions using the following command:
|
||||
|
||||
```
|
||||
chmod +x ./dev
|
||||
```
|
||||
|
||||
### Add shapefiles for database population
|
||||
|
||||
Please unzip and add the following shapefiles to the `./backend/right_tree/api/data/resources` directory. It should include all the files required by the shapefile and use naming conventions as follows:
|
||||
|
||||
**Ecological Districts Shapefile:**
|
||||
```
|
||||
backend/right_tree/api/data/resources/ecological_districts/
|
||||
- DOC_EcologicalDistricts_2021_08_02.cpg
|
||||
- DOC_EcologicalDistricts_2021_08_02.dbf
|
||||
- DOC_EcologicalDistricts_2021_08_02.prj
|
||||
- DOC_EcologicalDistricts_2021_08_02.sbn
|
||||
- DOC_EcologicalDistricts_2021_08_02.sbx
|
||||
- DOC_EcologicalDistricts_2021_08_02.shp
|
||||
- DOC_EcologicalDistricts_2021_08_02.shp.xml
|
||||
- DOC_EcologicalDistricts_2021_08_02.shx
|
||||
```
|
||||
|
||||
**Ecological Districts Shapefile:**
|
||||
```
|
||||
backend/right_tree/api/data/resources/fundamental_soil_layers/
|
||||
- fundamental-soil-layers-new-zealand-soil-classification.cpg
|
||||
- fundamental-soil-layers-new-zealand-soil-classification.dbf
|
||||
- fundamental-soil-layers-new-zealand-soil-classification.prj
|
||||
- fundamental-soil-layers-new-zealand-soil-classification.shp
|
||||
- fundamental-soil-layers-new-zealand-soil-classification.shx
|
||||
- fundamental-soil-layers-new-zealand-soil-classification.xml
|
||||
```
|
||||
### Add spreadsheet data for database population
|
||||
|
||||
The plant spreadsheet should be renamed as `plant_data.xlsx` and placed in the `./backend/right_tree/api/data/resources` directory.
|
||||
## Running application for development
|
||||
### Initial build
|
||||
|
||||
Builds the Django backend docker image. This may need to be re-run if any new dependencies are added.
|
||||
```
|
||||
./dev build
|
||||
```
|
||||
|
||||
### Initialise database
|
||||
|
||||
Creates `right_tree` database and installs `postgis` extensions.
|
||||
|
||||
```
|
||||
chmod +x ./database/init_database.sh
|
||||
./database/init_database.sh
|
||||
```
|
||||
|
||||
#### Initial build
|
||||
|
||||
Builds the Django backend docker image. This may need to be re-run if any new dependencies are added.
|
||||
```
|
||||
docker-compose build
|
||||
./dev init_database
|
||||
```
|
||||
|
||||
### Run web application
|
||||
|
@ -36,7 +70,7 @@ docker-compose build
|
|||
Starts up the applications including the frontend, backend and database.
|
||||
|
||||
```
|
||||
docker-compose up
|
||||
./dev start
|
||||
```
|
||||
|
||||
Once running the components can be accessed as follows:
|
||||
|
@ -46,3 +80,30 @@ Once running the components can be accessed as follows:
|
|||
| React Frontend | http://localhost:3000 |
|
||||
| Django Backend | http://localhost:8000 |
|
||||
| Database | postgis://localhost:5432 |
|
||||
|
||||
## Available commands
|
||||
|
||||
Other commands can be run using the following.
|
||||
```
|
||||
./dev <command>
|
||||
```
|
||||
|
||||
A summary of available commands are outlined below. Note that if the command requires the application to be running (`Requires Run`) please execute `./dev start` in another terminal before running that command.
|
||||
|
||||
| Command | Description | Requires Run |
|
||||
| --- | --- | --- |
|
||||
| `create_database` | Removes the existing database and data. Then it creates the `right_tree` database within a fresh postgis database instance. | No
|
||||
| `makemigrations` | Performs the django `makemigrations` command in the backend container. | Yes
|
||||
| `migrate` | Performs the django `migrate` command in the backend container. | Yes
|
||||
| `createsuperuser` | Performs the django `createsuperuser` command in the backend container. | Yes
|
||||
| `load_fixtures` | Performs the django `loaddata` command in the backend container. This loads all the fixtures found in the `/backend/right_tree/api/data/fixtures` directory. | Yes
|
||||
| `load_shapefiles` | Performs the custom `loadshapefiles` command in the backend container. This loads the ecological districts and soil layers shape files in `c`. | Yes
|
||||
| `create_plant_fixtures` | Performs the custom `createplantfixtures` command in the backend container. This loads the plant spreadsheet data from `/backend/right_tree/api/data/resources/plant_data.xlsx`. Requires the fixtures to be applied and shapefiles loaded. | Yes
|
||||
| `reset_plants` | Performs the custom `resetplants` command in the backend container. This removes all plant entries from the database. | Yes
|
||||
| `load_plant_fixtures` | Loads the `/backend/right_tree/api/data/fixtures/plants.json` fixture. Requires the `plants.json` file to be created (`./dev create_plant_fixtures`) and the plant table to be empty (`./dev reset_plants`). | Yes
|
||||
| `load_plants` | Creates plants fixtures and loads them into a fresh plant table in the database. Requires the fixtures to be applied and shapefiles loaded. | Yes
|
||||
| `populate_database` | Populates the `right_tree` database with base data (fixtures), provided shapefiles and plant spreadsheet data. Requires the database to be created. | No
|
||||
| `init_database` | Creates and populates the database | No
|
||||
| `reset_database` | Removes, recreates and populates the database | No
|
||||
| `build` | Builds required images | No
|
||||
| `start` | Runs all services including the frontend, backend and postgres database | No
|
3
backend/.gitignore
vendored
3
backend/.gitignore
vendored
|
@ -1,3 +1,6 @@
|
|||
*.pyc
|
||||
*.sqlite3
|
||||
__pycache__
|
||||
|
||||
resources
|
||||
right_tree/api/data/fixtures/plants.json
|
|
@ -6,6 +6,16 @@ ENV DJANGO_SUPERUSER_PASSWORD=admin
|
|||
|
||||
WORKDIR /app
|
||||
|
||||
RUN apt update && \
|
||||
apt install -y --no-install-recommends \
|
||||
gdal-bin \
|
||||
libxml2 libxml2-dev gettext \
|
||||
libxslt1-dev libjpeg-dev libpng-dev libpq-dev libgdal-dev \
|
||||
software-properties-common g++ \
|
||||
zlib1g-dev libgeos-dev libproj-dev \
|
||||
sqlite3 spatialite-bin libsqlite3-mod-spatialite && \
|
||||
apt clean
|
||||
|
||||
COPY ./requirements.txt /app/requirements.txt
|
||||
|
||||
RUN pip install -U --no-cache-dir -r requirements.txt
|
||||
|
|
|
@ -2,3 +2,4 @@ Django==3.2.8
|
|||
psycopg2-binary>=2.8
|
||||
djangorestframework==3.12.4
|
||||
django-cors-headers==3.10.0
|
||||
openpyxl==3.0.9
|
|
@ -1,3 +1,10 @@
|
|||
from django.contrib import admin
|
||||
from right_tree.api.models import Plant, SoilOrder, SoilLayer, SoilVariant, EcologicalRegion, EcologicalDistrictLayer, ToleranceLevel
|
||||
|
||||
# Register your models here.
|
||||
admin.site.register(Plant)
|
||||
admin.site.register(SoilOrder)
|
||||
admin.site.register(SoilLayer)
|
||||
admin.site.register(SoilVariant)
|
||||
admin.site.register(EcologicalRegion)
|
||||
admin.site.register(EcologicalDistrictLayer)
|
||||
admin.site.register(ToleranceLevel)
|
||||
|
|
0
backend/right_tree/api/data/__init__.py
Normal file
0
backend/right_tree/api/data/__init__.py
Normal file
555
backend/right_tree/api/data/fixtures/eco_regions.json
Normal file
555
backend/right_tree/api/data/fixtures/eco_regions.json
Normal file
|
@ -0,0 +1,555 @@
|
|||
[
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 1,
|
||||
"fields": {
|
||||
"name": "Aorrangi"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 2,
|
||||
"fields": {
|
||||
"name": "Aspiring"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 3,
|
||||
"fields": {
|
||||
"name": "Auckland"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 4,
|
||||
"fields": {
|
||||
"name": "Banks"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 5,
|
||||
"fields": {
|
||||
"name": "Canterbury Foothills"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 6,
|
||||
"fields": {
|
||||
"name": "Canterbury Plains"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 7,
|
||||
"fields": {
|
||||
"name": "Catlins"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 8,
|
||||
"fields": {
|
||||
"name": "Central Otago"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 9,
|
||||
"fields": {
|
||||
"name": "Central Volcanic Plateau"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 10,
|
||||
"fields": {
|
||||
"name": "Clarence"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 11,
|
||||
"fields": {
|
||||
"name": "Coromandel"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 12,
|
||||
"fields": {
|
||||
"name": "D'Archiac"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 13,
|
||||
"fields": {
|
||||
"name": "East Cape"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 14,
|
||||
"fields": {
|
||||
"name": "Eastern Hawkes Bay"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 15,
|
||||
"fields": {
|
||||
"name": "Eastern Northland"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 16,
|
||||
"fields": {
|
||||
"name": "Eastern Volcanic Plateau"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 17,
|
||||
"fields": {
|
||||
"name": "Eastern Wairarapa"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 18,
|
||||
"fields": {
|
||||
"name": "Egmont"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 19,
|
||||
"fields": {
|
||||
"name": "Fiord"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 20,
|
||||
"fields": {
|
||||
"name": "Gore"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 21,
|
||||
"fields": {
|
||||
"name": "Hawdon"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 22,
|
||||
"fields": {
|
||||
"name": "Hawkes Bay"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 23,
|
||||
"fields": {
|
||||
"name": "Heron"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 24,
|
||||
"fields": {
|
||||
"name": "Inland Marlborough"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 25,
|
||||
"fields": {
|
||||
"name": "Kaikoura"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 26,
|
||||
"fields": {
|
||||
"name": "Kaimanawa"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 27,
|
||||
"fields": {
|
||||
"name": "Kaipara"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 28,
|
||||
"fields": {
|
||||
"name": "Kakanui"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 29,
|
||||
"fields": {
|
||||
"name": "King Country"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 30,
|
||||
"fields": {
|
||||
"name": "Lakes"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 31,
|
||||
"fields": {
|
||||
"name": "Lammerlaw"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 32,
|
||||
"fields": {
|
||||
"name": "Lowry"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 33,
|
||||
"fields": {
|
||||
"name": "MacKenzie"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 34,
|
||||
"fields": {
|
||||
"name": "Makarewa"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 35,
|
||||
"fields": {
|
||||
"name": "Manawatu"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 36,
|
||||
"fields": {
|
||||
"name": "Manawatu Gorge"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 37,
|
||||
"fields": {
|
||||
"name": "Mavora"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 38,
|
||||
"fields": {
|
||||
"name": "Moawhango"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 39,
|
||||
"fields": {
|
||||
"name": "Molesworth"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 40,
|
||||
"fields": {
|
||||
"name": "Nelson"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 41,
|
||||
"fields": {
|
||||
"name": "North Westland"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 42,
|
||||
"fields": {
|
||||
"name": "North-west Nelson"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 43,
|
||||
"fields": {
|
||||
"name": "Northern Northland"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 44,
|
||||
"fields": {
|
||||
"name": "Northern Volcanic Plateau"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 45,
|
||||
"fields": {
|
||||
"name": "Olivine"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 46,
|
||||
"fields": {
|
||||
"name": "Otago Coast"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 47,
|
||||
"fields": {
|
||||
"name": "Pahiatua"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 48,
|
||||
"fields": {
|
||||
"name": "Pareora"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 49,
|
||||
"fields": {
|
||||
"name": "Poor Knights"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 50,
|
||||
"fields": {
|
||||
"name": "Puketeraki"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 51,
|
||||
"fields": {
|
||||
"name": "Rakiura"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 52,
|
||||
"fields": {
|
||||
"name": "Rangitikei"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 53,
|
||||
"fields": {
|
||||
"name": "Raukumara"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 54,
|
||||
"fields": {
|
||||
"name": "Richmond"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 55,
|
||||
"fields": {
|
||||
"name": "Rodney"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 56,
|
||||
"fields": {
|
||||
"name": "Ruahine"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 57,
|
||||
"fields": {
|
||||
"name": "Sounds-Wellington"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 58,
|
||||
"fields": {
|
||||
"name": "Southland Foothills"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 59,
|
||||
"fields": {
|
||||
"name": "Spenser"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 60,
|
||||
"fields": {
|
||||
"name": "Tainui"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 61,
|
||||
"fields": {
|
||||
"name": "Taranaki"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 62,
|
||||
"fields": {
|
||||
"name": "Tararua"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 63,
|
||||
"fields": {
|
||||
"name": "Tasman"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 64,
|
||||
"fields": {
|
||||
"name": "Te Paki"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 65,
|
||||
"fields": {
|
||||
"name": "Te Wae Wae"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 66,
|
||||
"fields": {
|
||||
"name": "Three Kings"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 67,
|
||||
"fields": {
|
||||
"name": "Tongariro"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 68,
|
||||
"fields": {
|
||||
"name": "Urewera"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 69,
|
||||
"fields": {
|
||||
"name": "Waikaia"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 70,
|
||||
"fields": {
|
||||
"name": "Waikato"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 71,
|
||||
"fields": {
|
||||
"name": "Wainono"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 72,
|
||||
"fields": {
|
||||
"name": "Wairarapa Plains"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 73,
|
||||
"fields": {
|
||||
"name": "Wairau"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 74,
|
||||
"fields": {
|
||||
"name": "Wairoa"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 75,
|
||||
"fields": {
|
||||
"name": "Waitaki"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 76,
|
||||
"fields": {
|
||||
"name": "Western Northland"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 77,
|
||||
"fields": {
|
||||
"name": "Western Volcanic Plateau"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 78,
|
||||
"fields": {
|
||||
"name": "Whataroa"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.ecologicalregion",
|
||||
"pk": 79,
|
||||
"fields": {
|
||||
"name": "Whatkatane"
|
||||
}
|
||||
}
|
||||
]
|
170
backend/right_tree/api/data/fixtures/soil_order_mappings.json
Normal file
170
backend/right_tree/api/data/fixtures/soil_order_mappings.json
Normal file
|
@ -0,0 +1,170 @@
|
|||
[
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 1,
|
||||
"fields": {
|
||||
"code": "A",
|
||||
"name": "Anthropic"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 2,
|
||||
"fields": {
|
||||
"code": "B",
|
||||
"name": "Brown"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 3,
|
||||
"fields": {
|
||||
"code": "G",
|
||||
"name": "Gley"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 4,
|
||||
"fields": {
|
||||
"code": "L",
|
||||
"name": "Allophanic"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 5,
|
||||
"fields": {
|
||||
"code": "N",
|
||||
"name": "Granular"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 6,
|
||||
"fields": {
|
||||
"code": "E",
|
||||
"name": "Melanic"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 7,
|
||||
"fields": {
|
||||
"code": "O",
|
||||
"name": "Organic"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 8,
|
||||
"fields": {
|
||||
"code": "X",
|
||||
"name": "Oxidic"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 9,
|
||||
"fields": {
|
||||
"code": "P",
|
||||
"name": "Pallic"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 10,
|
||||
"fields": {
|
||||
"code": "Z",
|
||||
"name": "Podzol"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 11,
|
||||
"fields": {
|
||||
"code": "M",
|
||||
"name": "Pumice"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 12,
|
||||
"fields": {
|
||||
"code": "W",
|
||||
"name": "Raw"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 13,
|
||||
"fields": {
|
||||
"code": "R",
|
||||
"name": "Recent"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 14,
|
||||
"fields": {
|
||||
"code": "S",
|
||||
"name": "Semi"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 15,
|
||||
"fields": {
|
||||
"code": "U",
|
||||
"name": "Ultic"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 16,
|
||||
"fields": {
|
||||
"code": "i",
|
||||
"name": "Ice"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 17,
|
||||
"fields": {
|
||||
"code": "t",
|
||||
"name": "Town"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 18,
|
||||
"fields": {
|
||||
"code": "r",
|
||||
"name": "River"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 19,
|
||||
"fields": {
|
||||
"code": "e",
|
||||
"name": "Estu"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 20,
|
||||
"fields": {
|
||||
"code": "l",
|
||||
"name": "Lake"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilorder",
|
||||
"pk": 21,
|
||||
"fields": {
|
||||
"code": "q",
|
||||
"name": "Quar"
|
||||
}
|
||||
}
|
||||
]
|
23
backend/right_tree/api/data/fixtures/soil_variants.json
Normal file
23
backend/right_tree/api/data/fixtures/soil_variants.json
Normal file
|
@ -0,0 +1,23 @@
|
|||
[
|
||||
{
|
||||
"model": "api.soilvariant",
|
||||
"pk": 1,
|
||||
"fields": {
|
||||
"name": "Wet"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilvariant",
|
||||
"pk": 2,
|
||||
"fields": {
|
||||
"name": "Mesic"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.soilvariant",
|
||||
"pk": 3,
|
||||
"fields": {
|
||||
"name": "Dry"
|
||||
}
|
||||
}
|
||||
]
|
23
backend/right_tree/api/data/fixtures/tolerance_levels.json
Normal file
23
backend/right_tree/api/data/fixtures/tolerance_levels.json
Normal file
|
@ -0,0 +1,23 @@
|
|||
[
|
||||
{
|
||||
"model": "api.tolerancelevel",
|
||||
"pk": 1,
|
||||
"fields": {
|
||||
"level": "M"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.tolerancelevel",
|
||||
"pk": 2,
|
||||
"fields": {
|
||||
"level": "H"
|
||||
}
|
||||
},
|
||||
{
|
||||
"model": "api.tolerancelevel",
|
||||
"pk": 3,
|
||||
"fields": {
|
||||
"level": "L"
|
||||
}
|
||||
}
|
||||
]
|
0
backend/right_tree/api/management/__init__.py
Normal file
0
backend/right_tree/api/management/__init__.py
Normal file
|
@ -0,0 +1,51 @@
|
|||
from openpyxl import load_workbook
|
||||
|
||||
|
||||
def get_pk_mapping(object, mapping_key="name"):
|
||||
""" Returns a dictionary mapping a django model primary key to another given field.
|
||||
"""
|
||||
pk_mapping = {}
|
||||
for instance in object.objects.all():
|
||||
pk_mapping[getattr(instance, mapping_key)] = instance.pk
|
||||
|
||||
return pk_mapping
|
||||
|
||||
|
||||
def get_col_mappings(sheet, start_col, row_index):
|
||||
""" Returns a dictionary that maps a spreadsheet cell value to a corresponding column index.
|
||||
"""
|
||||
col_mappings = {}
|
||||
for row in sheet.iter_rows(min_col=start_col, min_row=row_index, max_row=row_index, values_only=True):
|
||||
for i, col_name in enumerate(row):
|
||||
col_mappings[col_name] = i
|
||||
|
||||
return col_mappings
|
||||
|
||||
|
||||
def get_pk_list_from_str(values_str, pk_mapping, fixes={}):
|
||||
""" Given a list of comma separated values from the spreadsheet. Returns a list of primary keys that
|
||||
correspond to the relevant values with any given mapping fixes applied.
|
||||
"""
|
||||
pk_list = []
|
||||
for value in values_str.split(','):
|
||||
processed_value = value.lstrip().rstrip().replace(
|
||||
'_', ' ').replace('-', ' ').replace('’', '\'')
|
||||
|
||||
# Applies any mapping adjustments between spreadsheet data and the database values
|
||||
if fixes and processed_value in fixes:
|
||||
processed_value = fixes[processed_value]
|
||||
|
||||
# Adds the pk value for the value in the databse
|
||||
if processed_value in pk_mapping:
|
||||
pk_list.append(pk_mapping[processed_value])
|
||||
|
||||
return pk_list
|
||||
|
||||
|
||||
def get_spreadsheet(data_path, spreadsheet_filename):
|
||||
""" Returns a spreadsheet from a resources directory given the data path and
|
||||
spreadsheet filename.
|
||||
"""
|
||||
spreadsheet_path = data_path / 'resources' / spreadsheet_filename
|
||||
workbook = load_workbook(filename=spreadsheet_path)
|
||||
return workbook.active
|
|
@ -0,0 +1,181 @@
|
|||
from django.core.management.base import BaseCommand
|
||||
|
||||
import json
|
||||
from pathlib import Path
|
||||
|
||||
import right_tree.api.data
|
||||
from ._spreadsheet_helpers import *
|
||||
from right_tree.api.models import EcologicalRegion, SoilOrder, SoilVariant, ToleranceLevel
|
||||
|
||||
# Mapping adjustments between the shapefile ecological regions and those in the spreadsheet
|
||||
ECO_REGION_ADJUSTMENTS = {
|
||||
"Whakatane": "Whatkatane",
|
||||
"North West Nelson": "North-west Nelson",
|
||||
"Aorangi": "Aorrangi",
|
||||
"Mackenzie": "MacKenzie",
|
||||
"Southland Hills": "Southland Foothills",
|
||||
"Sounds Wellington": "Sounds-Wellington"
|
||||
}
|
||||
|
||||
# Relevant columns and information used to retrieve information from the spreadsheet
|
||||
PLANT_COLS = {
|
||||
'name': {"expected_type": str, "max_length": 50},
|
||||
'maxheight': {"expected_type": float},
|
||||
'spacing': {"expected_type": float},
|
||||
'commonname': {"expected_type": str, "null_allowed": True, "max_length": 50},
|
||||
'synonym': {"expected_type": str, "null_allowed": True, "max_length": 200},
|
||||
'region': {"expected_type": list, "model_name": "ecological_regions"},
|
||||
'soilorder': {"expected_type": list, "model_name": "soil_order"},
|
||||
'wet': {"expected_type": list, "model_name": "soil_variants"},
|
||||
'mesic': {"expected_type": list, "model_name": "soil_variants"},
|
||||
'dry': {"expected_type": list, "model_name": "soil_variants"},
|
||||
'water': {"expected_type": int, "model_name": "water_tolerance"},
|
||||
'drought': {"expected_type": int, "model_name": "drought_tolerance"},
|
||||
'frost': {"expected_type": int, "model_name": "frost_tolerance"},
|
||||
'salinity': {"expected_type": int, "model_name": "salinity_tolerance"},
|
||||
'purpose': {"expected_type": str, "null_allowed": True},
|
||||
'stage': {"expected_type": int},
|
||||
'growthform': {"expected_type": str, "model_name": "growth_form", "null_allowed": True, "max_length": 50}
|
||||
}
|
||||
|
||||
# Spreadsheet constants
|
||||
SPREADSHEET_FILENAME = 'plant_data.xlsx'
|
||||
DATA_START_COL = 3
|
||||
DATA_START_ROW = 7
|
||||
INFO_HEADER_ROW = 6
|
||||
|
||||
# Data directory path
|
||||
DATA_DIR_PATH = Path(right_tree.api.data.__file__).resolve().parent
|
||||
|
||||
# Mappings between values in the spreadsheet and primary key values in the database
|
||||
ECO_REGION_PK_MAPPING = get_pk_mapping(EcologicalRegion)
|
||||
SOIL_ORDER_PK_MAPPING = get_pk_mapping(SoilOrder)
|
||||
SOIL_VARIANT_PK_MAPPING = get_pk_mapping(SoilVariant)
|
||||
TOLERANCE_PK_MAPPING = get_pk_mapping(ToleranceLevel, "level")
|
||||
|
||||
# Spreadsheet and corresponding value to column index mappings
|
||||
SPREADSHEET = get_spreadsheet(DATA_DIR_PATH, SPREADSHEET_FILENAME)
|
||||
INFO_COL_INDEXES = get_col_mappings(
|
||||
SPREADSHEET, DATA_START_COL, INFO_HEADER_ROW)
|
||||
|
||||
# Template for the plant json to add as an entry for the fixtures
|
||||
PLANT_JSON_TEMPLATE = {
|
||||
"model": "api.plant",
|
||||
"pk": None,
|
||||
"fields": {}
|
||||
}
|
||||
|
||||
|
||||
def check_field_type(field, field_value):
|
||||
""" Checks the validity of a feild value collected from the spreadsheet
|
||||
"""
|
||||
expected_field_type = PLANT_COLS[field]['expected_type']
|
||||
model_field_name = PLANT_COLS[field].get('model_name', field)
|
||||
null_allowed = PLANT_COLS[field].get('null_allowed', False)
|
||||
max_length = PLANT_COLS[field].get('max_length', False)
|
||||
|
||||
is_valid_type = isinstance(field_value, expected_field_type)
|
||||
is_int_when_float = expected_field_type == float and isinstance(
|
||||
field_value, int)
|
||||
is_valid_null = field_value is None and null_allowed
|
||||
is_over_max_length = max_length and isinstance(
|
||||
field_value, str) and len(field_value) > max_length
|
||||
|
||||
if not(is_valid_type or is_int_when_float or is_valid_null):
|
||||
raise TypeError(
|
||||
f"Invalid json type for field {model_field_name} with value {field_value}. Expected '{expected_field_type}' but got '{type(field_value)}'.")
|
||||
elif is_over_max_length:
|
||||
raise TypeError(
|
||||
f"Invalid string length for {model_field_name} with value {field_value}. Expected length to be under {max_length} but was {len(field_value)}.")
|
||||
|
||||
|
||||
def get_plant_json_from_row(row_data):
|
||||
""" Returns a json object representing a plant row from the spreadsheet.
|
||||
"""
|
||||
plant_json_fields = {}
|
||||
for field, field_index in INFO_COL_INDEXES.items():
|
||||
if field not in PLANT_COLS:
|
||||
continue
|
||||
|
||||
model_field_name = PLANT_COLS[field].get('model_name', field)
|
||||
try:
|
||||
if field == "region":
|
||||
regions_list = get_pk_list_from_str(
|
||||
row_data[field_index], ECO_REGION_PK_MAPPING, ECO_REGION_ADJUSTMENTS)
|
||||
plant_json_fields[model_field_name] = regions_list
|
||||
|
||||
elif field == "soilorder":
|
||||
soil_orders_list = get_pk_list_from_str(
|
||||
row_data[field_index], SOIL_ORDER_PK_MAPPING)
|
||||
plant_json_fields[model_field_name] = soil_orders_list
|
||||
|
||||
elif field in {'wet', 'mesic', 'dry'}:
|
||||
soil_variant_pk = SOIL_VARIANT_PK_MAPPING[field.capitalize()]
|
||||
plant_json_fields[model_field_name] = plant_json_fields.get(
|
||||
model_field_name, []) + [soil_variant_pk]
|
||||
|
||||
elif field in {'water', 'drought', 'frost', 'salinity'}:
|
||||
plant_json_fields[model_field_name] = TOLERANCE_PK_MAPPING[row_data[field_index]]
|
||||
|
||||
elif field in PLANT_COLS:
|
||||
plant_json_fields[model_field_name] = row_data[field_index]
|
||||
|
||||
check_field_type(field, plant_json_fields[model_field_name])
|
||||
|
||||
except Exception as e:
|
||||
name_index = INFO_COL_INDEXES['name']
|
||||
print(
|
||||
f"Error occured while adding the row for {row_data[name_index]}.")
|
||||
print(F"{type(e)}: {e}")
|
||||
print("SKIPPING ROW...")
|
||||
print("----------------------------------------------")
|
||||
|
||||
return {}
|
||||
|
||||
plant_json = PLANT_JSON_TEMPLATE.copy()
|
||||
plant_json["fields"] = plant_json_fields
|
||||
|
||||
return plant_json
|
||||
|
||||
|
||||
def get_plant_json_fixture(sheet):
|
||||
""" Returns a django fixture json that represents the plant information extracted from the spreadsheet.
|
||||
"""
|
||||
plant_json_fixture = []
|
||||
skipped_count = 0
|
||||
created_count = 0
|
||||
|
||||
for row in sheet.iter_rows(min_col=DATA_START_COL, min_row=DATA_START_ROW, values_only=True):
|
||||
plant_json = get_plant_json_from_row(row)
|
||||
|
||||
# If there is invalid data in a row, it will be skipped
|
||||
if plant_json != {}:
|
||||
plant_json_fixture.append(plant_json)
|
||||
created_count += 1
|
||||
else:
|
||||
skipped_count += 1
|
||||
|
||||
# Print summary of data extraction from the spreadsheet
|
||||
print("Created plants fixture.")
|
||||
print(f"Rows Created: {created_count}")
|
||||
print(f"Rows Skipped: {skipped_count}")
|
||||
|
||||
return plant_json_fixture
|
||||
|
||||
|
||||
def save_plant_fixture(fixture):
|
||||
""" Saves the plant fixture to the django api fixtures directory.
|
||||
"""
|
||||
fixture_filepath = DATA_DIR_PATH / 'fixtures' / 'plants.json'
|
||||
fixture_filepath.write_text(json.dumps(fixture))
|
||||
|
||||
|
||||
class Command(BaseCommand):
|
||||
help = 'Ingests the plant spreadsheet data into the database'
|
||||
|
||||
def handle(self, *args, **options):
|
||||
self.stdout.write('Creating plant fixtures...')
|
||||
plant_fixture = get_plant_json_fixture(SPREADSHEET)
|
||||
save_plant_fixture(plant_fixture)
|
||||
self.stdout.write(self.style.SUCCESS(
|
||||
'Plant fixtures created and saved successfully.'))
|
44
backend/right_tree/api/management/commands/loadshapefiles.py
Normal file
44
backend/right_tree/api/management/commands/loadshapefiles.py
Normal file
|
@ -0,0 +1,44 @@
|
|||
from django.core.management.base import BaseCommand
|
||||
from django.contrib.gis.utils import LayerMapping
|
||||
|
||||
from pathlib import Path
|
||||
|
||||
import right_tree.api.data
|
||||
from right_tree.api.models import SoilLayer, EcologicalDistrictLayer
|
||||
|
||||
# Auto-generated `LayerMapping` dictionary for SoilLayers model
|
||||
soillayer_mapping = {
|
||||
'nzsc_class': 'nzsc_class',
|
||||
'nzsc_group': 'nzsc_group',
|
||||
'nzsc_order': {'code': 'nzsc_order'},
|
||||
'shape_leng': 'SHAPE_Leng',
|
||||
'geom': 'POLYGON',
|
||||
}
|
||||
|
||||
# Auto-generated `LayerMapping` dictionary for ecologicaldistrictlayer model
|
||||
ecologicaldistrictlayer_mapping = {
|
||||
'ecological': 'ECOLOGICAL',
|
||||
'ecologic_1': 'ECOLOGIC_1',
|
||||
'ecologic_2': {'name': 'ECOLOGIC_2'},
|
||||
'shape_leng': 'SHAPE_Leng',
|
||||
'shape_area': 'SHAPE_Area',
|
||||
'geom': 'POLYGON',
|
||||
}
|
||||
|
||||
# Shapefiles
|
||||
soillayer_shp = Path(right_tree.api.data.__file__).resolve().parent / 'resources' / 'fundamental_soil_layers' / 'fundamental-soil-layers-new-zealand-soil-classification.shp'
|
||||
ecologicaldistrictlayer_shp = Path(right_tree.api.data.__file__).resolve().parent / 'resources' / 'ecological_districts' / 'DOC_EcologicalDistricts_2021_08_02.shp'
|
||||
|
||||
class Command(BaseCommand):
|
||||
help = 'Ingests the shapefile data for ecological regions and soil layers.'
|
||||
|
||||
def handle(self, *args, **options):
|
||||
self.stdout.write('Loading soil layers...')
|
||||
soil_lm = LayerMapping(SoilLayer, soillayer_shp, soillayer_mapping, transform=False)
|
||||
soil_lm.save(strict=True)
|
||||
self.stdout.write(self.style.SUCCESS('Soil layers loaded succesfully.'))
|
||||
|
||||
self.stdout.write('Loading ecological district layers...')
|
||||
ecologicaldistrictlayer_lm = LayerMapping(EcologicalDistrictLayer, ecologicaldistrictlayer_shp, ecologicaldistrictlayer_mapping, transform=False)
|
||||
ecologicaldistrictlayer_lm.save(strict=True)
|
||||
self.stdout.write(self.style.SUCCESS('Ecological district layers loaded succesfully.'))
|
11
backend/right_tree/api/management/commands/resetplants.py
Normal file
11
backend/right_tree/api/management/commands/resetplants.py
Normal file
|
@ -0,0 +1,11 @@
|
|||
from django.core.management.base import BaseCommand
|
||||
from right_tree.api.models import Plant
|
||||
|
||||
|
||||
class Command(BaseCommand):
|
||||
help = 'Removes all plant objects from the database'
|
||||
|
||||
def handle(self, *args, **options):
|
||||
self.stdout.write(self.style.WARNING(
|
||||
'Removing all plant objects from the database.'))
|
||||
Plant.objects.all().delete()
|
|
@ -1,6 +1,8 @@
|
|||
# Generated by Django 3.2.8 on 2021-10-06 18:32
|
||||
# Generated by Django 3.2.8 on 2021-10-15 01:23
|
||||
|
||||
import django.contrib.gis.db.models.fields
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
@ -11,11 +13,77 @@ class Migration(migrations.Migration):
|
|||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='EcologicalRegion',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('name', models.CharField(max_length=50, unique=True)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='SoilOrder',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('code', models.CharField(max_length=1, unique=True)),
|
||||
('name', models.CharField(max_length=50, unique=True)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='SoilVariant',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('name', models.CharField(max_length=10, unique=True)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='ToleranceLevel',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('level', models.CharField(max_length=1)),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='SoilLayer',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('nzsc_class', models.CharField(max_length=4)),
|
||||
('nzsc_group', models.CharField(max_length=2)),
|
||||
('shape_leng', models.FloatField()),
|
||||
('geom', django.contrib.gis.db.models.fields.PolygonField(srid=2193)),
|
||||
('nzsc_order', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='api.soilorder')),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='Plant',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('name', models.TextField()),
|
||||
('name', models.CharField(max_length=50, unique=True)),
|
||||
('commonname', models.CharField(blank=True, max_length=50, null=True)),
|
||||
('maxheight', models.FloatField()),
|
||||
('spacing', models.FloatField()),
|
||||
('synonym', models.CharField(blank=True, max_length=200, null=True)),
|
||||
('purpose', models.TextField(blank=True, null=True)),
|
||||
('stage', models.PositiveIntegerField()),
|
||||
('growth_form', models.CharField(blank=True, max_length=50, null=True)),
|
||||
('drought_tolerance', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='drought_tolerance', to='api.tolerancelevel')),
|
||||
('ecological_regions', models.ManyToManyField(to='api.EcologicalRegion')),
|
||||
('frost_tolerance', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='frost_tolerance', to='api.tolerancelevel')),
|
||||
('salinity_tolerance', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='salinity_tolerance', to='api.tolerancelevel')),
|
||||
('soil_order', models.ManyToManyField(to='api.SoilOrder')),
|
||||
('soil_variants', models.ManyToManyField(to='api.SoilVariant')),
|
||||
('water_tolerance', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='water_tolerance', to='api.tolerancelevel')),
|
||||
],
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='EcologicalDistrictLayer',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('ecological', models.CharField(max_length=5)),
|
||||
('ecologic_1', models.CharField(max_length=50)),
|
||||
('shape_leng', models.FloatField()),
|
||||
('shape_area', models.FloatField()),
|
||||
('geom', django.contrib.gis.db.models.fields.PolygonField(srid=2193)),
|
||||
('ecologic_2', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='api.ecologicalregion')),
|
||||
],
|
||||
),
|
||||
]
|
||||
|
|
|
@ -1,4 +1,78 @@
|
|||
from django.db import models
|
||||
from django.contrib.gis.db import models
|
||||
|
||||
|
||||
class SoilOrder(models.Model):
|
||||
code = models.CharField(unique=True, max_length=1)
|
||||
name = models.CharField(unique=True, max_length=50)
|
||||
|
||||
def __str__(self):
|
||||
return f"{self.name} ({self.code})"
|
||||
|
||||
|
||||
class SoilVariant(models.Model):
|
||||
name = models.CharField(unique=True, max_length=10)
|
||||
|
||||
def __str__(self):
|
||||
return self.name
|
||||
|
||||
|
||||
class SoilLayer(models.Model):
|
||||
nzsc_class = models.CharField(max_length=4)
|
||||
nzsc_group = models.CharField(max_length=2)
|
||||
nzsc_order = models.ForeignKey(SoilOrder, on_delete=models.CASCADE)
|
||||
shape_leng = models.FloatField()
|
||||
geom = models.PolygonField(srid=2193)
|
||||
|
||||
def __str__(self):
|
||||
return self.nzsc_class
|
||||
|
||||
|
||||
class EcologicalRegion(models.Model):
|
||||
name = models.CharField(unique=True, max_length=50)
|
||||
|
||||
def __str__(self):
|
||||
return self.name
|
||||
|
||||
|
||||
class EcologicalDistrictLayer(models.Model):
|
||||
ecological = models.CharField(max_length=5)
|
||||
ecologic_1 = models.CharField(max_length=50)
|
||||
ecologic_2 = models.ForeignKey(EcologicalRegion, on_delete=models.CASCADE)
|
||||
shape_leng = models.FloatField()
|
||||
shape_area = models.FloatField()
|
||||
geom = models.PolygonField(srid=2193)
|
||||
|
||||
def __str__(self):
|
||||
return f"{self.ecologic_1} ({self.ecologic_2})"
|
||||
|
||||
|
||||
class ToleranceLevel(models.Model):
|
||||
level = models.CharField(max_length=1)
|
||||
|
||||
def __str__(self):
|
||||
return self.level
|
||||
|
||||
|
||||
class Plant(models.Model):
|
||||
name = models.TextField()
|
||||
name = models.CharField(unique=True, max_length=50)
|
||||
commonname = models.CharField(null=True, blank=True, max_length=50)
|
||||
maxheight = models.FloatField()
|
||||
spacing = models.FloatField()
|
||||
synonym = models.CharField(null=True, blank=True, max_length=200)
|
||||
water_tolerance = models.ForeignKey(
|
||||
ToleranceLevel, related_name='water_tolerance', on_delete=models.CASCADE)
|
||||
drought_tolerance = models.ForeignKey(
|
||||
ToleranceLevel, related_name='drought_tolerance', on_delete=models.CASCADE)
|
||||
frost_tolerance = models.ForeignKey(
|
||||
ToleranceLevel, related_name='frost_tolerance', on_delete=models.CASCADE)
|
||||
salinity_tolerance = models.ForeignKey(
|
||||
ToleranceLevel, related_name='salinity_tolerance', on_delete=models.CASCADE)
|
||||
purpose = models.TextField(null=True, blank=True)
|
||||
stage = models.PositiveIntegerField()
|
||||
growth_form = models.CharField(null=True, blank=True, max_length=50)
|
||||
ecological_regions = models.ManyToManyField(EcologicalRegion)
|
||||
soil_order = models.ManyToManyField(SoilOrder)
|
||||
soil_variants = models.ManyToManyField(SoilVariant)
|
||||
|
||||
def __str__(self):
|
||||
return self.name
|
||||
|
|
|
@ -37,6 +37,7 @@ INSTALLED_APPS = [
|
|||
'django.contrib.sessions',
|
||||
'django.contrib.messages',
|
||||
'django.contrib.staticfiles',
|
||||
'django.contrib.gis',
|
||||
|
||||
'rest_framework',
|
||||
'corsheaders',
|
||||
|
@ -81,7 +82,7 @@ WSGI_APPLICATION = 'right_tree.wsgi.application'
|
|||
|
||||
DATABASES = {
|
||||
'default': {
|
||||
'ENGINE': 'django.db.backends.postgresql',
|
||||
'ENGINE': 'django.contrib.gis.db.backends.postgis',
|
||||
'NAME': 'right_tree',
|
||||
'USER': 'postgres',
|
||||
'PASSWORD': 'postgres',
|
||||
|
|
|
@ -1,3 +0,0 @@
|
|||
docker-compose down --remove-orphans --volumes
|
||||
docker-compose up postgres | sed '/PostgreSQL init process complete; ready for start up./q'
|
||||
docker-compose down
|
89
dev
Executable file
89
dev
Executable file
|
@ -0,0 +1,89 @@
|
|||
#!/bin/bash
|
||||
|
||||
cmd_create_database() {
|
||||
echo "Creating right_tree database..."
|
||||
docker-compose down --remove-orphans --volumes
|
||||
docker-compose -f docker-compose.yaml up postgres | sed '/PostgreSQL init process complete; ready for start up./q'
|
||||
docker-compose down
|
||||
}
|
||||
|
||||
cmd_makemigrations() {
|
||||
echo "Creating database migrations..."
|
||||
docker-compose exec django-backend python manage.py makemigrations --no-input
|
||||
}
|
||||
|
||||
cmd_migrate() {
|
||||
echo "Running database migrations..."
|
||||
docker-compose exec django-backend python manage.py migrate
|
||||
}
|
||||
|
||||
cmd_createsuperuser() {
|
||||
echo "Loading shapefiles into the database..."
|
||||
docker-compose exec django-backend python manage.py createsuperuser --noinput
|
||||
}
|
||||
|
||||
cmd_load_fixtures() {
|
||||
echo "Loading fixtures..."
|
||||
docker-compose exec django-backend bash -c "python manage.py loaddata right_tree/api/data/fixtures/*.json"
|
||||
}
|
||||
|
||||
cmd_load_shapefiles() {
|
||||
echo "Loading shapefiles into the database..."
|
||||
docker-compose exec django-backend python manage.py loadshapefiles
|
||||
}
|
||||
|
||||
cmd_create_plant_fixtures() {
|
||||
echo "Creates fixtures for plants using spreadsheet."
|
||||
docker-compose exec django-backend python manage.py createplantfixtures
|
||||
}
|
||||
|
||||
cmd_reset_plants() {
|
||||
echo "Resetting plants..."
|
||||
docker-compose exec django-backend python manage.py resetplants
|
||||
}
|
||||
|
||||
cmd_load_plant_fixtures() {
|
||||
echo "Loading plants..."
|
||||
docker-compose exec django-backend python manage.py loaddata right_tree/api/data/fixtures/plants.json
|
||||
}
|
||||
|
||||
cmd_load_plants() {
|
||||
cmd_create_plant_fixtures
|
||||
cmd_reset_plants
|
||||
cmd_load_plant_fixtures
|
||||
}
|
||||
|
||||
cmd_populate_database() {
|
||||
echo "Populating the database..."
|
||||
docker-compose up -d django-backend postgres
|
||||
|
||||
cmd_makemigrations
|
||||
cmd_migrate
|
||||
cmd_createsuperuser
|
||||
cmd_load_fixtures
|
||||
cmd_load_shapefiles
|
||||
cmd_load_plants
|
||||
|
||||
docker-compose down
|
||||
}
|
||||
|
||||
cmd_init_database() {
|
||||
cmd_create_database
|
||||
cmd_populate_database
|
||||
}
|
||||
|
||||
cmd_reset_database() {
|
||||
cmd_init_database
|
||||
}
|
||||
|
||||
cmd_build() {
|
||||
docker-compose build
|
||||
}
|
||||
|
||||
cmd_start() {
|
||||
docker-compose up
|
||||
}
|
||||
|
||||
# Run the command
|
||||
cmd="$1"
|
||||
"cmd_$cmd" "$@"
|
|
@ -17,10 +17,7 @@ services:
|
|||
- ./backend:/app
|
||||
ports:
|
||||
- "8000:8000"
|
||||
command: bash -c "./manage.py makemigrations;
|
||||
./manage.py migrate;
|
||||
./manage.py createsuperuser --noinput;
|
||||
./manage.py runserver 0.0.0.0:8000"
|
||||
command: bash -c "./manage.py runserver 0.0.0.0:8000"
|
||||
|
||||
react-frontend:
|
||||
image: node:16-alpine3.11
|
||||
|
@ -39,7 +36,7 @@ services:
|
|||
container_name: postgres
|
||||
volumes:
|
||||
- local-postgres-data:/var/lib/postgresql/data
|
||||
- ./database/init:/docker-entrypoint-initdb.d
|
||||
- ./create_database.sql:/docker-entrypoint-initdb.d/create_database.sql
|
||||
ports:
|
||||
- "5432:5432"
|
||||
environment:
|
||||
|
|
Loading…
Reference in a new issue