Skip to content

FastGeosuitability is a PostGIS geospatial api to perform suitability analysis with geographical tables within a spatial database.

License

Notifications You must be signed in to change notification settings

mkeller3/FastGeosuitability

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

24 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

FastGeosuitability

FastGeosuitability is a geosuitability api to help determine what areas are most suitable based off of a given set of variables. The api will rank each location with a set of weighted and final score to help you determine the best location. FastGeosuitability is written in Python using the FastAPI web framework.

Built with inspiration from ArcGIS Business Analyst.


Source Code: https://github.com/mkeller3/FastGeosuitability


Requirements

FastGeosuitability requires PostGIS >= 2.4.0.

Configuration

In order for the api to work you will need to edit the .env with your database connections.

DB_HOST=localhost
DB_DATABASE=data
DB_USERNAME=postgres
DB_PASSWORD=postgres
DB_PORT=5432

Usage

Running Locally

To run the app locally uvicorn main:app --reload

Production

Build Dockerfile into a docker image to deploy to the cloud.

API

Method URL Description
POST /api/v1/services/map_suitability/ Map Suitability
POST /api/v1/services/point_suitability/ Point Suitability
POST /api/v1/services/polygon_suitability/ Polygon Suitability
POST /api/v1/services/line_suitability/ Line Suitability
GET /api/v1/health_check Server health check: returns 200 OK

Map Suitability

The map suitability endpoints allows you to perform a suitability analysis on a set of geometries already loaded into your database.

Map Suitability Parameters

  • table=table - name of the table.
  • table_column=table-col - name of a unique column in your table.
  • table_values=[values] - list of unique values from your table column.
  • variables=[{variables}] - list of variables to perform site suitability against.
  • return_geometry=bool - boolean to determine if geometry is returned.
  • filter=cql-expr - filters features via a CQL expression.

Map Suitability Example

In the example below, I am trying to rank all the counties in Illinois based off of 3 variables. In this situation, I am trying to find the best county to live in if I wanted to be by a large amount of Walmart's, Chick Fil A's, and Starbucks. My table parameter is counties with my table_column set to fips, since each county has a unique fips code. I then listed all of the fips code in the table_values parameter, but could also add a filter parameter of state_name = 'Illinois'. Next, I added in my list of variables for Walmart's, Chick Fil A's, and Starbucks.

"variables":[
      {
          "table": "walmart_locations",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 25
      },
      {
          "table": "chick_fil_a_locations",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 25
      },
      {
          "table": "starbucks",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 50
      }
  ]

For each variable, you will need to define an object containing a certain list of keys and values. For more information about defining variables, go to the variables section which goes into full detail.

Map Suitability Results

Your results will be returned as a geojson collection. For each polygon, you will see a new set of properties returned. For more information about results, go to the results section which goes into full detail.

For this example, Cook County would be the best county for me to live in, if I wanted to be near a large amount of Walmart's, Chick Fil A's, and Starbucks with a final score of 100.

Map Suitability Example Input

{
    "table": "counties",
    "table_column": "fips",
    "table_values": [
        "17077",
        "17043",
        "17115",
        "17097",
        "17099",
        "17111",
        "17113",
        "17001",
        "17007",
        "17019",
        "17029",
        "17031",
        ...
    ],
    "return_geometry": true,
    "variables":[
        {
            "table": "walmart_locations",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 25
        },
        {
            "table": "chick_fil_a_locations",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 25
        },
        {
            "table": "starbucks",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 50
        }
    ]
}

Map Suitability Example Response

{
    "type": "FeatureCollection",
    "features": [
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "fips": "17031",
                "walmart_locations_count_gid": 42,
                "chick_fil_a_locations_count_gid": 8,
                "starbucks_count_gid": 312,
                "weighted_score_walmart_locations_count_gid": 25.0,
                "score_walmart_locations_count_gid": 1.0,
                "weighted_score_chick_fil_a_locations_count_gid": 25.0,
                "score_chick_fil_a_locations_count_gid": 1.0,
                "weighted_score_starbucks_count_gid": 50.0,
                "score_starbucks_count_gid": 1.0,
                "final_score": 100.0
            }
        },
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "fips": "17043",
                "walmart_locations_count_gid": 8,
                "chick_fil_a_locations_count_gid": 7,
                "starbucks_count_gid": 69,
                "weighted_score_walmart_locations_count_gid": 4.2682926829268295,
                "score_walmart_locations_count_gid": 0.17073170731707318,
                "weighted_score_chick_fil_a_locations_count_gid": 21.428571428571427,
                "score_chick_fil_a_locations_count_gid": 0.8571428571428571,
                "weighted_score_starbucks_count_gid": 10.932475884244374,
                "score_starbucks_count_gid": 0.21864951768488747,
                "final_score": 36.62933999574263
            }
        },
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "fips": "17097",
                "walmart_locations_count_gid": 9,
                "chick_fil_a_locations_count_gid": 2,
                "starbucks_count_gid": 41,
                "weighted_score_walmart_locations_count_gid": 4.878048780487805,
                "score_walmart_locations_count_gid": 0.1951219512195122,
                "weighted_score_chick_fil_a_locations_count_gid": 3.571428571428571,
                "score_chick_fil_a_locations_count_gid": 0.14285714285714285,
                "weighted_score_starbucks_count_gid": 6.430868167202572,
                "score_starbucks_count_gid": 0.12861736334405144,
                "final_score": 14.880345519118947
            }
        }...
    ]
}

