Skip to content

django-data-interrogator is a plug in table builder that allows users to easily extract information from a Django database

License

Notifications You must be signed in to change notification settings

Aristotle-Metadata-Enterprises/django-data-interrogator

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

django-data-interrogator is a plug in table builder that allows users to easily interrogate information from a django database. In essence it provides a smart and sane frontend for building tabular data from the django database queryset API - specifically values, filter, order_by and a handful of annotations.

Add the Data Interrogator to your INSTALLED_APPS:

INSTALLED_APPS = (
  # ...
  'data_interrogator',
)
  1. Make a list of base_models (models you wish to interrogate) and enter models into witness protection (models you want to disallow access to):
DATA_INTERROGATION_DOSSIER = {
    'base_models': [
        {'model':("yourapplabel","YourModelName")},
        {'model':("yourapplabel","YourOtherModelName")},
     ],
    'excluded_models' : ["User","Revision","Version"]
}

Notes: base_models are used to query the django ContentType database. The values in excluded_models are matched against columns that might be returned, and any columns that match will be dropped from output.

  1. Make a view to capture form requests and pass the request off to the interrogator:
def custom_table(request):
    return interrogation_room(request, template='your/interrogation/template.html')
  1. Make sure your template can handle the interrogation procedures:
{% load data_interrogator %}

