Skip to content

Latest commit

 

History

History
288 lines (218 loc) · 8.77 KB

README.md

File metadata and controls

288 lines (218 loc) · 8.77 KB

Tutorial 09 - Data models: build a simple deposit form

The goal of this tutorial is to learn how we can build a new simple form to be able to deposit new records. The invenio-deposit module is not in the scope of this exercise.

Table of Contents

We now have to enable users to deposit new records. For this exercise, we won't use what invenio-records-rest already provides out-of-the-box, but we will implement a custom view.

We will need:

  • to render a simple form to create a record, where the user can input the value of each field
  • a new view where to post the form, validate the input and create the new record
  • a new view to display a success message or an error

Step 1: Bootstrap exercise

If you completed the previous tutorial, you can skip this step. If instead you would like to start from a clean state run the following commands:

$ cd ~/src/training/
$ ./start-from.sh 08-data-models-from-scratch

Step 2: Create a simple form

We are going to create a new module in our project which will contain all our files. Let's create a new module folder: my-site/my_site/deposit/. When creating a new module, do not forget to add __init__.py:

my-site/my_site/deposit/__init__.py

"""Deposit module."""

Then, we create the form, using Flask-WTF.

my-site/my_site/deposit/forms.py

"""Simple deposit form module."""

from __future__ import absolute_import, print_function

from flask_wtf import FlaskForm
from wtforms import StringField, validators


class RecordForm(FlaskForm):
    """A simple deposit form."""

    title = StringField(
        'Title', [validators.DataRequired()]
    )
    contributor_name = StringField(
        'Name of the contributor', [validators.DataRequired()]
    )

Step 3: Display the form

We have to create a new view for the form. We will take advantage of the HTTP verbs GET, to render the form, and POST, to handle the submitted data. We are also going to protect the view with the login_required decorator: in this way, Invenio will redirect to the login view if the user is anonymous and we can then set the owner of the newly created record to the id of the logged in user.

my-site/my_site/deposit/views.py

"""Views for deposit of records."""

from __future__ import absolute_import, print_function

from flask import Blueprint, redirect, render_template, url_for
from flask_login import login_required
from flask_security import current_user

from .forms import RecordForm
from .api import create_record


# define a new Flask Blueprint that is register under the url path /deposit
blueprint = Blueprint(
    'deposit',
    __name__,
    url_prefix='/deposit',
    template_folder='templates',
    static_folder='static',
)

@blueprint.route('/create', methods=('GET', 'POST'))
@login_required
def create():
    """The create view."""
    form = RecordForm()
    # if the form is submitted and valid
    if form.validate_on_submit():
        # we create one contributor object with the submitted name
        contributors = [dict(name=form.contributor_name.data)]
        # set the owner as the current logged in user
        owner = int(current_user.get_id())
        # create the record
        create_record(
          dict(
            title=form.title.data,
            contributors=contributors,
            owner=owner,
          )
        )
        # redirect to the success page
        return redirect(url_for('deposit.success'))
    return render_template('deposit/create.html', form=form)


@blueprint.route("/success")
@login_required
def success():
    """The success view."""
    return render_template('deposit/success.html')

Templates creation: create a folder templates and a subfolder deposit (the name of the blueprint). Then, inside, two html files (our Jinja templates): create.html and success.html.

my-site/my_site/deposit/templates/deposit/create.html

{%- extends config.BASE_TEMPLATE %}

{% macro errors(field) %}
  {% if field.errors %}
  <span class="help-block">
    <ul class=errors>
    {% for error in field.errors %}
      <li>{{ error }}</li>
    {% endfor %}
    </ul>
  {% endif %}
  </span>
{% endmacro %}

{% block page_body %}
  <div class="container">
    <div class="row">
      <div class="col-md-12">
        <h2>Create record</h2>
      </div>
      <div class="col-md-offset-3 col-md-6 well">
        <form action="{{ url_for('deposit.create') }}" method="POST">
            <div class="form-group {{ 'has-error' if form.title.errors }}">
              <label for="title">{{ form.title.label }}</label>
              {{ form.title(class_="form-control")|safe }}
              {{ errors(form.title) }}
            </div>
            <div class="form-group {{ 'has-error' if form.contributor_name.errors }}">
              <label for="contributor_name">{{ form.contributor_name.label }}</label>
              {{ form.contributor_name(class_="form-control")|safe }}
              {{ errors(form.contributor_name) }}
            </div>
            {{ form.csrf_token }}
            {{ errors(form.csrf_token) }}
            <button type="submit" class="btn btn-default">Create</button>
        </form>
      </div>
    </div>
  </div>
{% endblock page_body %}

my-site/my_site/deposit/templates/deposit/success.html

{%- extends config.BASE_TEMPLATE %}

{% block page_body %}
  <div class="container">
    <div class="row">
      <div class="col-md-12">
        <div class="alert alert-success">
          <b>Success!</b>
        </div>
        <a href="{{ url_for('deposit.create') }}" class="btn btn-warning">Create more</a>
      </div>
    </div>
  </div>
{% endblock page_body %}

Step 4: Create the submitted record

We will now implement the create_record method: its responsibilities are to create the record in the database, mint a new persistent identifier and index it to make it searchable.

my-site/my_site/deposit/api.py

"""Deposit APIs."""

from __future__ import absolute_import, print_function

import uuid

from flask import current_app
from invenio_db import db
from invenio_indexer.api import RecordIndexer
from invenio_pidstore import current_pidstore
from invenio_records.api import Record


def create_record(data):
    """Create a record.

    :param dict data: The record data.
    """
    with db.session.begin_nested():
        # create uuid
        rec_uuid = uuid.uuid4()
        # create PID
        current_pidstore.minters['recid'](rec_uuid, data)
        # create record
        created_record = Record.create(data, id_=rec_uuid)
        # index the record
        RecordIndexer().index(created_record)
    db.session.commit()

Step 5: Update the entrypoints

Last step: add the new view in the app entrypoints. When the Invenio app will run, it will find the new blueprint and register its routes.

Add this in my-site/setup.py:

        'invenio_base.blueprints': [
            'my_site = my_site.theme.views:blueprint',
            'my_site_records = my_site.records.views:blueprint',
+           'my_site_deposit = my_site.deposit.views:blueprint',
        ],
        'invenio_assets.webpack': [
            'my_site_theme = my_site.theme.webpack:theme',
        ],

Re-install the app to register the entrypoints:

$ pipenv run pip install -e .

Finally, let's create an user to access to the protected form.

$ pipenv run my-site users create [email protected] -a --password=123456

Step 6: Try it!

Try it! Ensure docker-compose is running and restart the server (if not done already automatically) and visit https://127.0.0.1:5000/deposit/create:

$ ./scripts/server
$ firefox https://127.0.0.1:5000/deposit/create

Login with the credentials set before: username [email protected] and password 123456. Now, create a record by entering some data and submitting the form.

Deposit form page

Hit the Create button! You should see the Success message.

Record create success

Finally, you can visit https://127.0.0.1:5000/api/records/?prettyprint=1 to verify it is indexed correctly:

$ firefox https://127.0.0.1:5000/api/records/?prettyprint=1

JSON api response

What did we learn

  • We have seen how to create a new view with templates
  • We have seen how to protect a view requiring login
  • We have built a new form with validation
  • We learned how to create a record minting its PID and then index it