Map Suitability Map

map_suitability

Point Suitability

The point suitability endpoints allows you to perform a suitability analysis on a set of known latitude and longitudes.

Point Suitability Parameters

  • points=[points] - list of points to perform suitability analysis against.
  • buffer_in_kilometers=distance-in-kilometers - distance in kilometers to search for data around each point.
  • variables=[{variables}] - list of variables to perform site suitability against.
  • return_geometry=bool - boolean to determine if geometry is returned.

Point Suitability Example

In the example below, I am trying to rank a series of larger cities within the Midwest based off of 3 variables like the example before. In this situation, I am trying to find the best city to live in if I wanted to be by a large amount of Walmart's, Chick Fil A's, and Starbucks. Next, I added in my list of variables for Walmart's, Chick Fil A's, and Starbucks.

"variables":[
      {
          "table": "walmart_locations",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 25
      },
      {
          "table": "chick_fil_a_locations",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 25
      },
      {
          "table": "starbucks",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 50
      }
  ]

For each variable, you will need to define an object containing a certain list of keys and values. For more information about defining variables, go to the variables section which goes into full detail.

Point Suitability Results

Your results will be returned as a geojson collection. For each polygon, you will see a new set of properties returned. For more information about results, go to the results section which goes into full detail.

For this example, Chicago would be the best city for me to live in, if I wanted to be near a large amount of Walmart's, Chick Fil A's, and Starbucks with a final score of 92.

Point Suitability Example Input

{
    "points": [
        {
            "latitude": 40.45,
            "longitude": -88.95
        },
        {
            "latitude": 41.63212,
            "longitude": -87.85594
        }
    ],
    "buffer_in_kilometers": 50,
    "return_geometry": false,
    "variables":[
        {
            "table": "walmart_locations",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 50
        },
        {
            "table": "chick_fil_a_locations",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 50
        }
    ]
}

Point Suitability Example Response

{
    "type": "FeatureCollection",
    "features": [
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "longitude": -87.85594,
                "latitude": 41.63212,
                "walmart_locations_count_gid": 42,
                "chick_fil_a_locations_count_gid": 12,
                "weighted_score_walmart_locations_count_gid": 50.0,
                "score_walmart_locations_count_gid": 1.0,
                "weighted_score_chick_fil_a_locations_count_gid": 50.0,
                "score_chick_fil_a_locations_count_gid": 1.0,
                "final_score": 100.0
            }
        },
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "longitude": -88.95,
                "latitude": 40.45,
                "walmart_locations_count_gid": 3,
                "chick_fil_a_locations_count_gid": 2,
                "weighted_score_walmart_locations_count_gid": 0,
                "score_walmart_locations_count_gid": 0,
                "weighted_score_chick_fil_a_locations_count_gid": 0,
                "score_chick_fil_a_locations_count_gid": 0,
                "final_score": 0
            }
        }
    ]
}

Point Suitability Map

point_suitability

Polygon Suitability

The polygon suitability endpoints allows you to perform a suitability analysis on a set of known geojson polygons.

Polygon Suitability Parameters

  • geojson_collection=geojson_collection - a geojson collection of polygons.
  • variables=[{variables}] - list of variables to perform site suitability against.
  • return_geometry=bool - boolean to determine if geometry is returned.

Polygon Suitability Example

In the example below, I am trying to rank a series of larger urban areas around Chicago based off of 3 variables like the example before. In this situation, I am trying to find the best suburb to live in if I wanted to be by a large amount of Walmart's, Chick Fil A's, and Starbucks. Next, I added in my list of variables for Walmart's, Chick Fil A's, and Starbucks.

