Web Dev

Developing Front-End Microservices With Polymer Web Components And Test-Driven Development (Part 2/5): Polishing The First Component

book-form-scIn the previous article we created our first Polymer Web Component using Test-Driven Development approach. While all the specifications we wrote were implemented, there are still few things we should do to polish this component. The tc-book-form currently has the form with all the input fields required by the back-end. It also has buttons to add or update and delete a book. Those buttons are calling iron-ajax Polymer element that sends requests to the back-end. Both front-end (in form of Web Component) and back-end are developed as a single microservice.

The next task in front of us is to polish the component. Later on in this series we’ll create another component that will list all books. We’ll also develop the code that will enable us to establish communication between those two components. Once we’re done with both, we’ll explore ways to integrate the microservices we developed into a separate Web Application.

This article is part of Developing Front-End Microservices With Polymer Web Components And Test-Driven Development series:

You can continue working on the code we wrote in the previous article. If you had trouble completing it, please checkout the polymer-book-form branch of the books-service repository. It contains all the code from the previous article.

git checkout polymer-book-form

If you have trouble following the exercises in this article, the complete source code can be found in the polymer-book-form-2 branch of the books-service repository.

Specifying Book Retrieval

We should provide a way for users of our component to open a book. As in the previous examples, we’ll start with definition of another iron-ajax component. This time, we’ll use it to retrieve a book. Back-end API for retrieving a book is similar to the DELETE request we already implemented. The request should be made to the /api/v1/books/_id/[ID] URL with [ID] replaced with the actual ID of the book we want to retrieve. The major difference is that this time the method should be GET.

The specification of the getAjax element is following.

[client/test/tc-book-form.html]

describe('getAjax element', function() {

    it('is defined', function() {
        assert.isDefined(myEl.$.getAjax);
    });

});

The implementation is following.

[client/components/tc-books/tc-book-form.html]

<iron-ajax id="getAjax"></iron-ajax>

Since the default method of the iron-ajax element is GET, there is no need to specify it explicitly.

With getAjax up-and-running, we should define the open function that users of our component can call. It should populate the form fields with the response from the server or, in case we pass 0 as book ID, clear the fields. For this function to operate correctly we should set the correct URL to getAjax, call generateRequest and display or hide the Delete button depending on whether ID is passed to the function or not.

[client/test/tc-book-form.html]

describe('open function', function() {

    it('is defined', function() {
        assert.isDefined(myEl.open)
    });

    it('sets getAjax.url', function() {
        var bookId = 584;
        myEl.requestUrl = '/url/to/my/service';

        myEl.open(bookId);

        assert.equal(myEl.$.getAjax.url, myEl.requestUrl + '/_id/' + bookId);
    });

    it('calls getAjax.generateRequest', function() {
        var el = fixture('fixture');
        el.$.getAjax.generateRequest = sinon.mock();

        el.open(373);

        sinon.assert.calledOnce(el.$.getAjax.generateRequest);
    });

    it('resets _data when bookId is 0', function() {
        var data = {something: 'else'};
        myEl._data = data;

        myEl.open(0);

        assert.notEqual(myEl._data, data);
        assert.isUndefined(myEl._data._id);
    });

    it('displays delete element when bookId is greater than 0', function() {
        myEl.$.delete.hidden = true;

        myEl.open(429);

        assert.isFalse(myEl.$.delete.hidden);
    });

    it('hides delete element when bookId is 0', function() {
        myEl.$.delete.hidden = false;

        myEl.open(0);

        assert.isTrue(myEl.$.delete.hidden);
    });

});

The implementation is following.

[client/components/tc-books/tc-book-form.html]

open: function(bookId) {
    if (bookId > 0) {
        this.$.getAjax.url = this.requestUrl + '/_id/' + bookId;
        this.$.getAjax.generateRequest();
        this.$.delete.hidden = false;
    } else {
        this._data = {};
        this.$.delete.hidden = true;
    }
}

Finally, we might want to define the default behavior of our component. It should present the fields ready to insert a new book. Besides standard Web Components lifecycle callbacks, Polymer adds a new one called ready. It is called when all elements inside the component’s local DOM have been configured.

In our case, it should be enough if ready calls the open function we just implemented.

[client/test/tc-book-form.html]

describe('ready function', function() {

    it('calls open function', function() {
        var el = fixture('fixture');
        el.open = sinon.mock();

        el.ready();

        sinon.assert.calledWith(el.open, 0);
    });

});

The implementation is following.

[client/components/tc-books/tc-book-form.html]

ready: function() {
    this.open(0);
}

The only thing we have left inside this component is to specify what should be done when we receive responses from the back-end server.

