Skip to content

eric-mokuma/knex-todo-cli

Repository files navigation

knex-todo-cli

Command-line todo app using Knex.

We're building a simple command-line tool to manage our list of todos. We're finally at the point of storing our data in a database! Woooo! We're using the Knex module to talk to our SQLite3 database.

Remember: you can always look at the knex documentation when creating your database functionality, especially when building queries or schema.


Setup

0. Installation and settings

  • Install dependencies knex and sqlite3

    More about installing

    You can do them both at once like this.

    npm install knex sqlite3
  • Set file permissions using chmod +x todo.js

    More about file permissions

    Since this is a CLI (command-line interface) tool, instead of running our app using node todo.js list, we'd like to be able to run it like any other utility/script on our computer to make it easier to use. Running chmod +x todo.js in your terminal adds the executable flag to the file. Now you can run it in your console using ./todo.js list. This means our programme will begin with the todo.js file.

    Note: if you run ./todo.js list now, you will get an error because we still need to complete some more steps before we can show the contents of our database.

  • Create the Knex configuration file (knexfile.js) with npm run knex init

  • Convert the knex file to an ecmascript module

    How to convert knexfile to a module

    To be an ESM module, we just replace:

    module.exports = {

    with:

    export default {

Setting up the database

1. The first migration

  • Use npm run knex migrate:make todos to create a migration file

  • Convert the migration to an ESM module

    How to convert your migration to a module

    To convert our migration functions we just replace this..

    exports.up = function (knex) { 

    ... with

    export function up(knex) {

    and replace ...

    exports.down = function (knex) { 

    ... with

    export function down(knex) {
  • Edit the new file in the new migrations folder so it will add (and drop) a table called todos

    More about the todos table

    It should have the following fields: _ id (auto incrementing) _ task: string

    The documentation for dropTable might be helpful.

  • Use npm run knex migrate:latest to apply the changes to the database

2. Seeds

  • Use npm run knex seed:make test-tasks to create a seed file

  • Edit the new file in the new seeds folder so it will add new tasks to the todos table

    Tip First, we need to convert it to an ESM module by changing from this:
    exports.seed = async function (knex) {

    to this...

    export async function seed(knex) {

    The documentation for del and insert might be helpful.

  • Run npm run knex seed:run to add the new data to the database

3. Viewing data in the database

  • Choose and set up a way to view the contents of the database
More about viewing data

There are a number of different options for peeking into your SQLite database. We recommend you use the SQLite Viewer VS Code extension. Alternatively, you can install a desktop application, such as the DB Browser for SQLite (installed on the campus computers) or DBeaver (great for all of the common relational databases - not just SQLite). Or you can use an online tool such as this SQLite Viewer.


Creating, reading, updating, and deleting tasks (CRUD)

4. Display tasks and IDs

We want to be able to update and delete our tasks. But before we do that we need to be able to identify them. This part has been completed for you as a demonstration.

  • Familiarise yourself with the commands.js file

    More about the commands file

    If you type ./todo.js list in your terminal, this should output a list of tasks. The input + output should look like this:

    $ ./todo.js list
    
    1: vacuum
    2: buy groceries

    Notice two things about this example:

    • the commands are all separated into a different module, so that todo just calls a required function from commands.js
    • commands.js has a dependency on db.js to interact with the database, but todo does not (it doesn't need it)
  • Familiarise yourself with the contents of the todo file

More about the todo file

In particular, what is process.argv? And how is it being used to get the command (cmd) that was typed (in our example, list)?

Start by using console.log to explore this, and try adding more inputs to see how that changes the result (i.e. ./todo.js list hello testing 123)

5. Delete a task by ID

  • Enable users to complete a task by entering a command such as ./todo.js done 1 which will remove the task with id of 1 from the database

    More about deleting a task

    You'll want to add a new function in db.js that can delete a row given its id. Look how the other functions work. You might need to review promises.

    To use the new function, add a function in commands.js called deleteTodo (or similar). Remember that you will need to pass an argument through from the todo module to so you can tell your DB function which task to delete. You likely want to pass the id as the first parameter to deleteTodo, as below. This is so the optional db parameter can be safely omitted.

    function deleteTodo(id, db = connection) {}
    

    Additional hint: accessing that userInputs array might come in handy right about now...

    If it helps, look at how the list function is structured to give you some ideas. What is happening with those catch and finally bits of code? What happens when you remove the finally block?

6. Add a new task

It's all very well and good being able to delete tasks, but what happens when we run out of things to do?

  • Enable users to add a new task by entering a command such as ./todo.js add 'pet cat'
More about adding a task

You will need to add a function to db.js so we can insert a new task into our database, and also add a function to commands.js (that we will then call from our todo file) to make use of this.

7. Update a task by ID

Users make mistakes. Let them update a task.

  • Enable users to update a task by id with a command such as ./todo.js update 1 'clean my room thoroughly'

    More about updating a task

    As before, add a function to db.js that does the actual updating of the database. Then add a function to commands.js that makes use of it.


Other task interactions

8. Add ability to search

Busy people are complaining about having 200 tasks in their consoles. Add a feature that searches in the task string for a given word.

  • Enable users to search for tasks containing a search term by entering a command such as ./todo.js search 'wire'

9. Preparing to complete tasks (non-destructively)

Users want to be able to mark a task as complete without removing it from the database.

  • Use npm run knex migrate:make add-completed-column to create a new empty migration. Then update the new migration to add a column to the table

    More about the new migration

    The documentation for knex.schema.table might be helpful when modifying an existing table.

    What data type should we use to store our new field(s)?

  • Fill in the down function in your migration. It should undo the changes made in the .up function

  • Run and check the migration, and re-run the seeds

    More about checking the migration and seeds
    1. Run npm run knex migrate:latest to run the new migration applying the changes to the database. If you don't get any errors, inspect the database in the SQLite Manager (The application called DB Browser for SQLite that you set up in section 3). Is it what you expected? What happened to existing data in the database?
    2. Run npm run knex migrate:rollback and look in your database.
    3. Run npm run knex migrate:latest and look again.
    4. Run npm run knex seed:run and look again.

10. Finish the "mark task complete" feature

  • Enable users to mark a task complete, without deleting it from the database

    More about completing...completion

    It's up to you to decide how far you want to go with this. Should listing all the tasks show completed and uncompleted tasks? Maybe you should add the task completed status when printing out a task. Maybe you can filter by completed when listing?


Stretch

More about stretch challenges

What is the next feature that would make this tool more useful for you? A priority field? Sorting? Tags? Archival? Whatever it is, add it!


A note on using the debugger

More about debugging

You'll find this challenge already has debugging set up for you, if you would like to use it. However, it won't start working until you complete the initial setup steps below! In addition, because we're debugging a console program, you'll need to change the args property in you debugger configuration to the actual command you'd like to debug. For example,

  "program": "${workspaceFolder}/todo.js",
  "args": [
      "done",
      "1"
  ]

would debug the ./todo.js done 1 command. Ask a teacher for help if you're not sure!


Provide feedback on this repo

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published