"variables":[
      {
          "table": "walmart_locations",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 25
      },
      {
          "table": "chick_fil_a_locations",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 25
      },
      {
          "table": "starbucks",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 50
      }
  ]

For each variable, you will need to define an object containing a certain list of keys and values. For more information about defining variables, go to the variables section which goes into full detail.

Polygon Suitability Results

Your results will be returned as a geojson collection. For each polygon, you will see a new set of properties returned. For more information about results, go to the results section which goes into full detail.

For this example, Lombard, IL would be the best city for me to live in, if I wanted to be near a large amount of Walmart's, Chick Fil A's, and Starbucks with a final score of 87.5.

Polygon Suitability Example Input

{
    "geojson_collection": {
        "type": "FeatureCollection",
        "features": [
            {
                "type": "Feature",
                "properties": {},
                "geometry": {
                    "type": "Polygon",
                    "coordinates": [
                        [
                            [
                                -88.11447143554688,
                                41.91147545749747
                            ],
                            [
                                -88.10623168945312,
                                41.85421933478601
                            ],
                            [
                                -88.06777954101562,
                                41.80919639152055
                            ],
                            [
                                -87.94830322265625,
                                41.83682786072714
                            ],
                            [
                                -87.93045043945311,
                                41.89512180073503
                            ],
                            [
                                -88.01696777343749,
                                41.92578147109541
                            ],
                            [
                                -88.11447143554688,
                                41.91147545749747
                            ]
                        ]
                    ]
                }
            },
            {
                "type": "Feature",
                "properties": {},
                "geometry": {
                    "type": "Polygon",
                    "coordinates": [
                        [
                            [
                                -87.9345703125,
                                42.06560675405716
                            ],
                            [
                                -88.01422119140625,
                                42.06356771883277
                            ],
                            [
                                -87.92633056640625,
                                42.02787400232195
                            ],
                            [
                                -87.84530639648436,
                                42.01563154037739
                            ],
                            [
                                -87.82333374023438,
                                42.02991418347818
                            ],
                            [
                                -87.84393310546875,
                                42.07478160216737
                            ],
                            [
                                -87.9345703125,
                                42.06560675405716
                            ]
                        ]
                    ]
                }
            },...            
        ]
    },
    "return_geometry": false,
    "variables": [
        {
            "table": "walmart_locations",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 50
        },
        {
            "table": "chick_fil_a_locations",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 50
        }
    ]
}

Polygon Suitability Example Response

{
    "type": "FeatureCollection",
    "features": [
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "index": 0,
                "walmart_locations_count_gid": 2,
                "chick_fil_a_locations_count_gid": 3,
                "starbucks_count_gid": 21,
                "weighted_score_walmart_locations_count_gid": 12.5,
                "score_walmart_locations_count_gid": 0.5,
                "weighted_score_chick_fil_a_locations_count_gid": 25.0,
                "score_chick_fil_a_locations_count_gid": 1.0,
                "weighted_score_starbucks_count_gid": 50.0,
                "score_starbucks_count_gid": 1.0,
                "final_score": 87.5
            }
        },
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "index": 2,
                "walmart_locations_count_gid": 3,
                "chick_fil_a_locations_count_gid": 1,
                "starbucks_count_gid": 18,
                "weighted_score_walmart_locations_count_gid": 25.0,
                "score_walmart_locations_count_gid": 1.0,
                "weighted_score_chick_fil_a_locations_count_gid": 8.333333333333332,
                "score_chick_fil_a_locations_count_gid": 0.3333333333333333,
                "weighted_score_starbucks_count_gid": 42.10526315789473,
                "score_starbucks_count_gid": 0.8421052631578947,
                "final_score": 75.43859649122805
            }
        },
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "index": 4,
                "walmart_locations_count_gid": 2,
                "chick_fil_a_locations_count_gid": 0,
                "starbucks_count_gid": 2,
                "weighted_score_walmart_locations_count_gid": 12.5,
                "score_walmart_locations_count_gid": 0.5,
                "weighted_score_chick_fil_a_locations_count_gid": 0,
                "score_chick_fil_a_locations_count_gid": 0,
                "weighted_score_starbucks_count_gid": 0,
                "score_starbucks_count_gid": 0,
                "final_score": 12.5
            }
        },
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "index": 1,
                "walmart_locations_count_gid": 1,
                "chick_fil_a_locations_count_gid": 0,
                "starbucks_count_gid": 4,
                "weighted_score_walmart_locations_count_gid": 0,
                "score_walmart_locations_count_gid": 0,
                "weighted_score_chick_fil_a_locations_count_gid": 0,
                "score_chick_fil_a_locations_count_gid": 0,
                "weighted_score_starbucks_count_gid": 5.263157894736842,
                "score_starbucks_count_gid": 0.10526315789473684,
                "final_score": 5.263157894736842
            }
        },
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "index": 3,
                "walmart_locations_count_gid": 1,
                "chick_fil_a_locations_count_gid": 0,
                "starbucks_count_gid": 2,
                "weighted_score_walmart_locations_count_gid": 0,
                "score_walmart_locations_count_gid": 0,
                "weighted_score_chick_fil_a_locations_count_gid": 0,
                "score_chick_fil_a_locations_count_gid": 0,
                "weighted_score_starbucks_count_gid": 0,
                "score_starbucks_count_gid": 0,
                "final_score": 0
            }
        }
    ]
}