Specifying HTTP Response Events

We are performing three different requests to the back-end server; GET, PUT and DELETE. Responses from those requests can be successful with code 200 or contain an error. Whatever the response, we should notify users of our component so that they can decide what to do next. Notifications can be done by firing custom events. Polymer has fire as one of it’s utility functions.

We’ll start with a function that will handle GET response and continue with the rest of 200 responses. Apart from making sure that function itself is defined, we should take the response and set it to the _data property that we defined in the previous article. Finally, we’ll fire the getBook custom event.

[client/test/tc-book-form.html]

describe('_handleGetResponse function', function() {

    it('is defined', function() {
        assert.isDefined(myEl._handleGetResponse);
    });

    it('sets _data property', function() {
        var expected = {something: 'not seen before'};
        var event = {
            detail: {
                response: expected
            }
        };
        myEl._data = {};

        myEl._handleGetResponse(event);

        assert.equal(myEl._data, expected);
    });

    it('fires getBook event', function() {
        var data = {something: 'else'};
        var response = {
            detail: {
                response: data
            }
        };
        var actual;
        var onGetBook = sinon.spy(function(e) {
            actual = e.detail;
        });
        myEl.addEventListener('getBook', onGetBook);

        myEl._handleGetResponse(response);

        sinon.assert.calledOnce(onGetBook);
        assert.equal(actual, myEl._data);
    });

});

The implementation is following.

[client/components/tc-books/tc-book-form.html]

_handleGetResponse: function(e) {
    this._data = e.detail.response;
    this.fire('getBook', this._data);
}

Similar should be done with PUT and DELETE responses.

[client/test/tc-book-form.html]

describe('_handlePutResponse', function() {

    it('is defined', function() {
        assert.isDefined(myEl._handlePutResponse);
    });

    it('fires putBook event', function() {
        myEl._data = {something: 'else'};
        var actual;
        var onPutBook = sinon.spy(function(e) {
            actual = e.detail;
        });
        myEl.addEventListener('putBook', onPutBook);

        myEl._handlePutResponse();

        sinon.assert.calledOnce(onPutBook);
        assert.equal(actual, myEl._data);
    });

    it('displays delete element', function() {
        myEl.$.delete.hidden = true;

        myEl._handlePutResponse();

        assert.isFalse(myEl.$.delete.hidden);
    });

});

describe('_handleDeleteResponse', function() {

    it('is defined', function() {
        assert.isDefined(myEl._handleDeleteResponse);
    });

    it('calls open function', function() {
        var el = fixture('fixture');
        el.open = sinon.mock();

        el._handleDeleteResponse();

        sinon.assert.calledWith(el.open, 0)
    });

    it('fires deleteBook event', function() {
        var data = {something: 'else'};
        myEl._data = data;
        var actual;
        var onDeleteBook = sinon.spy(function(e) {
            actual = e.detail;
        });
        myEl.addEventListener('deleteBook', onDeleteBook);

        myEl._handleDeleteResponse();

        sinon.assert.calledOnce(onDeleteBook);
        assert.equal(actual, data);
    });

});

The implementation is following.

[client/components/tc-books/tc-book-form.html]

_handlePutResponse: function() {
    this.fire('putBook', this._data);
    this.$.delete.hidden = false;
},
_handleDeleteResponse: function() {
    var data = this._data;
    this.open(0);
    this.fire('deleteBook', data);
}

We should also react when the server responds with an error. We already have the generic _handleError function that can be used.

[client/test/tc-book-form.html]

describe('_handlePutError function', function() {

    it('is defined', function() {
        assert.isDefined(myEl._handlePutError);
    });

    it('calls _handleError', function() {
        var el = fixture('fixture');
        el._handleError = sinon.mock();

        el._handlePutError();

        sinon.assert.calledWith(el._handleError, 'The book could not be stored in the server');
    });

});

describe('_handleGetError function', function() {

    it('is defined', function() {
        assert.isDefined(myEl._handleGetError);
    });

    it('calls _handleError', function() {
        var el = fixture('fixture');
        el._handleError = sinon.mock();

        el._handleGetError();

        sinon.assert.calledWith(el._handleError, 'The book could not be retrieved from the server');
    });

});

describe('_handleDeleteError function', function() {

    it('is defined', function() {
        assert.isDefined(myEl._handleDeleteError);
    });

    it('calls _handleError', function() {
        var el = fixture('fixture');
        el._handleError = sinon.mock();

        el._handleDeleteError();

        sinon.assert.calledWith(el._handleError, 'The book could not be removed from the server');
    });

});

The implementation is following.

