My solutions to problems posed in "Automate The Boring Stuff with Python" by Al Sweigart
After a bunch of self directed projects, I think it is about time that I started following a book on the important components of coding in python. This was prompted by my overzealous attempt to take on multiple projects that were too big for me to handle. This file will be used as a short description of each project, bonus features that I may have added, and notes about what I learned from each project.
For those leafing through my code: this is largely a collection of my progress as a python programmer. The early projects may contain incorrect styling and a lack of comments and docstrings. I also have messed up some nomenclature where I called "functions" "methods" and vice versa. My later projects are a much better representation of where I am in my development path.
Chapter 3: Functions
- Project 1: collatz.py
- Used the collatz sequence to make a function that takes any positive integer as an argument and prints the collatz sequence (which always converges to 1, eventually).
- My first time learning about and using 'try' and 'except' statements.
- Bonus feature: I made it so the user can continue to test sequence with different integers and quit the program by typing in 'q' and pressing enter.
Chapter 4: Lists
-
Project 1: comma-code.py
- A simple program that turns a list into a string with commas in between each item and ', and' before the last item. This was mostly review and a nice warm up to using lists.
-
Project 2: coin-flip-streaks.py
- This program will count the amount of streaks of either heads or tails >= 6 in a list of 100 randomly generated "coin flips". This is performed 10,000 times to get a stastical idea of how often streaks occur. Another fun project with lists.
-
Project 3: character-picture-grid.py
- Simply printing an image with characters and the list data type in a 2x2 format. This problem was quite mind bending at first, but once I wrapped my head around how the loop was working with the list I got it down. Working with multi-dimensional systems can be difficult at first.
Chapter 5: Dictionaries & Structuring Data
-
Project 1: chess-dictionary-validator.py
- Checks a dictionary of locations (keys) and piece types (items) of a chess board in a dictionary and returns "True" if the board is valid
- First time learning about and using dictionaries. This project was very challenging at first, but proved to be another great coding experience.
- Bonus feature: I noticed the author of ATBS overlooked one criteria of validity for a chess board! Bishops are the only pieces that cannot exist on the same color. This was a pretty difficult feature to implement and it probably doubled my project time (worth it, though!). I made it so the function can detect if you have placed both of your bishops on white or both on black spaces. I also added a feature that prints a message to the user stating why the board is invalid, and simply states "Pass" if it is valid.
-
Project 2: fantasy-inventory.py
- A basic dictionary exercise, more use of items(), setdefault(), and other basic dictionary functions. This project was very short as the templates were provided.
Chapter 6: Manipulating Strings
-
Project 1: print-table.py
- Takes a list of lists and prints them in a neat right-justified format
- Learned some important methods to be used with strings and user input validation
- Bonus feature I added the ability for the function to put a title above the table in the format '-----title-----' (to the correct width of the table)
-
Project 2: myzombie.py
- I had way too much fun with this project. The author implemented a simulator for the game "zombie dice" and made it so you can make a bot and run it in tournaments against other bots.
- This project was my first time dealing with classes & methods directly, and although they are outside of the scope of this textbook I still ended up having to make my own method and learn a little bit about them.
- My Bot Bonus Feature: I started off by creating a bot that did the same thing as the "Stop at 2 shotguns bot" as it was consistently winning tournaments. I added a feature that takes into account the color of the die that have already been rolled and will roll again, even if the zombie has already been hit by two shotguns. Specifically, if all three of the red die (risky die) have already been rolled, my bot will take a chance on one more roll since only green and yellow die are in the cup. This intuition, however, did not beat the "stop at 2 shotguns" bot. I ran 10 tournaments of 1000 games and my bot came out on top 40% of the time with "stop at 2 shotguns" beating me out the other times. I don't know how Al's "zombiedice" package works, and if it doesn't actually reset the cup then my extra code doesn't do anything except make my bot do worse. I could look more into this later but I'm happy with calling this project complete.
Chapter 7: Pattern Matching with Regular Expressions
-
Follow along project: email-phone-extractor.py
- Bonus feature: I don't talk about the follow along projects unless I add something outside what is already given, so here it is. This program takes text from the clipboard and checks for emails and phone numbers. I added a feature (which probably could've just been a separate program) that detects different formats of dates and puts them all in the format of either mm/dd/yy or m/d/yy. I could go further and refine it to only do dd/mm/yy, but i'm okay with how far i've taken this feature since I had to wrestle with re.sub(), .findall(), and .search() plenty already.
-
Project 1: date-detection.py
- Turns out my bonus feature from the follow along project was a good warm up. This program detects dates from text provided by the clipboard.
- I made it so it can detect multiple dates with <.findall()> and then checks wether those dates are valid (assuming the author's assumption that the dates are in the format dd/mm/yyyy and only span years 1000 to 2999). It seems like the author wanted it to be a bit simpler and only detect one date, but I think making it detect all dates made it a fun challenge.
- Learned about list comprehension in order to delete invalid values. I did this in a bit of a roundabout way where I just replaced invalid dates with "0" while in the loop, then used list comprehension to delete any 0s. If I went deeper into this I would probably figure out how to use del, remove() or pop(), but I had the added complexity of having three lists that needed to correspond to eachother. So if I wanted to delete an invalid day I needed to figure out it's index and delete the corresponding "month". That being said, learning about list comprehension was pretty fun.
-
Project 2: detect-password-strength.py
- A quick exercise with more regular expression fun
- Tests a password for 8 characters, one lower case letter, one upper case letter, and one number. Returns 'strong' if it passes all of these tests.
-
Project 3: regex-strip.py
- A function that mimics the .strip() method for strings by using regular expressions
- learned how to insert a variable into a regex statement. This proved to be more difficult than expected and I ended up taking a few steps back and learning about strings in regex.
- Limitations: The function I wrote cannot strip a specific string in order. For example, 'bananas-austin-abanasab' will still strip to '-austin-' if 'bananas' is passed as the strip argument, because regex is matching any character in 'bananas'. I think this could be fixed by making a loop that generates a regex expression such as 'b.*a.*n.a.....' and uses this in the regex before continuing with the rest of the function.
Chapter 8: Input Validation with PyInputPlus
-
Project 1: sandwich-maker.py
- A really fun introduction to pyinputplus by making a program that asks users for sandwich preferences and then gives them a final cost
- Learned about the importance of data validation and how to use PyInputPlus to take a lot of weight off of my own code writing
- It was difficult getting a final cost together at the end, because all of the sandwich componenets were stored in different variables. If I were to start this project again, I would make a list where all of the ingredients are located. This would make the get_cost() function much easier & quicker to use.
-
Project 2: minimal-multiply-quiz
- This project is paired with the follow along project multiply-quiz.py
- minimal-multiply-quiz.py does the same thing as the follow along project, but without using pyinputplus (spoiler alert: pyinputplus does a lot of work for you in the background)
- The thing that I missed the most from pyinputplus was the "timeout" key word argument. Implementing timeout functionality took most of the development time while I wrapped my head around how to fit it into the loops.
Chapter 9: Reading and Writing Files
-
Project 1: mcb.py
- A multi-clipboard program that allows the user to save multiple pieces of text using the shelve module
- Learned how to use and modify shelf objects in order to save important python parameters
- Added features to clear and delete specific saved items. Once I figured out problems with using the command line format of the program, this was a quick solution.
-
Project 2: mad-lib.py
-
mad-lib.py will search for mad-lib-blank.txt and replace any all-caps parts of speech with user supplied input. It will then put the results into madLib-X.txt where X is a version number.
-
I actually had to go through a mini Software Development Life Cycle because I was merging multiple new concepts for myself (regex and file i/o).
-
I'm pretty proud of myself for condensing a massive train of if statments into a much smaller 4-line solution (actual code below). I've found that the more lines something takes to complete, the more skeptical of my approach to the problem. The solution here is much more simple and beautiful. I changed this code:
for word in keyword_regex.findall(contents): if word == 'ADJECTIVE': user_input = pyip.inputStr(prompt='Enter an adjective: ') contents = re.sub('ADJECTIVE', user_input, contents, count=1) elif word == 'VERB': user_input = pyip.inputStr(prompt='Enter a verb: ') contents = re.sub('VERB', user_input, contents, count=1) elif word == 'ADVERB': user_input = pyip.inputStr(prompt='Enter an adverb: ') contents = re.sub('ADVERB', user_input, contents, count=1) # Continues for different parts of speech...
Into the following:
for word in keyword_regex.findall(contents): statement = 'Enter a ' + word.lower() + ': ' user_input = pyip.inputStr(prompt=statement) contents = re.sub(word, user_input, contents, count=1)
-
-
Project 3: regex-search.py
- A program that takes a user-supplied regex expression and searches all .txt files. Any files that contain a match are printed to the user.
- Learned more about file writing, learned how to use glob() function to find any file ending in '.txt'
- This project took a few hours because I am just learning file io. I am amazed at how little code this is, while still yielding such useful results. What a fun small project!
Chapter 10: Organizing Files
-
Follow along notes: Before talking about my projects for this chapter, I first want to explain that I had quite some difficulty with Al's program for backing up and zipping files from a directory. His way took me quite some time to understand. Luckily, I found an example of zipping files on GeeksForGeeks which was much simpler. I ended up using the extremely useful function get_all_paths() in all of the projects in this chapter. Turns out it is much easier to separate each problem into (1) getting the paths you want to manipulate and then (2) doing whatever you want with them, instead of doing it all in two or three for loops using os.walk() as Al did. For those who would like to copy/paste, the function is as follows:
def get_all_paths(directory): ''' returns the path of each file within a directory, and every file within each sub-directory, and so on. ''' file_paths = [] for path, dirs, files in os.walk(directory): for filename in files: filepath = os.path.join(path, filename) file_paths.append(filepath) return file_paths
-
Project 1 selective_copy.py
- After wrestling with the follow-along project, this problem was straight-forward and enjoyable. selective-copy.py copies all files that match a user specified extension and puts them in the desired folder.
- I learned how to write zip files using the 'with' statement to handle any exceptions without creating bugs with writing, and make code more readable.
-
Project 2: find-large-files.py
- This project walks through the specified directory and returns filepaths with a specified length
- Learned about the different methods in the very useful os.path module
- A pretty simple extension of the get_all_paths() function implemented in selective-copy.py
-
Project 3: del-numbspace.py
- del-numbspace.py searches for files of the format text008.py (any extension) in a specified directory and renames them without the leading zeros
- Learned how to use the shell utility module, exercise more file organization skills, and even more regular expressions
Chapter 11: Debugging
- This chapter was ridiculously fun. I didn't realize there was a chapter on debugging and I thought I would have to learn it on my own, so it was nice to have it explained by the author.
- Debugged a file provided by the author, buggyCoinFlip.py that had a few bugs present. (No major projects to post about for this chapter.)
- Learned how to use logging and basic debugging features (Continue, Step Over, Step In, Step Out). I feel that these two skills will really make my code writing more efficient. I'm really excited to use this on more complicated programs.
Chapter 12: Web Scraping
-
Before I talk about projects, a few notes of what I learned from this chapter:
-
This chapter really showed me how interesting and diverse programming can be when I learned about linguistics and how it relates to the development of Unicode.
-
I had to exercise my true understanding of 'with' statements when I replaced one of the author's file writing statements with my own in xkcd-download.py. I changed the following:
imageFile = open(os.path.join('xkcd', os.path.basename(comicUrl)), 'wb') for chunk in res.iter_content(100000): imageFile.write(chunk) imageFile.close()
Into this chunk of code:
image_file = os.path.join('xkcd', os.path.basename(comic_url)) with open(image_file, 'wb') as file: for chunk in res.iter_content(100000): file.write(chunk)
I'm happy I was able to understand the use of the 'with' statement and apply it to this code. That being said, I don't think my block of code is better or more efficient (unless an exception is raised, which is unlikely in this part of the code), it's just a different way to do the same thing that I wanted to try.
-
This chapter was the toughest in terms of practice projects. I spent a lot of time figuring out webpage nuances and debugging my code. I'm starting to feel more comfortable with logging and debugging.
-
-
Project 1: emailer-cl.py
- This is a modification of the follow along project on Real Python to use command line arguments to send an email. The author suggested that I do this project with the selenium module, however, google would not let me login to my account with the lack of security so I had to do this follow along.
- Learned how to set up a secure connection to my coding email, login, and send mail
-
Project 2: play-2048.py
-
This project is a bot that plays 2048 by simply repeating up, right, down, left until the game is over
-
Learned more about interacting with webbpages with the selenium module. Also learned how to make the following try/except block:
while True: # --snip-- (play game here) try: selector = 'body > div.container > div.game-container > div.game-message.game-over' game_over = driver.find_element_by_css_selector(selector) break except NoSuchElementException: pass
And change it into this more readable block: (by importing suppress from contextlib)
while True: # --snip-- (play game here) with suppress(NoSuchElementException): selector = 'body > div.container > div.game-container > div.game-message.game-over' game_over = driver.find_element_by_css_selector(selector) break
This block detects when the 2048 game page prints the "game over" message, and stops the code accordingly. I thought this was a really cool tool to use, and I'm sure it will become useful later.
-
-
Project 3: link-verify.py
- link-verify.py contains a function that takes a url as an argument and checks that every link on the page is valid
- Learned how different links present themselves within a webpage (ones that navigate within the same website vs. outside the website, there's probably some better terminology here...)
-
Project 4: unsplash-download.py
- unsplash-download.py navigates to unsplash.com and downloads the first 3 images for the supplied search term
- This project was difficult for me. I originally tried to do this project first, but ended up having to do the other 3 above to give my mind a rest (which I'm glad I did). It was much easier after a few warm up projects.
- Learned how to better parse html with BeautifulSoup, and how to identify css selectors
Chapter 13: Working with Excel Spreadsheets
-
Project 1: blank_row_inserter.py
- This project takes an excel file and inserts 'm' blank rows at location 'n' (inbetween n and n+1)
- Learned how to move items within an excel file using python and save a copy
- Bonus Feature: Finally! I've been waiting for a good place to implement a bonus feature and this is the perfect spot. I added a feature that can detect excel equations and maintain their relative reference after being moved from the blank insertion. This feature ignores absolute references. I decided to add this after running a few example excel files, some of which contained equations which were left completely incorrect in their references after copying.
- This project really put into perspective how far I've come on my programming path. I'm excited to learn more.
-
Project 2: multiply_table_maker.py
- This project generates a simple multiplication table in an excel file
- More practice with openpyxl!
-
Project 3: cell_transpose.py
- A program that transposes all columns to rows in an excel file
- List comprehension practice, more practice with openpyxl and file writing
- Added more comments than usual on this one to clear up difficulties/confusion about transposing
-
Projects 4 & 5: txt_xlsx.py
- This program contains two functions: txt_to_xlsx which takes the contents of multiple text files and puts each line in a new row of an excel file, and xlsx_to_txt which does the reverse (only puts the data into one text file, though).
- A nice refresher on file reading and writing and more practice with openpyxl
Chapter 14: Working with Google Spreadsheets
-
A quick note: the projects in this chapter were extremely easy. The hard part of this chapter, and the part where I learned the most, is in setting up the google API. In fact, I decided to start with this textbook when I realized I was in over my head trying to write an email program 2 months before reaching this chapter. I didn't realize that there was a chapter that would help me with this problem, and I am now pleasantly surprised.
-
Project 1: download_forms.py
- Uses the ezsheets module to access the data acquired from a google form and extract the email addresses
- A very simple program that just required learning how to set up ezsheets so I could access the data in my gmail
- Learned how to use the ezsheets module and add APIs on my google account
-
Project 2: sheets_convert.py
- Let me just say, this is a funny project suggestion by the author. This project converts different spreadsheet file types by 'hijacking' google sheets into doing it for you
- This project was pretty simple, I learned how to use basic methods of the ezsheets module
-
Project 3: find_mistakes.py
- This project parses through a public google sheet posted by Al, and determines which row has a mathematical error (it's row 14399)
- Another quick project to get some more practice using ezsheets
Chapter 15: Working with PDF and Word Documents
-
Project 1: pdf_paranoia.py
- This project contains the function encrypt_pdfs() which walks through a specified file path and encrypts all unencrypted .pdf files and renames them with the suffix 'encrypted.pdf'. It also contains decrypt_pdfs() which does the opposite and adds the suffix 'decrypted.pdf'
- Learned how to copy pdf files using the PyPDF2 module
- One of the biggest projects I've finished thus far and I'm very proud of it. Copying, encrypting, and checking that it all worked for pdfs was pretty intense.
-
Project 2: custom_invitation.py
- A program that creates custom invitations using a .txt supplied list of names. This puts them all in the same word document on separate pages.
- Learned how to use the docx module to add text, change paragraph style and alignment, and add new pages in a word document
-
Project 3: pdf_brute_force.py
- This program uses a dictionary of all english words and attempts to decrypt a pdf with both lower and upper case versions of each word (A total of about 90k words). If the function is able to decrypt the pdf, it will print the password to the user.
- Combined my knowledge of file reading, dictionaries, and the PyPDF2 module to decrypt a pdf file! Very fun!
Chapter 16: Working with CSV files and JSON Data
- Follow Along Project: getOpenWeather.py
- A fun follow-along project which taught me how to work with JSON data and the openweathermaps API
- Learned more about using APIs. I learned how to store my API key in a seperate file to keep it secret.
- Working with JSON data is pretty simple in python (using only the json.loads() and json.dumps() functions) so there weren't any suggested projects for this chapter that use them
- Project 1: excel_to_csv.py
- Uses the csv and openpyxl modules to read excel files in a given directory and convert them into csv files. Since csv files cannot handle multiple sheets, they are stored in individual files in the format excelfilename_sheetname.csv
- Learned about the convenience of csv files and how they can be used with python
Chapter 17: Keeping Time, Launching Tasks, and Scheduling Programs
-
Follow-along project notes:
- The first follow-along project involved implementing a previous project, xkcd-download.py, except with threading to make it even faster. I went ahead and timed the two programs, each downloading the first 40 comics. The non-threaded version took ~34 seconds, and the threaded version took ~14 seconds. A pretty simple addition in the code that would save a lot of time if I were to download a couple thousand images. This project is reminiscent of my parallel GPU project that I worked on in undergrad. Very cool!
-
Project 1: prettified_stopwatch.py
- An extension of a follow-along project in the chapter. After doing the follow along project, this was a pretty simple exercise in using the time module.
- Learned basics of time keeping with the time module
-
Project 2: scheduled_comic.py
- A fun program that uses the requests and BeautifulSoup modules to download any new comics that appear on the web comic lunarbaboon.com (I just picked a random one).
- Using Windows Time Scheduler, this program is run once a day and saves the new files to the desktop so they are easy to see.
- Learned how to use the windows time scheduler with Python programs!
Chapter 18: Sending Email & Text Messages
-
Project 1: random_chore.py
- This project takes a random chore from a list of chores and assigns it to a user, then sends an email to that person alerting them which chore they have for the week
- Learned how to use the ezgmail module to send emails from my gmail. This module is much easier to use for basic functions instead of directly working with the google API
-
Project 2: umbrella_reminder.py
- Checks the local weather using the getOpenWeather API, then sends me a text reminding me to bring an umbrella if rain is in the forecast
- Learned how to use Twilio to create text alerts in my python programs
-
Project 3: email_unsubscriber.py
- Looks through email for any items that contain the word "unsubscribe" then opens the unsubscribe links in the webbrowser so the user can click through each tab and unsubscribe to many emails at once!
- Learned how to access email data through the imaplib module. More practice with Beautiful Soup. Overall a very rewarding project
- Limitations: This project had difficulty with some email links and was not able to open all of the ones in my inbox. I suspect that the diversity of these emails from different companies may be the culprit and I definitely foresaw some issues like this. That being said, I am going to call this project sufficient in doing it's job and continue on to the next chapter.
Chapter 19: Manipulating Images
-
Project 1 / Follow along project: add_logo.py
- A program that adds a logo to the bottom right corner of every image in the specified directory. Also resizes all images so that the logo is the same size in the corner.
- This is a follow along project that the author suggested adding a few features to as the first project for this chapter. I created the following changes:
- All images are resized such that the logo is 10 times smaller than the selected photos
- The function can identify .gif and .bmp files in addition to .jpg and .png
- Restructured the program into functions to make it easier to read, and so I can reuse the resize_image function later
-
Project 2: find_photo_folders.py
- find_photo_folders.py searches all photos in the given directory and will return the filepath if more than 50% of the files are images that are larger than 500 pixles square
- This was a good project and gave me some more practice with structuring dictionaries, and working with the Pillow module. This was a pretty simple program in all.
-
Project 3: custom_seat.py
- This project creates custom PNG files to be used as seating cards for a party. It gets the names of each attendee from guests.txt and creates an image with their name and a flower decoration. The image has a black border to assist in cutting once the image is printed out.
- Learned how to use the pillow module to add custom text, paste an image onto another, and center text on the page.
Chapter 20: Controlling the Mouse & Keyboard with GUI Automation
-
Follow-Along Project notes/Bonus Feature?: The follow-along project for this chapter, form_filler.py was a very fun exercise in automatically filling out a google form with a bunch of different data using GUI automation. I simplified some lines of code that I wanted to share here. The program the author had listed had the following repetitive 'if' block used to fill out a scale of radio buttons (both code and form pictured below):
if person['robocop'] == 1: pyautogui.write([' ', '\t'] , 0.5) elif person['robocop'] == 2: pyautogui.write(['right', '\t'] , 0.5) elif person['robocop'] == 3: pyautogui.write(['right', 'right', '\t'] , 0.5) elif person['robocop'] == 4: pyautogui.write(['right', 'right', 'right', '\t'] , 0.5) elif person['robocop'] == 5: pyautogui.write(['right', 'right', 'right', 'right', '\t'] , 0.5)
This block is repetitive because it is repeating the the 'right' command depending on the number that is stored in the dictionary. I figured this could easily be reduced to one line by using the stored number to calculate the amount of times the 'right' command should be passed. I did the following:
pyautogui.write(['\t', ' '] + ['right'] * (person['robocop'] - 1))
Please ignore that my 'tab' command is in a different place than the above code because I wrote my program differently.
-
Project 1: mouse_nudge.py
- mouse_nudge.py subtly moves the mouse back and forth by one pixel once every minute to keep the user looking busy (on, say, microsoft teams)
- This is an extremely simple program to dip my toes in the water with the Pyautogui module. It is only 5 lines of code, but could be useful in the future.
-
Project 2: read_clipboard.py
- pyautogui is useful for entering in data, but what if the user needs to be alerted if data is already in a text box? read_clipboard.py uses pyperclip and pyautogui to select text in a notepad and obtain the text
- learned the basics of pyautogui window methods and clicking methods
-
Project 3: hangouts_bot.py
-
hangouts_bot.py opens Google Hangouts in a new webbrowser, creates a group of my select friends (I imported their names from a separate file. See: import friends), then sends the group a message
-
This was a challenging program to write. I ran into many problems with the webbpage not loading fast enough, or different buttons popping up that my program wasn't equipped to detect. One extremely helpful solution that I implemented was the following function to pause the program until certain page elements appeared.
def waitForImage(img_name): ''' waitForImage will pause the program until the passed image file appears on the screen ''' while True: element = pyautogui.locateOnScreen(img_name) if element is not None: return
-
Overall, a very fun and challenging project for learning how to use the Pyautogui module.
-
-
Project 4: Sushi Go Round Bot
- This was a great way to finish the textbook. This program uses Pyautogui (and more) to make a bot that can play Sushi Go Round and was made with the help of this tutorial.
- I learned a lot going through this project. First off, this is my first real project where I am dipping my toes into true OOP principles. I used classes to improve readibility and created a bunch of functions that were then executed in a more beautiful structure than my single function scripts from the previous chapters. Please note that this program may be a bit messy for this reason. I organized it in the following manner: globals, classes, functions, main script.
- I greatly improved the performance of the bot by adding a timing mechanism which would keep track of customers who had already been served. The bot in the tutorial just made a roll for every customer that requested one and because of the delay in the roll reaching them, it would make the same roll too many times.
- Limitations: my bot is only built to work through the first level. More logic is required for those that would like to extend the bot to the next levels.
- Usage:
- Run the setup() method- this will open a browser and scroll down a specific distance on the webpage
- Determine the pixel location of the top-left corner of the game play area. Put those values into x_pad and y_pad. This can be done by easily running pyautogui.mouseInfo()
- You're good to go! Please reach out if you would like to try it out or have any issues
Conclusion
Doing every suggested project in "Automate the Boring Stuff with Python" by Al Sweigart was both challenging and rewarding. I feel that I have been properly introduced to Python now- I can read and write files, edit images, scrape the web, and more. In addition to these skills, I was often forced to look at the bigger picture as a programmer often has to do: break the problem down into workable chunks, evaluate how those fit into the minimally viable product, and then execute. After reading through this book, I've come to the point that many beginners often do: the realization that there is so much more to learn. Being hungry for even more, I have a small stack of textbooks so that I can continue my education in OOP principles, data structures & algorithms, and powerful frameworks used by Python. Thank you everyone for your support in my software journey.