{% lineup %} {# loads the form for selecting columns #}
{% interrogation_room %} {# loads the table where data is displayed #}
  1. Thats it!

The Interrogation dossier is a powerful way of altering how data is output. Along with specifying a model that can be a base_model, you can specify wrapsheets for them - i.e. special ways of displaying columns.

Below is an example dossier for a single model, with a wrapsheet for the column foo on the model YourModel:

DATA_INTERROGATION_DOSSIER = {
    'base_models': [
      { "model":("yourappname","YourModel"),
        "wrap_sheets": {
            "foo": {
                "columns": ['pk','bar'],
                "template": "yourapp/special_columns/for_foo.html",
            }
       },
    ]
 }

The columns value in the custom_cell_display specified additional columns of data to be retrieved when querying the specified attribute. So in the above example, whenever anyone requests the foo attribute when interrogating the YourModel model the pk and bar fields will also be retrieved, but will not be visible in the output table. However they will be accessible in the yourapp/special_columns/for_foo.html template which will be used when rendering the <td> table cell in the table.

Data Interrogator integrates nicely with Bootstrap and by default adds a table class to use Bootstrap's built in styling for tables. If you want to do additional customisation of the "interrogation room" table, just override the data_interrogator/table_display.html template. For example to convert the interrogation room table into one that is responsive and has table striping, just change the template to that below:

<table class="table table-responsive table-striped">
    <thead>
        <tr>
            {% for col in columns %}
                <th data-switchable='true' data-sortable='true'>{% clean_column_name col %}</th>
            {% endfor %}
        </tr>
    </thead>
    <tbody>
        {% for row in rows %}
        <tr>
            {% for col in columns %}
                <td>{% wrap_sheet row col %}</td>
            {% endfor %}
        </tr>
        {% endfor %}
    </tbody>
</table>

The data line-up is the form used to select models, columns and contstaints, this doesn't come with built-in support for Bootstrap, but can be overriden in a similar way to the example above by overriding the data_interrogator/lineup.html template.

Bootstrap-Table is a powerful front-end table manipulation plug-in for Bootstrap that providings additional filtering, sorting and searching within html tables. It also has an export extension that allows users to download data from the table in a variety of formats including XML, JSON, CSV and Excel spreedsheets.

Bootstrap-Table and Data Interrogator work well together, and just require loading the correct javascript libraries for Bootstrap-Table, and altering the data_interrogator/table_display.html template to add the right data attributes for driving the javascript, for example:

<table class="table" data-toggle="table"
       data-toolbar="#toolbar"
       data-search="true"
       data-show-filter="true"
       data-show-toggle="true"
       data-show-columns="true"
       data-show-export="true"
>
{# rest of template goes here #}

If we assume that we have an app with a model for Police Officers with the following models:

class PoliceOfficer:
    name = CharField(max_length=150)
    rank = CharField(max_length=150)
    precint = ForeignKey(Precinct)

class Precinct:
    name = CharField(max_length=150)
    number = IntegerField()
    captain = ForeignKey(PoliceOfficer, related_name="command")

class Arrest:
    officer = ForeignKey(PoliceOfficer)
    perp_name = CharField(max_length=150)
    crime = CharField(max_length=150)

With all of the above set up, you should have a page that looks similar to that below.

https://cloud.githubusercontent.com/assets/2173174/8870301/4511a998-3230-11e5-94e0-2a60968a814a.png

In the above image we can see a user can add or remove filtering constraints, columns and ordering fields. For example, in the above image, we are querying the "Person" model which contains a list of police officers, filtering where the rank field equals "Detective" and extracting the persons name, precinct number, precinct captain's name, and the count of their arrests, all of which is ordered by arrests largest-to-smallest.

name precinct.number precinct.captain.name count(arrest)
Jake Peralta 99 Raymond Holt 177
Amy Santiago 99 Raymond Holt 168
Roza Diaz 99 Raymond Holt 77
Charles Boyle 99 Raymond Holt 67
Michael Hitchcock 99 Raymond Holt 8
Norm Scully 99 Raymond Holt 6

Behind the scenes the data interrogator converts text fields into a format that can be used within the django QuerySet API. In this example, dots (.) become double underscores (__) that allow a query to follow foreign keys. So in the above query the column precinct.number becomes precinct__number, this can then be fed into the values function in the django queryset API <https://docs.djangoproject.com/en/1.8/ref/models/querysets/#django.db.models.query.QuerySet.values>. While 'dot notation' is used for simplicity regular django column names with underscores can be used.

Aliases can be set using the := command to convert django field or column names into human readable names. For example a column definition across multiple columns can be shortened like so: Precinct:=officer.precinct.name.

Simple calculations can be performed within queries to calculate against two columns. For example, officer.age - officer.graduation.age_at_graduation would calculate the duration between an officers current age, and when they graduated.

This can be used with an alias, like so: Years of service:=officer.age - officer.graduation.age_at_graduation

Current math functions allowed are addition (+), subtraction (-), multiplication (*) and division (/).

A small number of aggregate functions are available from the front end - currently Count(), Max() and Min(). Since these need to be set up in code, these need to be exectued using special syntax - that is just wrapping a column name in the aggregating command (like demonstrated above), with the argument count(arrests).

Supported aggregates are:

  • min(column): Returns the minimum value in the associated column.

  • max(column): Returns the minimum value in the associated column.

  • sum(column): Returns the total added value of all entries in the associated column.

  • avg(column): Returns the mean average of the associated column.

  • count(column): Returns the total number of entries in the associated column.

  • substr(column, start_position, end_position): Returns a substring of entries in the column. Example: substr(name, 0, 5) returns the first 5 letters of each entry in a column

  • concat(column1, column2, ...): Returns a joined string of a number of columns. Static strings can be included in quotes. Example: concat(first_name, " ", last_name, ".") retuns a single column with a full name with a space in the middle and a period at the end.

  • group(column): Returns a string that contains all columns concatenated together. Example: group(column)

  • sumif(column): Returns a sum of all values that meet a condition in a column. Example: sumif(age, age>18) will get the total age for all people over 18

  • lookup(column): Returns a lookup for a column. See below:

    Look ups allow for a pivot-table like extract of data from a matching joined. For example, if we have the arrests table above and want a list of officers, crimes they have arrested people for, and names of suspects the following query would provide this:

    name, Grand Theft Auto:=lookup(arrest.crime,"Grand Theft Auto",arrest.suspect), Larceny:=lookup(arrest.crime,"Larceny",arrest.suspect)

    name

    Grand Theft Auto

    Larceny

    Jake Peralta

    Mary Smith

    Bob Andrews

    Amy Santiago

    John Rogers

    Jeff Fakename

    Roza Diaz

    Walter Gower

    Rob Ogdens

The aggregators.py file provides the InterrogatorFunction which can be used to transform an argument string into a django expression. Each InterrogatorFunction has the following:

  • command: Class property that defines the name of the function in the user interface. eg. command = "my_func" will expose a my_func function to users in the UI.
  • aggregator: Class property that defines the django expression used in the function.
  • process_arguments(self, argument_string): Instance method that converts the string to arguments (args and kwargs) for the aggregator expression.

To refine data, filters can be used to reduce the resulting data. The following filters are currently supported, but may not work for all data types.

Filters are written with a field or column name, a filter type, and an argument without any quotes.

filter Django Equivalent Description Example
= (blank) Equal to name = sam
<> ne Not equal to name <> bob
< lt Less than age < 30
> gt Greater than age > 30
<= lte Less than or equal to date >= 2024-01-01
>= gte Greater than or equal to date <= 2024-01-01
&contains contains Contains the exact matching text name &contains Fred
&icontains icontains Contains the text in any case (matches both UPPER or lower case text) name &icontains fred
in in Value is in a list (the argument must be a comma separated list, eg 1,2,3) name in sam,bob,fred

Most django queries in filters match a field with a given string, however there are cases where you would like to compare values between columns. These can be achieved by using F() statements in django. A user can specify that a filter should compare columns with an F() statement by using a double equals in the filter. If for example, we wanted to see a list of officers who had also been arrested we could do this by filtering with name==arrest.perp_name which would be normalised in django to QuerySet.filter(name=F('perp_name')).

To look up a field in the list of values, we can use in, which will be normalised to Django's __in filter. For example you would like to look for officers with precinct numbers 98 or 99. It's achievable with precinct.number in 98,99 filter which normalised to django like Precinct.objects.filter(number__in=[98,99])

To exclude values from the search we could use not in which will be normalised to django .exclude() filter. For example you would like to look for officers with captians other than captain 'Raymond Holt' or 'Brad Prechet', So we use: precinct.captain.name not in Raymond Holt,Brad Prechet filter which normalised to django Precinct.objects.exclude(captain__name__in=[Raymond Holt,Brad Prechet])

  • cd dev
  • docker-compose up -d
  • docker-compose exec dev bash
  • django-admin [YOUR_COMMAND]

To play with data load the shops fixture

  • django-admin migrate
  • django-admin loaddata data.json

To run the development server

  • python manage.py runserver 0.0.0.0:8001
  • pipenv install django dj-database-url

  • pipenv shell to drop into the virtual environment

  • PYTHONPATH=./app DJANGO_SETTINGS_MODULE=app.settings python3 manage.py runserver 0.0.0.0:9000 to run the development server.

  • In VS Code, select the Python interpreter from the virtual environment:
    • Ctrl-Shift-P - Open the command selector
    • 'Python: Select interpreter': Select the one with the django-data-interrogator prefix.
  • In VS Code, edit the project's launch.json and add the following entry:

    {

    // Use IntelliSense to learn about possible attributes. // Hover to view descriptions of existing attributes. // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 "version": "0.2.0", "configurations": [

    {

    "name": "Python: Django", "type": "python", "request": "launch", "program": "${workspaceFolder}/manage.py", "args": [

    "runserver" "0.0.0.0:9000"

    ], "env": {

    "PYTHONPATH": "./app", "DJANGO_SETTINGS_MODULE": "app.settings"

    }, "django": true, "justMyCode": true

    }

    ]

    }

  • Press F5 to launch and debug.

About

django-data-interrogator is a plug in table builder that allows users to easily extract information from a Django database

Resources

License

Stars

Watchers

Forks

Packages

No packages published