[client/components/tc-books/tc-book-form.html]

_handlePutError: function() {
    this._handleError('The book could not be stored in the server');
},
_handleGetError: function() {
    this._handleError('The book could not be retrieved from the server');
},
_handleDeleteError: function() {
    this._handleError('The book could not be removed from the server');
}

Finally, we should make sure that the functions we just implemented are called from iron-ajax components. It contains onResponse and onError functions that can be used in such cases.

The full implementation or our iron-ajax components is following.

[client/components/tc-books/tc-book-form.html]

<iron-ajax id="getAjax"
           on-error="_handleGetError"
           on-response="_handleGetResponse">
</iron-ajax>
<iron-ajax id="putAjax"
           method="PUT"
           on-response="_handlePutResponse"
           on-error="_handlePutError"
           content-type="application/json"
           url="[[requestUrl]]">
</iron-ajax>
<iron-ajax id="deleteAjax"
           method="DELETE"
           on-response="_handleDeleteResponse"
           on-error="_handleDeleteError">
</iron-ajax>

This concludes the development of the tc-book-form component. Please open the http://localhost:8080/components/tc-books/demo/index.html to see the end result.

What can we do next? Let us explore ways to utilize it.

Utilizing Component Functions And Events

We’ll try to give a bit more life to the existing demo and simulate few of the usages the component might have later on when we import it to the application.

We can, for example, add HTML elements and script that would tell the tc-book-form component to open an existing book or a new one. Bear in mind that what we are about to do is temporary. In the next article we’ll create a new component that will list books and have options to tell the application to open a book.

Please open the components/tc-books/demo/index.html. It contains the demo of the component we just built.

We’ll start by adding an input and two buttons. Add following HTML below the body tag.

[components/tc-books/demo/index.html]

<body>
    <div>
        <input id="bookId">
        <button onclick="openBook()">Open</button>
    </div>
    <div>
        <button onclick="newBook()">New</button>
    </div>
    ...
</body>

Now we’ll create openBook and newBook that will call tc-book-form open function. Depending on the ID we pass, a new book or an existing one will be opened.

[components/tc-books/demo/index.html]

<script>
    var tcBookForm = document.querySelector('#tc-book-form');
    function openBook() {
        var bookId = document.querySelector('#bookId');
        tcBookForm.open(bookId.value);
    }
    function newBook() {
        tcBookForm.open(0);
    }
</script>

As you can see, there is no need for us to know internals of the component but only those properties and functions that are exposed to us.

book-form-demo

We can try it out by opening the http://localhost:8080/components/tc-books/demo/index.html.

  1. Fill in the form and press submit. The book info has been stored in the back-end.
  2. Press the New button. The book form has been cleared and is ready for us to populate it with data of a new book.
  3. Put the number of the book you submitted into the input field left of the Open button. Press the Open button. The book was retrieved from the back-end and the form has been populated.

We might, for example, display notifications whenever some book is added, updated, removed or retrieved. This can be done with paper-toast component and utilization of custom events that our component is firing.

Add following to the index.html body.

[components/tc-books/demo/index.html]

<paper-toast id="toast"></paper-toast>

We should also add a bit of JavaScript.

[components/tc-books/demo/index.html]

<script>
    ...
    var toast = document.querySelector('#toast');
    tcBookForm.addEventListener('getBook', onGetBook);
    tcBookForm.addEventListener('putBook', onPutBook);
    tcBookForm.addEventListener('deleteBook', onDeleteBook);
    function onGetBook(e) {
        bookChanged(e.detail.title, 'retrieved');
    }
    function onPutBook(e) {
        bookChanged(e.detail.title, 'saved');
    }
    function onDeleteBook(e) {
        bookChanged(e.detail.title, 'deleted');
    }
    function bookChanged(title, status) {
        toast.text = title + ' has been ' + status + '.';
        toast.show();
    }
    ...

From now on you’ll see messages whenever one of the book operations is run inside the component.

To Be Continued

What we did above is only a quick “workaround” to showcase how we can interact with Polymer Web Components. In the next article we’ll develop the second component that will list all available books and be able to fire custom events. Those events will be used to communicate between the two components. Once both components are done we’ll move on and use them in a separate Web Application. The story continues in The Second Component article.

Viktor Farcic

Viktor Farcic is a Software Developer currently focused on transitions from Waterfall to Agile processes with special focus on Behavior-Driven Development (BDD), Test-Driven Development (TDD) and Continuous Integration (CI).
Subscribe
Notify of
guest

This site uses Akismet to reduce spam. Learn how your comment data is processed.

0 Comments
Inline Feedbacks
View all comments
Back to top button