Polygon Suitability Map

polygon_suitability

Line Suitability

The line suitability endpoints allows you to perform a suitability analysis on a set of known geojson lines.

Line Suitability Parameters

  • geojson_collection=geojson_collection - a geojson collection of lines.
  • buffer_in_kilometers=distance-in-kilometers - distance in kilometers to search for data around each line.
  • variables=[{variables}] - list of variables to perform site suitability against.
  • return_geometry=bool - boolean to determine if geometry is returned.

Line Suitability Example

In the example below, I am trying to rank two different areas of I-55 in Illinois with a buffer of 20 kilometers based off of 3 variables like the example before. In this situation, I am trying to find which part of I-55 has the largest amount of Walmart's, Chick Fil A's, and Starbucks. Next, I added in my list of variables for Walmart's, Chick Fil A's, and Starbucks.

"variables":[
      {
          "table": "walmart_locations",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 25
      },
      {
          "table": "chick_fil_a_locations",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 25
      },
      {
          "table": "starbucks",
          "column": "gid",
          "type": "count",
          "influence": "high",
          "weight": 50
      }
  ]

For each variable, you will need to define an object containing a certain list of keys and values. For more information about defining variables, go to the variables section which goes into full detail.

Line Suitability Results

Your results will be returned as a geojson collection. For each line, you will see a new set of properties returned. For more information about results, go to the results section which goes into full detail.

For this example, I-55 in Chicago is a better location for having a large amount of Walmart's, Chick Fil A's, and Starbucks with a final score of 100.

Line Suitability Example Input

{
    "geojson_collection": {
        "type": "FeatureCollection",
        "features": [
            {
                "type": "Feature",
                "properties": {},
                "geometry": {
                    "type": "LineString",
                    "coordinates": [
                        [
                            -89.6099853515625,
                            39.93922484079194
                        ],
                        [
                            -89.439697265625,
                            40.12009038025332
                        ],
                        [
                            -89.373779296875,
                            40.199854889057676
                        ],
                        [
                            -89.27490234375,
                            40.18726672309203
                        ],
                        [
                            -89.197998046875,
                            40.29628651711716
                        ],
                        [
                            -89.0716552734375,
                            40.421860362045194
                        ],
                        [
                            -89.0057373046875,
                            40.47202439692057
                        ]
                    ]
                }
            },
            {
                "type": "Feature",
                "properties": {},
                "geometry": {
                    "type": "LineString",
                    "coordinates": [
                        [
                            -88.187255859375,
                            41.50446357504803
                        ],
                        [
                            -88.165283203125,
                            41.63597302844412
                        ],
                        [
                            -88.0224609375,
                            41.73033005046653
                        ],
                        [
                            -87.890625,
                            41.77950486590359
                        ],
                        [
                            -87.791748046875,
                            41.80407814427234
                        ]
                    ]
                }
            }
        ]
    },
    "buffer_in_kilometers": 20,
    "return_geometry": true,
    "variables": [
        {
            "table": "walmart_locations",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 25
        },
        {
            "table": "chick_fil_a_locations",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 25
        },
        {
            "table": "starbucks",
            "column": "gid",
            "type": "count",
            "influence": "high",
            "weight": 50
        }
    ]
}

Line Suitability Example Response

