This app enables users to monitor their monthly spending through predefined budgets and provides a yearly 'overview' sheet for a comprehensive view. It offers various categories for users to systematically record their expenses (defined in Structure). Once expenses are logged, users can access a summary of their monthly spendings in a Google Sheets format.
The application is available for viewing here.
The Google Sheets for this application is available for use here.
This application was initially developed with the additional functionality of currency choice and conversion, using the forex-python API. However, as the project neared its completion, the API malfunctioned and essentially shut down for maintenance, rendering this functionality unresponsive and with several terminal errors. As this was outside of control, a decision was made to remove this functionality and re-encorporate it at a later stage, perhaps using a different API.
The last commit message associated with currency conversion has a hash of d30cd8ca8066823956a70192c097ff8337439da0, and can be found here.
- Develop a user-friendly and efficient expense tracking tool to assist users in refining their spending behavior.
- Generate an immediate summary table that displays the user's expenditure records and available budget.
- Establish a Google Sheet as a repository for all recorded expenses, allowing users to review their spending history on:
- a monthly basis.
- a yearly basis.
- As a first time user, I need an intuitive and easy-to-understand interface.
- As a first time user, I need clearly defined and easy-to-follow instructions for using the application.
- As a first time user, I want the ability to quickly select options and enter my expenses.
- As a first time user, I expect immediate feedback on my entries, with an option to easily exit and restart in case of errors.
- As a first time user, I prefer having a single view where my expenses and budget are logged together for personal analysis. This should include a remainder calculation after all of the expenses have been deducted from my budget.
- As a regular user, I seek the capability to log multiple expenses simultaneously in different categories and see the total spent in each category.
- As a regular user, I need a visual cue to indicate when my spending exceeds my monthly budget.
- As a regular user, I desire a monthly expense record for future reference in a recognisable and legible format.
Due to the solely-backend nature of this project, enhancement strategies for the UI's visual appeal are limited. Nonetheless:
- Design a well-structured terminal interface that utilizes spacing effectively for optimal readability.
- Clear the terminal screen after each significant input to avoid visual littering and focus attention towards input validation and the next step.
- Implement distinct headings where suitable, to differentiate them from regular text, preferably using colour or table header elements.
- Use emojis and icons to signify different input requirements.
- Use spacing where appropriate (\n) or (tab) to offset elements from each other/ edge of the terminal.
The structure of the 'Tag Track' application is as follows:
- Title
- Name input prompt with validation.
- Month input prompt (number list) with validation.
- Retrieval of any existing budget/ budget remainder.
- Option of using the values or changing them.
- Total Monthly Budget input prompt with validation.
- Or category selection based on the user's choice of using existing budget or changing it.
- List of Categorised expenses for selection:
- Rent
- Groceries
- Vehicle
- Cafe/Restaurant
- Online Shopping
- Other
- Category Input prompt
- Expense input prompt with validation.
- Visual Appendage in the form of a table with relevant budget calculation. The table also retrieves the values of previous logs for each category and displays them in a different colour.
- Press 'u' to upload expenses to Google Sheets OR Type 'q' to quit...
- Ability to quit at most input prompts.
The purpose of the application is to gather expense data from the user and input this information into Google Sheets as individual rows. Google Sheets serves as a reference and as a recognisable and legible format in which the user may review their expenses, and see their annual spendings.
The scope of this application is as follows:
-
Name input with validation (to personalise the experience).
-
Choice of month with validation and confirmation prompt, essential as the app works on a monthly framework for budgeting and final tallies. Users can select a month numerically (e.g., 1 for January, 2 for February), with a confirmation step to prevent accidental selections. The chosen month appears as a heading in both Google Sheets and the terminal table.
-
Budget and Budget Remainder retrieval. After selecting a month, the application checks for any existing values in cells B1 - Month Budget, and F1 - Month Remainder. If found, prompts the user whether they wish to use these values or change them. If the user selects to use - the application jumps to category selection for expense logging. Else, continues to budget input.
-
Monthly Budget Input - later transferred to Google Sheets as data for calculation. All expenses get appended with the Euro currency symbol.
-
User inputs are confirmed with Type 'p' to proceed, 'c' to change, or 'q' to quit:. The input accepts lower and uppercase letters, strips them incase the user pressed space after the letter, and applies logic accordingly.
-
A list of expense categories, aiding users in organized entry of expenses. These categories also help in sorting expenses in Google Sheets and are used for generating expense totals in each category for each month in the annual Overview sheet.
-
After entering expenses, the application automatically calculates and displays the total monthly expenses per category. This feature helps users understand their spending patterns and notice any imbalances.
-
A prompt to ask whether users would like to upload their expenses to Google sheets. This is the final step and serves as a reference directory for the user should they wish to review or amend their monthly spendings at a later stage. The application displays a goodbye message after everything has been uploaded and exits.
The aim of the application is to provide an easy-to-use and interactive platform for monitoring expenses within a budget framework.
Users can track their spendings against a monthly budget and have the flexibility to check their remaining balance, whether it's in surplus or deficit, at any time. Through Tag-Track, users gain insights into their spending patterns, identify which months incur higher expenses, and can adjust their spending behavior accordingly.
- Anyone aged 16+ yrs.
- Anyone who would like to impove their spending habits.
- Easy-To-Use Interface
- Visual Expense Tracking and Logging Confirmation
- Category Selection
- Expense Input
- Conclusive User Table with Expenses and Budget Remainder
- Google Sheets Logging and Calculations
-
To enhance readability, spaces will be included at the start of sentences, distancing them from the terminal's edge.
-
Colors will be used judiciously to distinguish various pieces of information, thereby improving their legibility.
-
Icons/emojis will be used to signify the intent of prompts, key pieces of information, warnings, and tips.
-
On Google Sheets, each month and the final overview sheet will be set as a separate spread, so users have concise, separate areas in which to view their expenses.
Below is the flowchart for the application. Noted is the general flow of the application, the user story, and several validations with ValueError considerations and decision flows.
Color plays a key role in the tag-track interface, with specific hues chosen for their significance and legibility. The title of the tag-tracker and the headers of all tables feature green, a color universally linked with finance and money. Green is also chosen for its high legibility in the colorama Python library, making it a suitable choice.
For instances where a user exceeds their monthly budget, red is utilized in the final summary table to highlight the negative overdraft. In contrast, when the budget is not exceeded, the remaining balance is displayed in a darker shade of green, maintaining the financial theme.
Yellow is chosen for showing finances that were logged previously - mostly because it differs the most from red and green, and is easy to read against the dark terminal screen.
The main reason for using colour in the application is to provide a somewhat engaging interface for the user, and to distinguish key pieces of text from the terminal face.
Emojis/icons are used throughout the application to visually aid users in distinguishing prompt intents. A breakdown can be found below:
β οΈ : signifies a checkpoint or warning message.- β€ : signifies an input prompt.
- β : failed validation.
- π : indicates the type of expected input.
- β : successful validation.
- π‘ : signifies insight/note/tip.
- β : signifies a pending request.
- π : signifies exiting.
Users can choose a month for tracking their expenses from a numerical list. This selection process retrieves the corresponding monthly sheet from Google Sheets, along with any existing data relevant to that month. Later in the application, both the user's expenses and budget information are recorded onto the appropriate Google Sheet for that month, and calculations for total expenses are made for the corresponding month row in the overview sheet.
Users are given the option to select from 6 different categories to record their expenses. These categories represent columns in the month sheets, to which users can log their expenses. These same columns are replicated in the overview sheet, providing users with a detailed annual summary of their expenses across all categories.
After input, users are asked whether they wish to log another expense, or go to the next step (the conclusive expense table).
The user may log several times for one category. If there is more than one expense log per category, the expenses are added together and the total is displayed in the conclusive user expense table and later logged to Google Sheets (if the user chooses to).
Note: This image does not relate to the path followed through the application as it was taken at a later stage, but the journey and logic remains the same.
As mentioned, the user's month selection retrieves the corresponding Google Sheet with all relevant data already on it. This data includes a check for an existing budget previously logged by the user:
If a month budget exists, the user is prompted to choose to continue with this budget, or update it to a new value. The input is then validated and the application proceeds according to the user's choice.
In the first image below, there is an existing budget of 3000 euro for the month of January.
In the second image below, the user has selected to log for the month of January, the application detects an existing budget, displays it to the user, and asks the user whether they wish to continue with it or change its value.
Note: This image does not relate to the path followed through the application as it was taken at a later stage, but the journey and logic remains the same.
If there is no existing budget, the user is prompted to enter a budget for their chosen month. The input is validated and the user is either directed to the next step, or given a validation error with a tip on the type of input expected, and asked to re-enter their budget.
The conclusive user table displays a summary of the user's logged expenses. The top row shows the Month and Budget. The next rows show the total spendings for each category, and the final row displays the budget remainder after all spendings were deducted. If the budget remainder is positive, the result is displayed in green. If the remainder has an overdraft, the result is displayed in red.
Under the conclusive user expense table, the user is asked whether they want to log their expenses to Google Sheets. If not, the application displays a goodbye message and does not log anything. If yes, the expenses are logged to Google Sheets and the overview sheet is automatically updated with new totals.
Below is an example after the user has been logging expenses over the course of a few months. Expenses are added up and logged to the overview sheet.
If the user types 'q' at any of the checkpoints, the application displays a goodbye message and exits. Any relevant values are reset.
- gspread - for interacting with Google Sheets.
- Google Cloud Console - for Google Drive and Google Sheets APIs.
- os - for clearing the terminal face.
- google.oauth2.service_account for Credentials.
- art - for ASCII art.
- colorama - for text colours.
- prettytable - for tables.
- termcolor - used in title and heading colours.
- currencies - for currency symbols.
- amiresponsive - for the responsive png
- Code Beautify - for beautifying my Python Code.
- CI Python Linter - for detecting lint in my code.
After running the code through Code Beautify, the Code Institute Python Linter picked up on 3 unique errors:
- E501 line too long (103 > 79 characters)
- E711 comparison to None should be 'if cond is None
- E712 comparison to False should be 'if cond is False
The first is due to personal oversight, and was immediately corrected where the problem was highlighted.
The latter two were due a mix up of syntax between JS and Python, and were ammended.
A full overview can be seen below:
After fixing all issues highlighted by the linter, the code was run through the linter again, yielding no errors:
Note: This image reflects a more recent version of the code, which has undergone significant development compared to the initial image. To ensure code quality, it was thoroughly checked with a linter multiple times to identify and correct any issues. The image was captured prior to the final submission of the code. Any encountered errors were noted to be the same three issues, as outlined in the table below.
Issues | Fixes |
---|---|
E501 line too long (x > 79 characters) | This issue was fixed by using """ """ for multi-row strings, and by concisely rewording |
E711 comparison to None should be 'if cond is None | By changing '==' to 'is' |
E712 comparison to False should be 'if cond is False | By changing '==' to 'is' |
There were a few issues while the project was being developed, having mostly to do with string, integer and float use cases. In the image below, you can see the user's conclusive table, in which the month, budget, and expenses are detailed.
As in the current application, each category is detailed in a separate row. If the category was logged more than once by the user, the value of the expenses are added and appended to the existing category row. Before appending, the values are logged to a dictionary, detailing each of the individually logged expense values - and in one instance, as detailed in the 'Vehicle':'4324342' element. The issue is quite clear here - the expenses are pushed as strings and thus my attempt to add the values '432' and '432' resulted in '432432', instead of 864. This problem was fixed by converting the values into integers and then adding them (second image below, but with different values).
Another more significant issue arose when trying to log values that had a decimal point as expenses, i.e. 69.99 for Online Shopping. Please see image below.
This would trigger my number validation function, which checked and returned if a given input isdigit(). This function was modified to instead check if a given input was a float, as this accepts both integer values (e.g., 20.0), and floats. Please see image below.
This error was encountered after implementing a function that would clear the terminal screen. While it worked as intended in the VSCode terminal, it threw the error shown above in the Heroku terminal. After some research here, the reason for the bug was pinpointed being due to Heroku's Unix/Linux-like nature, whereas my function was designed only to handle terminal clearing on a Windows operating system. The first iteration of the function is shown below, with the improved version after it.
After running a conversion from one currency to another using the same values, the conversion values showed differences on average of .40 in several instances. This issue was likely due to the way the program was designed, where the user's logged expenses were not correctly rounded to 2 decimals.
A RatesNotAvailableError issue with the python-forex library for currency conversion was discovered on 17/02/24 once the application was nearing completion. This issue was due to the ratesapi.io being temporarily unavailable. Due to this issue and the project's nearing deadline, a decision was made to remove currency selection from the project - rendering much of its functionality described previously in this README obsolete.
This issue was highlighted in the library's repository in the 'issues' tab, and upon further investigation, it appears the API is taken down regularly for maintenance. Below is an image of correspondence for reference. This exchange is available for viewing here.
It is my intention to re-encorporate this logic into the application through the original API source following course completion.
While not necessarily an error, a warning was found on the terminal window regarding a deprecation of functionality associated with the gspread library.
While the warning was addressed (to the best of my abilities), the warning message failed to disappear. A useful link used while attempting to resolve this issue can be found here.
The application provides several instances of input validation, with relevant error messages and visual cues - as illustrated in the use of emojis:
β οΈ : signifies a checkpoint or warning message.- β€ : signifies an input prompt.
- β : failed validation.
- π : indicates the type of expected input.
- β : successful validation.
- π‘ : signifies insight/note/tip.
- β : signifies a pending request.
- π : signifies exiting. and colour cues in the conclusive user table (red and green for overdraft).
Overall, the application responds quickly, except where the budget remainder and conclusive user table are being formulated (generally takes ~20 seconds) for everything to print.
Please Note: The guide below details the steps in accordance with the original application deployment titled tag-track, however due to some billing issues the application fell asleep (as can be seen from the third image below) and later faded into a lighter purple. The application is now deployed using tag-track-revived, with the exact same steps.
The deployed application can be found at the top of this document.
- Your Python code must be placed in the
run.py
file. - Your dependencies must be placed in the
requirements.txt
file.
- Take caution if editing any of the other files or your code may not deploy properly.
- Log onto your Heroku dashboard.
- Set Up an authenticator for two-factor verification. I chose Google Authenticator because I already had it pre-installed on my phone.
- Create a new app as shown in the image below:
You will be asked to pick a name for your app, and select a region, before clicking Create app on the next page.
- Once you've created your app, go to the Deploy tab at the top.
Note: For demonstration purposes, I created a new app called 'gkggfk', but the process is the same everywhere.
- Select the middle box with GitHub's logo to connect your Heroku app to a Github Repository.
- If prompted, authorize Heroku to access your GitHub account.
- At the bottom, enter the name of the repository you wish to deploy to, and click Connect.
-
Under the "Automatic deploys" section, choose a branch from your GitHub repository that Heroku will watch for changes.
-
Enable automatic deploys by clicking Enable Automatic Deploys. With this, every push to the selected branch will automatically deploy a new version of your app.
When you create the app, you will need to add two buildpacks from the Settings tab. The ordering is as follows:
heroku/python
heroku/nodejs
You must then create a Config Var called PORT
. Set this to 8000
If you have credentials, you must create another Config Var called CREDS
and paste the relevant JSON into the value field.
-
If you've enabled automatic deploys, any push to the selected branch will automatically deploy your application.
-
If you prefer to deploy manually or want to deploy a branch without enabling automatic deploys, go to the "Manual deploy" section, select the branch, and click "Deploy Branch."
Your application has a similar look to the following Heroku URL: (https://.herokuapp.com), and can be found after clicking the Open App button on your dashboard in the top right.
The deployment terminal is set to 80 columns by 24 rows. That means that each line of text needs to be 80 characters or less otherwise it will be wrapped onto a second line.
To changes to your repository (or part of it) without affecting it's original state, you can 'fork' it (make a copy of it). This ensures the original repository remains unchanged. To fork a github repository, follow the following steps:
- Click into the github repository you want to fork.
- Click 'Fork' in the top right hand side of the top bar, and this should take you to a page titled 'Create a new fork'.
- You can now work in this copy of your repository without it affecting the original.
-
Cloning a repository essentially means downloading a copy of your repository that can be worked on locally. This method allows for version control and back up of code. To clone a github repository, follow the following steps:
-
Click into the github repository you want to clone.
-
Press the 'Code' button. This should open a section similar to the one below.
- Copy the link in this dropdown
- Open a terminal within your VSC (or whatever IDE you choose to use).
- In the terminal type 'git clone' and paste the URL.
- Press Enter - you now have a cloned version of your github repository.
Since the API crashed, I've had to remove some core logic from the application. It is my intention to re-encorporate it back into the application following course completion. The original code as it is from this commit message has been forked into a private repository for reference at a later stage.
Integrate more accurate Currency Conversions - when comparing forex-python rates with those from a quick google search, there appears to be a mismatch averaging around 20 units. I would like to incorporate some sort of API to handle this conversion for more accurate, real-time results.
It is my intention to develop the overview sheet to log all expenses for one month in the same currency. As things stand, only the category that gets an expense logged to it in a different currency is updated, whereas the entire month sheet gets it's values converted. This functionality was being developed when the currency conversion API crashed.
- Currencies (Research and two lines of logic).
- Emojis
- How to Use gspread
- Convert Num to Letter (stack overflow)
- Clear os screen
- gspread.exceptions.WorksheetNotFound
- screen recorder for README.md video files
- pypi.org for currency conversion example -snyk.io for Worksheet.NotFound
- Code Beautify for beautifying my python code
- CI Python Linter
- Forex Python for Currency Conversion
As always, big thank you to Harry Dhillon for mentoring me