{
    "type": "FeatureCollection",
    "features": [
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "index": 1,
                "walmart_locations_count_gid": 27,
                "chick_fil_a_locations_count_gid": 10,
                "starbucks_count_gid": 116,
                "weighted_score_walmart_locations_count_gid": 25.0,
                "score_walmart_locations_count_gid": 1.0,
                "weighted_score_chick_fil_a_locations_count_gid": 25.0,
                "score_chick_fil_a_locations_count_gid": 1.0,
                "weighted_score_starbucks_count_gid": 50.0,
                "score_starbucks_count_gid": 1.0,
                "final_score": 100.0
            }
        },
        {
            "type": "Feature",
            "geometry": null,
            "properties": {
                "index": 0,
                "walmart_locations_count_gid": 5,
                "chick_fil_a_locations_count_gid": 2,
                "starbucks_count_gid": 6,
                "weighted_score_walmart_locations_count_gid": 0,
                "score_walmart_locations_count_gid": 0,
                "weighted_score_chick_fil_a_locations_count_gid": 0,
                "score_chick_fil_a_locations_count_gid": 0,
                "weighted_score_starbucks_count_gid": 0,
                "score_starbucks_count_gid": 0,
                "final_score": 0
            }
        }
    ]
}

Line Suitability Map

line_suitability

Variables

In order to determine the suitability of each location, you will need to pass in a list of variables. For each variable, we will need a couple of details to help the api determine the suitability of the location.

Variable Parameters

  • table=table - name of the table.
  • column=table-col - name of a column in your table to perform suitability analysis with.
  • type=type - How to determine suitability score of variable. For more information about types go to the type descriptions area.
  • influence=influence - The type of influence to apply to the variable. Options low, high, ideal. For more information about influence type go to the influence descriptions area.
  • weight=weight - how much weight to apply to the variable. All variables must total up to 100.

Type Descriptions

Sum

Determine the sum all the values of the chosen column and table that intersect the suitability area. For polygons that do no intersect the entire suitability area, the api performs a percentage based sum. For example if the sum of the column for a polygon is 100, but only 80% of the polygon is within the suitability area. The api will account for the 20% loss of area and set the sum to 80. clip

Avg

Determine the average of all the values of the chosen column and table that intersect the suitability area. For polygons that do no intersect the entire suitability area, the api performs a percentage based average. For example if the average of the column for a polygon is 100, but only 80% of the polygon is within the suitability area. The api will account for the 20% loss of area and set the average to 80. clip

Count

Determine the number of features within a table that intersect the suitability area.

Influence Descriptions

Influence High

High Influence means that the higher the value of the type/column is, the higher the suitability score will be for that location.

Influence High Calculation
( 
    (Variable for location - Minimum variable for all locations) / 
    (Maximum variable for all locations - Minimum variable for all locations) 
) * weight
Influence High Calculation Example Python
weight = 20
location_variable = 3
minimum_variable = 1
maximum_variable = 5
score = ( 
    (location_variable - minimum_variable) /
    (maximum_variable - minimum_variable)
) * weight

print(score)
# score = 40

Influence Low

Low Influence means that the lower the value of the type/column is, the higher the suitability score will be for that location.

Influence Low Calculation
( 
    (Maximum variable for all locations - Variable for location) / 
    (Maximum variable for all locations - Minimum variable for all locations) 
) * weight
Influence Low Calculation Example Python
weight = 20
location_variable = 3
minimum_variable = 1
maximum_variable = 5
score = ( 
    (maximum_variable - location_variable) /
    (maximum_variable - minimum_variable)
) * weight

print(score)
# score = 20

Ideal

Ideal Influence means that the closer the value of the type/column is to the ideal value, the higher the suitability score will be for that location.

Influence Ideal Calculation
(
    1 -
    ( 
        (Absolute (Ideal Value - Variable for location) ) / 
        (Ideal Value  - Minimum variable for all locations OR Maximum variable for all locations - Variable for location) 
        {Whichever is of a higher value}
    )
) * weight
Influence Ideal Calculation Example Python
weight = 20
location_variable = 3
minimum_variable = 1
maximum_variable = 5
ideal_value = 4
score = (
    1 -
    ( 
        abs(ideal_value - location_variable) /
        (ideal_value - minimum_variable)
    )
) * weight

print(score)
# score = 13.33

Results

Every api endpoint returns a geojson collection of polygons. For each variable, you will see a score and weighted score property. The score is based off of the formulas above with no multiplication of the weight. The weighted scores are based off of the formulas noted above. There will also be a final_score property for each site. This rate's each site on a scale from 0 - 100. A site with a score of 100 means it is most suitable based off your variables you submitted to the api.

About

FastGeosuitability is a PostGIS geospatial api to perform suitability analysis with geographical tables within a spatial database.

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages