All posts by steve@usefulscripts.co

Google Gemini: chat script

For my Gemini scripts you will need a Gemini API key: don’t worry, they’re easy to obtain. Go HERE to apply for your key. At the time of writing, some users may need to use a US proxy to obtain the API key. If the link doesn’t work, Google “Gemini API key” for the correct page.

This script allows you to send sequential linked queries to Gemini in a chat format, drawn from the chat.csv file. The answers are saved in the chatoutput.csv file. Unlike this script, this script is a continuous conversation, so the prompts are responded to as part of the same chat.

Usage

This is similar to the ChatGPT script, except that this does not use the OpenAI API, and therefore costs nothing to run. It does require a bit more setting up. Being able to send queries automatically to AI Chatbots means you can create high volumes of good quality content without intervention. For example, you can use this script to ask Gemini to write a story outline, and then ask it to completre each chapter, prompt by prompt.

Prerequisites

  1. You need a Gemini account
  2. You need a Gemini API key. Apply here: https://aistudio.google.com/app/apikey
  3. Now install the Gemini API
pip install -q -U google-generativeai

How to use it

  1. Save the script below in the folder of your choice.
  2. Edit the script and insert your API key
  3. Create an chat.csv file, which contains the prompts in the first column to be sent to Gemini
  4. The output is saved as chatoutput.csv
import pathlib
import textwrap
import csv
import time
import google.generativeai as genai

# THIS PRODUCES LINKED CONVERSATIONAL CHATS

#  !!! CAUTION: Hardcoding API keys is highly discouraged !!!
api_key = "YOUR-API-KEY"
genai.configure(api_key=api_key)
model = genai.GenerativeModel('gemini-pro')
chat = model.start_chat(history=[])
chat

# Rate limiting (adjust sleep time if needed)
RATE_LIMIT_SLEEP_SECONDS = 5  

def process_query(query):
    """Sends a query to Gemini and handles potential rate limiting"""
    try:
        response = chat.send_message(query)
        return response.text
    except genai.exceptions.RateLimitExceededError as e:
        print(f"Rate limit exceeded. Waiting {RATE_LIMIT_SLEEP_SECONDS} seconds...")
        time.sleep(RATE_LIMIT_SLEEP_SECONDS)
        return process_query(query)  # Retry the query

# Load queries from input.csv
input_filepath = pathlib.Path("chat.csv")
with input_filepath.open('r', newline='', encoding='utf-8') as csvfile:
    reader = csv.reader(csvfile)
    queries = list(reader)  

# Process queries and save responses
output_filepath = pathlib.Path("chatoutput.csv")
with output_filepath.open('w', newline='', encoding='utf-8') as csvfile:
    writer = csv.writer(csvfile)
    for query in queries:
        response = process_query(query[0])  # Assume query in the first column
        writer.writerow([query[0], response])
        print(f"Query: {query[0]}\nResponse: {textwrap.shorten(response, width=80)}\n")

Google Gemini: non-chat script

For my Gemini scripts you will need a Gemini API key: don’t worry, they’re easy to obtain. Go HERE to apply for your key. At the time of writing, some users may need to use a US proxy to obtain the API key. If the link doesn’t work, Google “Gemini API key” for the correct page.

This script allows you to send sequential queries to Gemini, drawn from the input.csv file. The answers are saved in the output.csv file. This is not a chat, so the prompts are responded to independently.

Usage

This is similar to the ChatGPT script, except that this does not use the OpenAI API, and therefore costs nothing to run. It does require a bit more setting up. Being able to send queries automatically to AI Chatbots means you can create high volumes of good quality content without intervention. For example, you can use this script to ask Gemini to write a product review in one shot, and then repeat the prompt for different products.

Prerequisites

  1. You need a Gemini account
  2. You need a Gemini API key. Apply here: https://aistudio.google.com/app/apikey
  3. Now install the Gemini API
pip install -q -U google-generativeai

How to use it

  1. Save the script below in the folder of your choice.
  2. Create an input.csv file, which contains the prompts in the first column to be sent to Gemini
  3. The output is saved as output.csv
import pathlib
import textwrap
import csv
import time

import google.generativeai as genai

#  !!! CAUTION: Hardcoding API keys is highly discouraged !!!
api_key = "YOU_API_KEY"
genai.configure(api_key=api_key)
model = genai.GenerativeModel('gemini-pro')

# Rate limiting (adjust sleep time if needed)
RATE_LIMIT_SLEEP_SECONDS = 5  

def process_query(query):
    """Sends a query to Gemini and handles potential rate limiting"""
    try:
        response = model.generate_content(query)
        return response.text
    except genai.exceptions.RateLimitExceededError as e:
        print(f"Rate limit exceeded. Waiting {RATE_LIMIT_SLEEP_SECONDS} seconds...")
        time.sleep(RATE_LIMIT_SLEEP_SECONDS)
        return process_query(query)  # Retry the query

# Load queries from input.csv
input_filepath = pathlib.Path("input.csv")
with input_filepath.open('r', newline='') as csvfile:
    reader = csv.reader(csvfile)
    queries = list(reader)  

# Process queries and save responses
output_filepath = pathlib.Path("output.csv")
with output_filepath.open('w', newline='') as csvfile:
    writer = csv.writer(csvfile)
    for query in queries:
        response = process_query(query[0])  # Assume query in the first column
        writer.writerow([query[0], response])
        print(f"Query: {query[0]}\nResponse: {textwrap.shorten(response, width=80)}\n")

a person holding a cell phone in their hand

Bard – DEPRECATED

Bard has been superseded by Google Gemini – you can find my Gemini scripts in the AI category.

This script allows you to send sequential queries to Bard, drawn from the input.csv file. The answers are saved in the output.csv file.

Usage

This is similar to the ChatGPT script, except that this does not use the OpenAI API, and therefore costs nothing to run. It does require a bit more setting up. Being able to send queries automatically to AI Chatbots means you can create high volumes of good quality content without intervention. For example, you could use this script to ask Bard to create an outline for a book, and then ask it to write each chapter in sequence. This tends to produce better quality responses than asking it to write a book in one go. Alternatively you could instruct it to read a document and then provide an analysis, step by step.

Prerequisites

  1. You need a Bard account
  2. Once you have the account you will need to log in with a Chrome browser and locate the __Secure-1PSID cookie. You do this by right clicking anywhere on the Bard page once you’ve logged in then click “Inspect”. In the console that opens, click on the “Application” tab, and in the “Storage” window, click on the Cookie dropdown. Now click on the https://bard.google.com item which should present all the cookies for that page. Copy the long string associated with the __Secure-1PSID cookie. (Be careful to choose the right one). Paste that into the script at the point shown.
  3. Now install the Bardapi and requests libraries
pip install requests
pip install bardapi

How to use it

  1. Save the script below in the folder of your choice.
  2. Edit the file and insert your Bard cookie string
  3. Also edit the prompts and timings if desired – you can change the speed of responses. In this script previous questions and answers are provided as context.
  4. Create an input.csv file, which contains the prompts in the first column to be sent to OpenAI.
  5. The output is saved as output.csv
import csv
from bardapi import Bard
import time
import os
import re
import requests

# Set up a reusable session
session = requests.Session()
session.headers = {
    "Host": "bard.google.com",
    "X-Same-Domain": "1",
    "User-Agent": "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/91.0.4472.114 Safari/537.36",
    "Content-Type": "application/x-www-form-urlencoded;charset=UTF-8",
    "Origin": "https://bard.google.com",
    "Referer": "https://bard.google.com/",
}

# Set cookies for the session
token = 'your_Bard_cookie_very_long_string_of_digits_etc_etc_etc.'  
session.cookies.set("__Secure-1PSID", token)

# Create a Bard instance with the reusable session
bard_instance = Bard(token=token, session=session, timeout=30)

# Open input and output CSV files with error handling
input_file = open('input.csv', 'r', encoding='utf-8', errors='replace')
output_file = open('output.csv', 'w', encoding='utf-8', newline='')

# Create CSV writers
input_csv = csv.reader(input_file)
output_csv = csv.writer(output_file)

# Write headers to output CSV
# output_csv.writerow(['Prompt', 'Response'])

# Set the desired API call rate (2 calls per minute)
calls_per_minute = 1
interval = 60 / calls_per_minute

# Regular expression pattern to match file paths in prompts
file_path_pattern = r"C:/Users/Steve/.*?\.txt"

context = ""

# Iterate through prompts and generate responses
for row in input_csv:
    prompt = row[0]

    # Check if the prompt contains a file path
    file_paths = re.findall(file_path_pattern, prompt)

    if file_paths:
        # Upload and generate a response for each file path found
        for file_path in file_paths:
            # Read the content of the file with the specified encoding and error handling
            with open(file_path, 'r', encoding='utf-8', errors='replace') as file:
                file_content = file.read()

            # Replace the file path with the content in the prompt
            prompt = prompt.replace(file_path, file_content)

    if not context:
        # If context is empty, set the prompt without the previous conversation context
        full_prompt = "Please answer this question: " + prompt
        print("Prompt:", full_prompt)
    else:
        # Otherwise, include the previous conversation context
        full_prompt = (
            "Please consider the previous conversation we have had, which I am recording for you here within the <context> tags: <context> "
            + context
            + " </context> Now, bearing in mind the conversation so far within the context tags which you have already responded to - no need to answer any of those questions again -  please answer this question: "
            + prompt
        )
        print("Prompt:", full_prompt)

    # Send an API request and get a response - print to check.
    response = bard_instance.get_answer(full_prompt)
    response_content = response['content']
    print(response_content)

    # Update context after each loop - depracated in Bard
    # context = response_content

    # Write to file
    output_csv.writerow([response_content])

    # Introduce a delay to limit the rate of API calls
    time.sleep(interval)

# Close files
input_file.close()
output_file.close()

a green square with a white knot on it

ChatGPT

This script allows you to send sequential queries to ChatGPT, drawn from the input.csv file. The answers are saved in the output.csv file.

Usage

Being able to send queries automatically to AI Chatbots means you can create high volumes of good quality content without intervention. For example, you could use this script to ask ChatGPT to create an outline for a book, and then ask it to write each chapter in sequence. This tends to produce better quality responses than asking it to write a book in one go. Alternatively you could instruct it to read a document and then provide an analysis, step by step.
In this script there is an estimate of the cost of the work, allowing you to manage your financial commitment. (Don’t worry though – as a new user of OpenAI you receive free credits, and even after they have been consumed, the costs are low.)

Prerequisites

  1. You need an account with OpenAI – click here to get started
  2. Once you have the account you will need your own API key. At the moment, you can create one at this link, although the link may change in future. (If in doubt use Google to help you find the API page.)
  3. Now install the openai library
pip install openai

How to use it

  1. Save the script below in the folder of your choice.
  2. Edit the file and insert your OpenAI API key
  3. Also edit the model if desired (in this script the model used is “gpt-4-1106-preview”. This may not be available in future, so please check with OpenAI)
  4. Create an input.csv file, which contains the prompts in the first column to be sent to OpenAI.
  5. The output is saved as output.csv
import csv
import openai
import time
# Try importing OpenAIError using the package's full path
from openai import OpenAIError

# Replace 'YOUR_API_KEY' with your actual OpenAI API key
openai.api_key = 'YOUR API KEY'

# PRICING
# gpt-4-1106-preview	$0.01 / 1K tokens	$0.03 / 1K tokens
# gpt-4-1106-vision-preview	$0.01 / 1K tokens	$0.03 / 1K tokens
# gpt-4	$0.03 / 1K tokens	$0.06 / 1K tokens
# gpt-4-32k	$0.06 / 1K tokens	$0.12 / 1K tokens
# gpt-3.5-turbo-1106	$0.0010 / 1K tokens	$0.0020 / 1K tokens
# gpt-3.5-turbo-instruct	$0.0015 / 1K tokens	$0.0020 / 1K tokens

# more here: https://openai.com/pricing


# Cost per token information (per thousand tokens)
input_token_cost_per_thousand = 0.01
output_token_cost_per_thousand = 0.03
###########################################



# Function to interact with ChatGPT and save responses to a CSV file with retries and rate limiting
def process_queries_with_retries_and_rate_limit(input_file, output_file, max_retries=3, retry_delay=5, max_requests_per_minute=60):
    with open(input_file, 'r') as input_csv, open(output_file, 'w', newline='') as output_csv:
        input_reader = csv.reader(input_csv)
        output_writer = csv.writer(output_csv)
        output_writer.writerow(['Input', 'Response', 'Input Tokens', 'Output Tokens', 'Total Cost'])
        
        conversation_state = []  # List to maintain conversation state including system and user messages
        total_cost = 0.0
        
        # Calculate how many seconds to wait between requests to stay under the rate limit
        min_time_between_requests = 60.0 / max_requests_per_minute

        last_request_time = None

        for row in input_reader:
            if not row:
                continue

            input_query = row[0]
            print(f"Processing query: {input_query}")
            
            for retry in range(max_retries):
                try:
                    # Implement rate limiting logic
                    if last_request_time is not None:
                        elapsed_time = time.time() - last_request_time
                        if elapsed_time < min_time_between_requests:
                            time_to_wait = min_time_between_requests - elapsed_time
                            print(f"Rate limited. Waiting for {time_to_wait:.2f} seconds.")
                            time.sleep(time_to_wait)
                    
                    response = openai.ChatCompletion.create(
                        model="gpt-4-1106-preview", ###EDIT HERE TO CHANGE THE LANGUAGE MODEL
                        messages=conversation_state + [
                            {"role": "user", "content": input_query}
                        ]
                    )
                    
                    if 'choices' in response and response['choices']:
                        # Append the latest user and assistant messages to the conversation state
                        conversation_state.append({"role": "user", "content": input_query})
                        conversation_state.append(response['choices'][0]['message'])

                    output_tokens = response['usage']['total_tokens']
                    input_tokens = response['usage']['prompt_tokens']
                    output_cost = (output_tokens / 1000) * output_token_cost_per_thousand
                    total_cost += output_cost

                    output_writer.writerow([
                        input_query,
                        response['choices'][0]['message']['content'],
                        input_tokens,
                        output_tokens,
                        output_cost
                    ])

                    last_request_time = time.time()
                    break
############

  # Catch OpenAI's API exception using the correct reference
                except OpenAIError as e:
                    print(f"OpenAI API Error: {e}")
                    time.sleep(retry_delay)
                    if retry == max_retries - 1:
                        print(f"Max retries reached for query: {input_query}. Skipping.")

                
        print(f"Total cost of the routine: ${total_cost:.2f}")

if __name__ == "__main__":
    input_file_path = "input.csv"
    output_file_path = "output.csv"

    process_queries_with_retries_and_rate_limit(input_file_path, output_file_path)


red and white square illustration

Video Renamer

This script allows you to rename and edit metadata in .mp4 video files, in bulk.

Usage

Optimising videos for ranking for specific keywords on YouTube requires various elements to be configured for best effect. The video file name should include the keyword, as well as the file meta data. If you’re creating videos in bulk, this can be a time-consuming task. But this script takes the pain away!

Prerequisites

You will need to install the “moviepy” and “mutagen” libraries.

pip install moviepy
pip install mutagen

How to use it

  1. Save the script below in the folder of your choice.
  2. Create a new folder called “Videos”, and save all the videos that you want to optimise there.
  3. Open the script in Notepad++, and replace the “INSERT YOUR KEYWORD HERE” sections with your own keyword(s). You can optionally add additional tags in the tags section (TAG1, TAG2 etc, separated by commas).
  4. Run your script – your videos will be replaced by the modified videos.
import os
from moviepy.editor import VideoFileClip
from mutagen.mp4 import MP4

# Specify the folder where your mp4 videos are located
folder_path = "Videos"

# Specify the keyword you want to add to the video names
keyword = "INSERT YOUR KEYWORD HERE"

# Get a list of all files in the folder
files = os.listdir(folder_path)

# Loop through the files, edit metadata, and rename them
for i, file_name in enumerate(files):
    if file_name.endswith(".mp4"):
        # Generate the new file name with the keyword and index
        new_file_name = f"{keyword}-{i + 1}.mp4"

        # Construct the full paths to the old and new files
        old_file_path = os.path.join(folder_path, file_name)
        new_file_path = os.path.join(folder_path, new_file_name)

        # Edit metadata
        clip = VideoFileClip(old_file_path)

        # Create a temporary file with the edited video
        temp_file_path = os.path.join(folder_path, f"temp_{i + 1}.mp4")
        clip.write_videofile(temp_file_path, audio=True)

        # Set metadata using mutagen
        mp4 = MP4(temp_file_path)
        mp4['\xa9nam'] = "INSERT YOUR KEYWORD HERE"  # Title
        mp4['\xa9alb'] = "INSERT YOUR KEYWORD HERE"  # Subtitle
        mp4['\xa9gen'] = "INSERT YOUR KEYWORD HERE,TAG1, TAG2, TAG3"  # Tags
        mp4['\xa9cmt'] = "INSERT YOUR KEYWORD HERE"  # Comments
        mp4.save()

        # Remove the old file
        os.remove(old_file_path)

        # Rename the temporary file to the new name
        os.rename(temp_file_path, new_file_path)

        print(f"Edited and Renamed: {file_name} -> {new_file_name}")

Youtube icon

Top And Tail Video Creator

This program concatenates videos together, adding topvideo.mp4 at the start of any video, and tailvideo.mp4 at the end. It does this in bulk for any number of videos.

Usage

This is great for branding your videos with an opening (“top”) and closing (“tail”) sequence.

Prerequisites

Create your opening and closing videos, and name them “topvideo.mp4” and “tailvideo.mp4”.
Now save the videos you want to brand in a folder named “Videos”. Your branded videos will be created and saved in a new folder called “Completed Videos”

You will need to install the “moviepy” library.

pip install moviepy

How to use it

  1. Save the script below in the folder of your choice
  2. Create a new folder called “Videos”, and save all the videos that you want branding with a top and tail video.
  3. Make sure you have the topvideo.mp4 and tailvideo.mp4 saved in the main folder
  4. Run your script – your new videso will be found in the “Completed Videos” folder.
import os
from moviepy.editor import VideoFileClip, concatenate_videoclips

# Define the input folder for videos and the top and tail videos
videos_folder = "Videos"
top_video_path = "topvideo.mp4"
tail_video_path = "tailvideo.mp4"

# Create the output folder if it doesn't exist
output_folder = "Completed Videos"
os.makedirs(output_folder, exist_ok=True)

# Function to get video codec and resolution
def get_video_info(file_path):
    video = VideoFileClip(file_path)
    codec = video.fps
    resolution = video.size
    video.close()
    return codec, resolution

# Get the codec and resolution of the first video in the folder
first_video_path = os.path.join(videos_folder, os.listdir(videos_folder)[0])
first_video_codec, first_video_resolution = get_video_info(first_video_path)

# Load the top video and tail video
top_video = VideoFileClip(top_video_path)
tail_video = VideoFileClip(tail_video_path)

# Check if the top video codec and resolution match the first video
if top_video.fps != first_video_codec or top_video.size != first_video_resolution:
    # If not, convert the top video to match
    top_video = top_video.set_fps(first_video_codec).resize(first_video_resolution)

# Check if the tail video codec and resolution match the first video
if tail_video.fps != first_video_codec or tail_video.size != first_video_resolution:
    # If not, convert the tail video to match
    tail_video = tail_video.set_fps(first_video_codec).resize(first_video_resolution)

# List to store concatenated videos
concatenated_videos = []

# Loop through all files in the input folder
for filename in os.listdir(videos_folder):
    if filename.endswith(".mp4"):
        # Load the current video
        video_path = os.path.join(videos_folder, filename)
        video = VideoFileClip(video_path)

        # Concatenate the top video, current video, and tail video
        concatenated_video = concatenate_videoclips([top_video, video, tail_video])

        # Set the output filename
        output_filename = os.path.splitext(filename)[0] + "-edited.mp4"
        output_path = os.path.join(output_folder, output_filename)

        # Write the concatenated video to the output folder
        concatenated_video.write_videofile(output_path, codec="libx264", fps=video.fps)

        # Close the video objects
        video.close()
        concatenated_video.close()

# Close the top and tail video objects
top_video.close()
tail_video.close()

Remember to ensure that the top and tail videos match the format of the videos you are branding.

mic, microphone, sound check

Ultimate Text To Speech Converter

This program uses a multiple column .csv file to create .mp3 audio files, row by row, alternating voices.

Usage

Ever wanted to create a conversational style of audio file? This script allows you to do that. The script alternates voice selection by column, so by setting different voices you can create the illusion of a conversation!

Prerequisites

You will need to install the “edge TTS” and “pydub” libraries.

pip install edge-tts
pip install pydub

How to use it

  1. Save the script below in the folder of your choice
  2. Rename the file you want to convert to audio as “input.csv” and save it in the same folder as the script. Each row will be converted to a separate audio file.
  3. Run the script
  4. The audio files are saved in the “output” folder
import asyncio
import edge_tts
import csv
import os
import subprocess

async def process_text(text, voice, output_filename):
    try:
        communicate = edge_tts.Communicate(text, voice)
        await communicate.save(output_filename)
    except RuntimeError as e:
        print(e)

async def main():
    # Read CSV
    with open('input.csv', 'r') as csv_file:
        csv_reader = csv.reader(csv_file)
        rows = list(csv_reader)

    voices = ["en-GB-SoniaNeural", "en-GB-ThomasNeural"]
    output_folder = "output"

    if not os.path.exists(output_folder):
        os.makedirs(output_folder)

    for i, row in enumerate(rows, start=1):
        concatenated_audio = []

        for j, cell_text in enumerate(row):
            if cell_text.strip():
                voice = voices[j % len(voices)]
                output_filename = os.path.join(output_folder, f"{i}_{j}.mp3")
                
                try:
                    await process_text(cell_text, voice, output_filename)
                    concatenated_audio.append(output_filename)
                except RuntimeError as e:
                    print(e)

            await asyncio.sleep(0.1)  # Small gap between elements

        if concatenated_audio:
            combined_filename = os.path.join(output_folder, f"{i}.mp3")
            
            # Concatenate audio using ffmpeg
            concat_command = [
                "ffmpeg",
                "-i",
                f"concat:{'|'.join(concatenated_audio)}",
                "-c",
                "copy",
                combined_filename
            ]
            subprocess.run(concat_command, check=True)
            
            # Delete individual element audio files
            for audio_file in concatenated_audio:
                os.remove(audio_file)

    print("Finished")

loop = asyncio.new_event_loop()
loop.run_until_complete(main())
loop.close()

This script uses “Sonia” and “Thomas”, UK accented voices. At the end of this post I provide the complete list of Edge TTS voices: just replace “en-GB-SoniaNeural” and “en-GB-ThomasNeural” with your preferred voices.

Edge TTS Voices

How to read the voice labels:
EXAMPLE: en-IN-NeerjaNeural, Gender: Female
en – means the voice speaks English
IN – means the accent is Indian

Name: af-ZA-AdriNeural, Gender: FemaleName: af-ZA-WillemNeural, Gender: Male
Name: am-ET-AmehaNeural, Gender: MaleName: am-ET-MekdesNeural, Gender: Female
Name: ar-AE-FatimaNeural, Gender: FemaleName: ar-AE-HamdanNeural, Gender: Male
Name: ar-BH-AliNeural, Gender: MaleName: ar-BH-LailaNeural, Gender: Female
Name: ar-DZ-AminaNeural, Gender: FemaleName: ar-DZ-IsmaelNeural, Gender: Male
Name: ar-EG-SalmaNeural, Gender: FemaleName: ar-EG-ShakirNeural, Gender: Male
Name: ar-IQ-BasselNeural, Gender: MaleName: ar-IQ-RanaNeural, Gender: Female
Name: ar-JO-SanaNeural, Gender: FemaleName: ar-JO-TaimNeural, Gender: Male
Name: ar-KW-FahedNeural, Gender: MaleName: ar-KW-NouraNeural, Gender: Female
Name: ar-LB-LaylaNeural, Gender: FemaleName: ar-LB-RamiNeural, Gender: Male
Name: ar-LY-ImanNeural, Gender: FemaleName: ar-LY-OmarNeural, Gender: Male
Name: ar-MA-JamalNeural, Gender: MaleName: ar-MA-MounaNeural, Gender: Female
Name: ar-OM-AbdullahNeural, Gender: MaleName: ar-OM-AyshaNeural, Gender: Female
Name: ar-QA-AmalNeural, Gender: FemaleName: ar-QA-MoazNeural, Gender: Male
Name: ar-SA-HamedNeural, Gender: MaleName: ar-SA-ZariyahNeural, Gender: Female
Name: ar-SY-AmanyNeural, Gender: FemaleName: ar-SY-LaithNeural, Gender: Male
Name: ar-TN-HediNeural, Gender: MaleName: ar-TN-ReemNeural, Gender: Female
Name: ar-YE-MaryamNeural, Gender: FemaleName: ar-YE-SalehNeural, Gender: Male
Name: az-AZ-BabekNeural, Gender: MaleName: az-AZ-BanuNeural, Gender: Female
Name: bg-BG-BorislavNeural, Gender: MaleName: bg-BG-KalinaNeural, Gender: Female
Name: bn-BD-NabanitaNeural, Gender: FemaleName: bn-BD-PradeepNeural, Gender: Male
Name: bn-IN-BashkarNeural, Gender: MaleName: bn-IN-TanishaaNeural, Gender: Female
Name: bs-BA-GoranNeural, Gender: MaleName: bs-BA-VesnaNeural, Gender: Female
Name: ca-ES-EnricNeural, Gender: MaleName: ca-ES-JoanaNeural, Gender: Female
Name: cs-CZ-AntoninNeural, Gender: MaleName: cs-CZ-VlastaNeural, Gender: Female
Name: cy-GB-AledNeural, Gender: MaleName: cy-GB-NiaNeural, Gender: Female
Name: da-DK-ChristelNeural, Gender: FemaleName: da-DK-JeppeNeural, Gender: Male
Name: de-AT-IngridNeural, Gender: FemaleName: de-AT-JonasNeural, Gender: Male
Name: de-CH-JanNeural, Gender: MaleName: de-CH-LeniNeural, Gender: Female
Name: de-DE-AmalaNeural, Gender: FemaleName: de-DE-ConradNeural, Gender: Male
Name: de-DE-KatjaNeural, Gender: FemaleName: de-DE-KillianNeural, Gender: Male
Name: el-GR-AthinaNeural, Gender: FemaleName: el-GR-NestorasNeural, Gender: Male
Name: en-AU-NatashaNeural, Gender: FemaleName: en-AU-WilliamNeural, Gender: Male
Name: en-CA-ClaraNeural, Gender: FemaleName: en-CA-LiamNeural, Gender: Male
Name: en-GB-LibbyNeural, Gender: FemaleName: en-GB-MaisieNeural, Gender: Female
Name: en-GB-RyanNeural, Gender: MaleName: en-GB-SoniaNeural, Gender: Female
Name: en-GB-ThomasNeural, Gender: MaleName: en-HK-SamNeural, Gender: Male
Name: en-HK-YanNeural, Gender: FemaleName: en-IE-ConnorNeural, Gender: Male
Name: en-IE-EmilyNeural, Gender: FemaleName: en-IN-NeerjaExpressiveNeural, Gender: Female
Name: en-IN-NeerjaNeural, Gender: FemaleName: en-IN-PrabhatNeural, Gender: Male
Name: en-KE-AsiliaNeural, Gender: FemaleName: en-KE-ChilembaNeural, Gender: Male
Name: en-NG-AbeoNeural, Gender: MaleName: en-NG-EzinneNeural, Gender: Female
Name: en-NZ-MitchellNeural, Gender: MaleName: en-NZ-MollyNeural, Gender: Female
Name: en-PH-JamesNeural, Gender: MaleName: en-PH-RosaNeural, Gender: Female
Name: en-SG-LunaNeural, Gender: FemaleName: en-SG-WayneNeural, Gender: Male
Name: en-TZ-ElimuNeural, Gender: MaleName: en-TZ-ImaniNeural, Gender: Female
Name: en-US-AnaNeural, Gender: FemaleName: en-US-AriaNeural, Gender: Female
Name: en-US-ChristopherNeural, Gender: MaleName: en-US-EricNeural, Gender: Male
Name: en-US-GuyNeural, Gender: MaleName: en-US-JennyNeural, Gender: Female
Name: en-US-MichelleNeural, Gender: FemaleName: en-US-RogerNeural, Gender: Male
Name: en-US-SteffanNeural, Gender: MaleName: en-ZA-LeahNeural, Gender: Female
Name: en-ZA-LukeNeural, Gender: MaleName: es-AR-ElenaNeural, Gender: Female
Name: es-AR-TomasNeural, Gender: MaleName: es-BO-MarceloNeural, Gender: Male
Name: es-BO-SofiaNeural, Gender: FemaleName: es-CL-CatalinaNeural, Gender: Female
Name: es-CL-LorenzoNeural, Gender: MaleName: es-CO-GonzaloNeural, Gender: Male
Name: es-CO-SalomeNeural, Gender: FemaleName: es-CR-JuanNeural, Gender: Male
Name: es-CR-MariaNeural, Gender: FemaleName: es-CU-BelkysNeural, Gender: Female
Name: es-CU-ManuelNeural, Gender: MaleName: es-DO-EmilioNeural, Gender: Male
Name: es-DO-RamonaNeural, Gender: FemaleName: es-EC-AndreaNeural, Gender: Female
Name: es-EC-LuisNeural, Gender: MaleName: es-ES-AlvaroNeural, Gender: Male
Name: es-ES-ElviraNeural, Gender: FemaleName: es-GQ-JavierNeural, Gender: Male
Name: es-GQ-TeresaNeural, Gender: FemaleName: es-GT-AndresNeural, Gender: Male
Name: es-GT-MartaNeural, Gender: FemaleName: es-HN-CarlosNeural, Gender: Male
Name: es-HN-KarlaNeural, Gender: FemaleName: es-MX-DaliaNeural, Gender: Female
Name: es-MX-JorgeNeural, Gender: MaleName: es-NI-FedericoNeural, Gender: Male
Name: es-NI-YolandaNeural, Gender: FemaleName: es-PA-MargaritaNeural, Gender: Female
Name: es-PA-RobertoNeural, Gender: MaleName: es-PE-AlexNeural, Gender: Male
Name: es-PE-CamilaNeural, Gender: FemaleName: es-PR-KarinaNeural, Gender: Female
Name: es-PR-VictorNeural, Gender: MaleName: es-PY-MarioNeural, Gender: Male
Name: es-PY-TaniaNeural, Gender: FemaleName: es-SV-LorenaNeural, Gender: Female
Name: es-SV-RodrigoNeural, Gender: MaleName: es-US-AlonsoNeural, Gender: Male
Name: es-US-PalomaNeural, Gender: FemaleName: es-UY-MateoNeural, Gender: Male
Name: es-UY-ValentinaNeural, Gender: FemaleName: es-VE-PaolaNeural, Gender: Female
Name: es-VE-SebastianNeural, Gender: MaleName: et-EE-AnuNeural, Gender: Female
Name: et-EE-KertNeural, Gender: MaleName: fa-IR-DilaraNeural, Gender: Female
Name: fa-IR-FaridNeural, Gender: MaleName: fi-FI-HarriNeural, Gender: Male
Name: fi-FI-NooraNeural, Gender: FemaleName: fil-PH-AngeloNeural, Gender: Male
Name: fil-PH-BlessicaNeural, Gender: FemaleName: fr-BE-CharlineNeural, Gender: Female
Name: fr-BE-GerardNeural, Gender: MaleName: fr-CA-AntoineNeural, Gender: Male
Name: fr-CA-JeanNeural, Gender: MaleName: fr-CA-SylvieNeural, Gender: Female
Name: fr-CH-ArianeNeural, Gender: FemaleName: fr-CH-FabriceNeural, Gender: Male
Name: fr-FR-DeniseNeural, Gender: FemaleName: fr-FR-EloiseNeural, Gender: Female
Name: fr-FR-HenriNeural, Gender: MaleName: ga-IE-ColmNeural, Gender: Male
Name: ga-IE-OrlaNeural, Gender: FemaleName: gl-ES-RoiNeural, Gender: Male
Name: gl-ES-SabelaNeural, Gender: FemaleName: gu-IN-DhwaniNeural, Gender: Female
Name: gu-IN-NiranjanNeural, Gender: MaleName: he-IL-AvriNeural, Gender: Male
Name: he-IL-HilaNeural, Gender: FemaleName: hi-IN-MadhurNeural, Gender: Male
Name: hi-IN-SwaraNeural, Gender: FemaleName: hr-HR-GabrijelaNeural, Gender: Female
Name: hr-HR-SreckoNeural, Gender: MaleName: hu-HU-NoemiNeural, Gender: Female
Name: hu-HU-TamasNeural, Gender: MaleName: id-ID-ArdiNeural, Gender: Male
Name: id-ID-GadisNeural, Gender: FemaleName: is-IS-GudrunNeural, Gender: Female
Name: is-IS-GunnarNeural, Gender: MaleName: it-IT-DiegoNeural, Gender: Male
Name: it-IT-ElsaNeural, Gender: FemaleName: it-IT-IsabellaNeural, Gender: Female
Name: ja-JP-KeitaNeural, Gender: MaleName: ja-JP-NanamiNeural, Gender: Female
Name: jv-ID-DimasNeural, Gender: MaleName: jv-ID-SitiNeural, Gender: Female
Name: ka-GE-EkaNeural, Gender: FemaleName: ka-GE-GiorgiNeural, Gender: Male
Name: kk-KZ-AigulNeural, Gender: FemaleName: kk-KZ-DauletNeural, Gender: Male
Name: km-KH-PisethNeural, Gender: MaleName: km-KH-SreymomNeural, Gender: Female
Name: kn-IN-GaganNeural, Gender: MaleName: kn-IN-SapnaNeural, Gender: Female
Name: ko-KR-InJoonNeural, Gender: MaleName: ko-KR-SunHiNeural, Gender: Female
Name: lo-LA-ChanthavongNeural, Gender: MaleName: lo-LA-KeomanyNeural, Gender: Female
Name: lt-LT-LeonasNeural, Gender: MaleName: lt-LT-OnaNeural, Gender: Female
Name: lv-LV-EveritaNeural, Gender: FemaleName: lv-LV-NilsNeural, Gender: Male
Name: mk-MK-AleksandarNeural, Gender: MaleName: mk-MK-MarijaNeural, Gender: Female
Name: ml-IN-MidhunNeural, Gender: MaleName: ml-IN-SobhanaNeural, Gender: Female
Name: mn-MN-BataaNeural, Gender: MaleName: mn-MN-YesuiNeural, Gender: Female
Name: mr-IN-AarohiNeural, Gender: FemaleName: mr-IN-ManoharNeural, Gender: Male
Name: ms-MY-OsmanNeural, Gender: MaleName: ms-MY-YasminNeural, Gender: Female
Name: mt-MT-GraceNeural, Gender: FemaleName: mt-MT-JosephNeural, Gender: Male
Name: my-MM-NilarNeural, Gender: FemaleName: my-MM-ThihaNeural, Gender: Male
Name: nb-NO-FinnNeural, Gender: MaleName: nb-NO-PernilleNeural, Gender: Female
Name: ne-NP-HemkalaNeural, Gender: FemaleName: ne-NP-SagarNeural, Gender: Male
Name: nl-BE-ArnaudNeural, Gender: MaleName: nl-BE-DenaNeural, Gender: Female
Name: nl-NL-ColetteNeural, Gender: FemaleName: nl-NL-FennaNeural, Gender: Female
Name: nl-NL-MaartenNeural, Gender: MaleName: pl-PL-MarekNeural, Gender: Male
Name: pl-PL-ZofiaNeural, Gender: FemaleName: ps-AF-GulNawazNeural, Gender: Male
Name: ps-AF-LatifaNeural, Gender: FemaleName: pt-BR-AntonioNeural, Gender: Male
Name: pt-BR-FranciscaNeural, Gender: FemaleName: pt-PT-DuarteNeural, Gender: Male
Name: pt-PT-RaquelNeural, Gender: FemaleName: ro-RO-AlinaNeural, Gender: Female
Name: ro-RO-EmilNeural, Gender: MaleName: ru-RU-DmitryNeural, Gender: Male
Name: ru-RU-SvetlanaNeural, Gender: FemaleName: si-LK-SameeraNeural, Gender: Male
Name: si-LK-ThiliniNeural, Gender: FemaleName: sk-SK-LukasNeural, Gender: Male
Name: sk-SK-ViktoriaNeural, Gender: FemaleName: sl-SI-PetraNeural, Gender: Female
Name: sl-SI-RokNeural, Gender: MaleName: so-SO-MuuseNeural, Gender: Male
Name: so-SO-UbaxNeural, Gender: FemaleName: sq-AL-AnilaNeural, Gender: Female
Name: sq-AL-IlirNeural, Gender: MaleName: sr-RS-NicholasNeural, Gender: Male
Name: sr-RS-SophieNeural, Gender: FemaleName: su-ID-JajangNeural, Gender: Male
Name: su-ID-TutiNeural, Gender: FemaleName: sv-SE-MattiasNeural, Gender: Male
Name: sv-SE-SofieNeural, Gender: FemaleName: sw-KE-RafikiNeural, Gender: Male
Name: sw-KE-ZuriNeural, Gender: FemaleName: sw-TZ-DaudiNeural, Gender: Male
Name: sw-TZ-RehemaNeural, Gender: FemaleName: ta-IN-PallaviNeural, Gender: Female
Name: ta-IN-ValluvarNeural, Gender: MaleName: ta-LK-KumarNeural, Gender: Male
Name: ta-LK-SaranyaNeural, Gender: FemaleName: ta-MY-KaniNeural, Gender: Female
Name: ta-MY-SuryaNeural, Gender: MaleName: ta-SG-AnbuNeural, Gender: Male
Name: ta-SG-VenbaNeural, Gender: FemaleName: te-IN-MohanNeural, Gender: Male
Name: te-IN-ShrutiNeural, Gender: FemaleName: th-TH-NiwatNeural, Gender: Male
Name: th-TH-PremwadeeNeural, Gender: FemaleName: tr-TR-AhmetNeural, Gender: Male
Name: tr-TR-EmelNeural, Gender: FemaleName: uk-UA-OstapNeural, Gender: Male
Name: uk-UA-PolinaNeural, Gender: FemaleName: ur-IN-GulNeural, Gender: Female
Name: ur-IN-SalmanNeural, Gender: MaleName: ur-PK-AsadNeural, Gender: Male
Name: ur-PK-UzmaNeural, Gender: FemaleName: uz-UZ-MadinaNeural, Gender: Female
Name: uz-UZ-SardorNeural, Gender: MaleName: vi-VN-HoaiMyNeural, Gender: Female
Name: vi-VN-NamMinhNeural, Gender: MaleName: zh-CN-XiaoxiaoNeural, Gender: Female
Name: zh-CN-XiaoyiNeural, Gender: FemaleName: zh-CN-YunjianNeural, Gender: Male
Name: zh-CN-YunxiNeural, Gender: MaleName: zh-CN-YunxiaNeural, Gender: Male
Name: zh-CN-YunyangNeural, Gender: MaleName: zh-CN-liaoning-XiaobeiNeural, Gender: Female
Name: zh-CN-shaanxi-XiaoniNeural, Gender: FemaleName: zh-HK-HiuGaaiNeural, Gender: Female
Name: zh-HK-HiuMaanNeural, Gender: FemaleName: zh-HK-WanLungNeural, Gender: Male
Name: zh-TW-HsiaoChenNeural, Gender: FemaleName: zh-TW-HsiaoYuNeural, Gender: Female
Name: zh-TW-YunJheNeural, Gender: MaleName: zu-ZA-ThandoNeural, Gender: Female
Name: zu-ZA-ThembaNeural, Gender: Male,
student, typing, keyboard

Advanced Text To Speech Converter

This program uses a single column .csv file to create .mp3 audio files, row by row.

Usage

This is a way of creating a library of audio files in bulk, from segments of text.

Prerequisites

You will need to install the “edge TTS” and “pydub” libraries.

pip install edge-tts
pip install pydub

How to use it

  1. Save the script below in the folder of your choice
  2. Rename the file you want to convert to audio as “input.csv” and save it in the same folder as the script. Each row will be converted to a separate audio file.
  3. Run the script
  4. The audio files are saved in the “output” folder
import asyncio
import edge_tts
import csv
import os

async def process_text(text, voice, output_filename):
    communicate = edge_tts.Communicate(text, voice)
    await communicate.save(output_filename)

async def main():
    # Read CSV
    with open('input.csv', 'r') as csv_file:
        csv_reader = csv.reader(csv_file)
        rows = list(csv_reader)

    voice = "en-GB-ThomasNeural"
    output_folder = "output"

    if not os.path.exists(output_folder):
        os.makedirs(output_folder)

    for i, row in enumerate(rows):
        concatenated_text = ' '.join(cell_text for cell_text in row if cell_text.strip())  # Concatenate occupied cell texts
        output_filename = os.path.join(output_folder, f"output_{i}.mp3")
        
        await process_text(concatenated_text, voice, output_filename)
        
        if i < len(rows) - 1:
            await asyncio.sleep(0.5)  # Half second gap between rows

    print("Finished")

loop = asyncio.new_event_loop()
loop.run_until_complete(main())
loop.close()

There are a few settings you can adjust:

1) The voice: this script uses “Libby”, a UK accented female voice. At the end of this post I provide the complete list of Edge TTS voices: just replace “en-GB-LibbyNeural” with your preferred voice
2) Reading speed: you can adjust this using the “rate=” line. In this case, the speed is 50% faster than nominal. Try different settings and see how the reading speed changes.

Edge TTS Voices

How to read the voice labels:
EXAMPLE: en-IN-NeerjaNeural, Gender: Female
en – means the voice speaks English
IN – means the accent is Indian

Name: af-ZA-AdriNeural, Gender: FemaleName: af-ZA-WillemNeural, Gender: Male
Name: am-ET-AmehaNeural, Gender: MaleName: am-ET-MekdesNeural, Gender: Female
Name: ar-AE-FatimaNeural, Gender: FemaleName: ar-AE-HamdanNeural, Gender: Male
Name: ar-BH-AliNeural, Gender: MaleName: ar-BH-LailaNeural, Gender: Female
Name: ar-DZ-AminaNeural, Gender: FemaleName: ar-DZ-IsmaelNeural, Gender: Male
Name: ar-EG-SalmaNeural, Gender: FemaleName: ar-EG-ShakirNeural, Gender: Male
Name: ar-IQ-BasselNeural, Gender: MaleName: ar-IQ-RanaNeural, Gender: Female
Name: ar-JO-SanaNeural, Gender: FemaleName: ar-JO-TaimNeural, Gender: Male
Name: ar-KW-FahedNeural, Gender: MaleName: ar-KW-NouraNeural, Gender: Female
Name: ar-LB-LaylaNeural, Gender: FemaleName: ar-LB-RamiNeural, Gender: Male
Name: ar-LY-ImanNeural, Gender: FemaleName: ar-LY-OmarNeural, Gender: Male
Name: ar-MA-JamalNeural, Gender: MaleName: ar-MA-MounaNeural, Gender: Female
Name: ar-OM-AbdullahNeural, Gender: MaleName: ar-OM-AyshaNeural, Gender: Female
Name: ar-QA-AmalNeural, Gender: FemaleName: ar-QA-MoazNeural, Gender: Male
Name: ar-SA-HamedNeural, Gender: MaleName: ar-SA-ZariyahNeural, Gender: Female
Name: ar-SY-AmanyNeural, Gender: FemaleName: ar-SY-LaithNeural, Gender: Male
Name: ar-TN-HediNeural, Gender: MaleName: ar-TN-ReemNeural, Gender: Female
Name: ar-YE-MaryamNeural, Gender: FemaleName: ar-YE-SalehNeural, Gender: Male
Name: az-AZ-BabekNeural, Gender: MaleName: az-AZ-BanuNeural, Gender: Female
Name: bg-BG-BorislavNeural, Gender: MaleName: bg-BG-KalinaNeural, Gender: Female
Name: bn-BD-NabanitaNeural, Gender: FemaleName: bn-BD-PradeepNeural, Gender: Male
Name: bn-IN-BashkarNeural, Gender: MaleName: bn-IN-TanishaaNeural, Gender: Female
Name: bs-BA-GoranNeural, Gender: MaleName: bs-BA-VesnaNeural, Gender: Female
Name: ca-ES-EnricNeural, Gender: MaleName: ca-ES-JoanaNeural, Gender: Female
Name: cs-CZ-AntoninNeural, Gender: MaleName: cs-CZ-VlastaNeural, Gender: Female
Name: cy-GB-AledNeural, Gender: MaleName: cy-GB-NiaNeural, Gender: Female
Name: da-DK-ChristelNeural, Gender: FemaleName: da-DK-JeppeNeural, Gender: Male
Name: de-AT-IngridNeural, Gender: FemaleName: de-AT-JonasNeural, Gender: Male
Name: de-CH-JanNeural, Gender: MaleName: de-CH-LeniNeural, Gender: Female
Name: de-DE-AmalaNeural, Gender: FemaleName: de-DE-ConradNeural, Gender: Male
Name: de-DE-KatjaNeural, Gender: FemaleName: de-DE-KillianNeural, Gender: Male
Name: el-GR-AthinaNeural, Gender: FemaleName: el-GR-NestorasNeural, Gender: Male
Name: en-AU-NatashaNeural, Gender: FemaleName: en-AU-WilliamNeural, Gender: Male
Name: en-CA-ClaraNeural, Gender: FemaleName: en-CA-LiamNeural, Gender: Male
Name: en-GB-LibbyNeural, Gender: FemaleName: en-GB-MaisieNeural, Gender: Female
Name: en-GB-RyanNeural, Gender: MaleName: en-GB-SoniaNeural, Gender: Female
Name: en-GB-ThomasNeural, Gender: MaleName: en-HK-SamNeural, Gender: Male
Name: en-HK-YanNeural, Gender: FemaleName: en-IE-ConnorNeural, Gender: Male
Name: en-IE-EmilyNeural, Gender: FemaleName: en-IN-NeerjaExpressiveNeural, Gender: Female
Name: en-IN-NeerjaNeural, Gender: FemaleName: en-IN-PrabhatNeural, Gender: Male
Name: en-KE-AsiliaNeural, Gender: FemaleName: en-KE-ChilembaNeural, Gender: Male
Name: en-NG-AbeoNeural, Gender: MaleName: en-NG-EzinneNeural, Gender: Female
Name: en-NZ-MitchellNeural, Gender: MaleName: en-NZ-MollyNeural, Gender: Female
Name: en-PH-JamesNeural, Gender: MaleName: en-PH-RosaNeural, Gender: Female
Name: en-SG-LunaNeural, Gender: FemaleName: en-SG-WayneNeural, Gender: Male
Name: en-TZ-ElimuNeural, Gender: MaleName: en-TZ-ImaniNeural, Gender: Female
Name: en-US-AnaNeural, Gender: FemaleName: en-US-AriaNeural, Gender: Female
Name: en-US-ChristopherNeural, Gender: MaleName: en-US-EricNeural, Gender: Male
Name: en-US-GuyNeural, Gender: MaleName: en-US-JennyNeural, Gender: Female
Name: en-US-MichelleNeural, Gender: FemaleName: en-US-RogerNeural, Gender: Male
Name: en-US-SteffanNeural, Gender: MaleName: en-ZA-LeahNeural, Gender: Female
Name: en-ZA-LukeNeural, Gender: MaleName: es-AR-ElenaNeural, Gender: Female
Name: es-AR-TomasNeural, Gender: MaleName: es-BO-MarceloNeural, Gender: Male
Name: es-BO-SofiaNeural, Gender: FemaleName: es-CL-CatalinaNeural, Gender: Female
Name: es-CL-LorenzoNeural, Gender: MaleName: es-CO-GonzaloNeural, Gender: Male
Name: es-CO-SalomeNeural, Gender: FemaleName: es-CR-JuanNeural, Gender: Male
Name: es-CR-MariaNeural, Gender: FemaleName: es-CU-BelkysNeural, Gender: Female
Name: es-CU-ManuelNeural, Gender: MaleName: es-DO-EmilioNeural, Gender: Male
Name: es-DO-RamonaNeural, Gender: FemaleName: es-EC-AndreaNeural, Gender: Female
Name: es-EC-LuisNeural, Gender: MaleName: es-ES-AlvaroNeural, Gender: Male
Name: es-ES-ElviraNeural, Gender: FemaleName: es-GQ-JavierNeural, Gender: Male
Name: es-GQ-TeresaNeural, Gender: FemaleName: es-GT-AndresNeural, Gender: Male
Name: es-GT-MartaNeural, Gender: FemaleName: es-HN-CarlosNeural, Gender: Male
Name: es-HN-KarlaNeural, Gender: FemaleName: es-MX-DaliaNeural, Gender: Female
Name: es-MX-JorgeNeural, Gender: MaleName: es-NI-FedericoNeural, Gender: Male
Name: es-NI-YolandaNeural, Gender: FemaleName: es-PA-MargaritaNeural, Gender: Female
Name: es-PA-RobertoNeural, Gender: MaleName: es-PE-AlexNeural, Gender: Male
Name: es-PE-CamilaNeural, Gender: FemaleName: es-PR-KarinaNeural, Gender: Female
Name: es-PR-VictorNeural, Gender: MaleName: es-PY-MarioNeural, Gender: Male
Name: es-PY-TaniaNeural, Gender: FemaleName: es-SV-LorenaNeural, Gender: Female
Name: es-SV-RodrigoNeural, Gender: MaleName: es-US-AlonsoNeural, Gender: Male
Name: es-US-PalomaNeural, Gender: FemaleName: es-UY-MateoNeural, Gender: Male
Name: es-UY-ValentinaNeural, Gender: FemaleName: es-VE-PaolaNeural, Gender: Female
Name: es-VE-SebastianNeural, Gender: MaleName: et-EE-AnuNeural, Gender: Female
Name: et-EE-KertNeural, Gender: MaleName: fa-IR-DilaraNeural, Gender: Female
Name: fa-IR-FaridNeural, Gender: MaleName: fi-FI-HarriNeural, Gender: Male
Name: fi-FI-NooraNeural, Gender: FemaleName: fil-PH-AngeloNeural, Gender: Male
Name: fil-PH-BlessicaNeural, Gender: FemaleName: fr-BE-CharlineNeural, Gender: Female
Name: fr-BE-GerardNeural, Gender: MaleName: fr-CA-AntoineNeural, Gender: Male
Name: fr-CA-JeanNeural, Gender: MaleName: fr-CA-SylvieNeural, Gender: Female
Name: fr-CH-ArianeNeural, Gender: FemaleName: fr-CH-FabriceNeural, Gender: Male
Name: fr-FR-DeniseNeural, Gender: FemaleName: fr-FR-EloiseNeural, Gender: Female
Name: fr-FR-HenriNeural, Gender: MaleName: ga-IE-ColmNeural, Gender: Male
Name: ga-IE-OrlaNeural, Gender: FemaleName: gl-ES-RoiNeural, Gender: Male
Name: gl-ES-SabelaNeural, Gender: FemaleName: gu-IN-DhwaniNeural, Gender: Female
Name: gu-IN-NiranjanNeural, Gender: MaleName: he-IL-AvriNeural, Gender: Male
Name: he-IL-HilaNeural, Gender: FemaleName: hi-IN-MadhurNeural, Gender: Male
Name: hi-IN-SwaraNeural, Gender: FemaleName: hr-HR-GabrijelaNeural, Gender: Female
Name: hr-HR-SreckoNeural, Gender: MaleName: hu-HU-NoemiNeural, Gender: Female
Name: hu-HU-TamasNeural, Gender: MaleName: id-ID-ArdiNeural, Gender: Male
Name: id-ID-GadisNeural, Gender: FemaleName: is-IS-GudrunNeural, Gender: Female
Name: is-IS-GunnarNeural, Gender: MaleName: it-IT-DiegoNeural, Gender: Male
Name: it-IT-ElsaNeural, Gender: FemaleName: it-IT-IsabellaNeural, Gender: Female
Name: ja-JP-KeitaNeural, Gender: MaleName: ja-JP-NanamiNeural, Gender: Female
Name: jv-ID-DimasNeural, Gender: MaleName: jv-ID-SitiNeural, Gender: Female
Name: ka-GE-EkaNeural, Gender: FemaleName: ka-GE-GiorgiNeural, Gender: Male
Name: kk-KZ-AigulNeural, Gender: FemaleName: kk-KZ-DauletNeural, Gender: Male
Name: km-KH-PisethNeural, Gender: MaleName: km-KH-SreymomNeural, Gender: Female
Name: kn-IN-GaganNeural, Gender: MaleName: kn-IN-SapnaNeural, Gender: Female
Name: ko-KR-InJoonNeural, Gender: MaleName: ko-KR-SunHiNeural, Gender: Female
Name: lo-LA-ChanthavongNeural, Gender: MaleName: lo-LA-KeomanyNeural, Gender: Female
Name: lt-LT-LeonasNeural, Gender: MaleName: lt-LT-OnaNeural, Gender: Female
Name: lv-LV-EveritaNeural, Gender: FemaleName: lv-LV-NilsNeural, Gender: Male
Name: mk-MK-AleksandarNeural, Gender: MaleName: mk-MK-MarijaNeural, Gender: Female
Name: ml-IN-MidhunNeural, Gender: MaleName: ml-IN-SobhanaNeural, Gender: Female
Name: mn-MN-BataaNeural, Gender: MaleName: mn-MN-YesuiNeural, Gender: Female
Name: mr-IN-AarohiNeural, Gender: FemaleName: mr-IN-ManoharNeural, Gender: Male
Name: ms-MY-OsmanNeural, Gender: MaleName: ms-MY-YasminNeural, Gender: Female
Name: mt-MT-GraceNeural, Gender: FemaleName: mt-MT-JosephNeural, Gender: Male
Name: my-MM-NilarNeural, Gender: FemaleName: my-MM-ThihaNeural, Gender: Male
Name: nb-NO-FinnNeural, Gender: MaleName: nb-NO-PernilleNeural, Gender: Female
Name: ne-NP-HemkalaNeural, Gender: FemaleName: ne-NP-SagarNeural, Gender: Male
Name: nl-BE-ArnaudNeural, Gender: MaleName: nl-BE-DenaNeural, Gender: Female
Name: nl-NL-ColetteNeural, Gender: FemaleName: nl-NL-FennaNeural, Gender: Female
Name: nl-NL-MaartenNeural, Gender: MaleName: pl-PL-MarekNeural, Gender: Male
Name: pl-PL-ZofiaNeural, Gender: FemaleName: ps-AF-GulNawazNeural, Gender: Male
Name: ps-AF-LatifaNeural, Gender: FemaleName: pt-BR-AntonioNeural, Gender: Male
Name: pt-BR-FranciscaNeural, Gender: FemaleName: pt-PT-DuarteNeural, Gender: Male
Name: pt-PT-RaquelNeural, Gender: FemaleName: ro-RO-AlinaNeural, Gender: Female
Name: ro-RO-EmilNeural, Gender: MaleName: ru-RU-DmitryNeural, Gender: Male
Name: ru-RU-SvetlanaNeural, Gender: FemaleName: si-LK-SameeraNeural, Gender: Male
Name: si-LK-ThiliniNeural, Gender: FemaleName: sk-SK-LukasNeural, Gender: Male
Name: sk-SK-ViktoriaNeural, Gender: FemaleName: sl-SI-PetraNeural, Gender: Female
Name: sl-SI-RokNeural, Gender: MaleName: so-SO-MuuseNeural, Gender: Male
Name: so-SO-UbaxNeural, Gender: FemaleName: sq-AL-AnilaNeural, Gender: Female
Name: sq-AL-IlirNeural, Gender: MaleName: sr-RS-NicholasNeural, Gender: Male
Name: sr-RS-SophieNeural, Gender: FemaleName: su-ID-JajangNeural, Gender: Male
Name: su-ID-TutiNeural, Gender: FemaleName: sv-SE-MattiasNeural, Gender: Male
Name: sv-SE-SofieNeural, Gender: FemaleName: sw-KE-RafikiNeural, Gender: Male
Name: sw-KE-ZuriNeural, Gender: FemaleName: sw-TZ-DaudiNeural, Gender: Male
Name: sw-TZ-RehemaNeural, Gender: FemaleName: ta-IN-PallaviNeural, Gender: Female
Name: ta-IN-ValluvarNeural, Gender: MaleName: ta-LK-KumarNeural, Gender: Male
Name: ta-LK-SaranyaNeural, Gender: FemaleName: ta-MY-KaniNeural, Gender: Female
Name: ta-MY-SuryaNeural, Gender: MaleName: ta-SG-AnbuNeural, Gender: Male
Name: ta-SG-VenbaNeural, Gender: FemaleName: te-IN-MohanNeural, Gender: Male
Name: te-IN-ShrutiNeural, Gender: FemaleName: th-TH-NiwatNeural, Gender: Male
Name: th-TH-PremwadeeNeural, Gender: FemaleName: tr-TR-AhmetNeural, Gender: Male
Name: tr-TR-EmelNeural, Gender: FemaleName: uk-UA-OstapNeural, Gender: Male
Name: uk-UA-PolinaNeural, Gender: FemaleName: ur-IN-GulNeural, Gender: Female
Name: ur-IN-SalmanNeural, Gender: MaleName: ur-PK-AsadNeural, Gender: Male
Name: ur-PK-UzmaNeural, Gender: FemaleName: uz-UZ-MadinaNeural, Gender: Female
Name: uz-UZ-SardorNeural, Gender: MaleName: vi-VN-HoaiMyNeural, Gender: Female
Name: vi-VN-NamMinhNeural, Gender: MaleName: zh-CN-XiaoxiaoNeural, Gender: Female
Name: zh-CN-XiaoyiNeural, Gender: FemaleName: zh-CN-YunjianNeural, Gender: Male
Name: zh-CN-YunxiNeural, Gender: MaleName: zh-CN-YunxiaNeural, Gender: Male
Name: zh-CN-YunyangNeural, Gender: MaleName: zh-CN-liaoning-XiaobeiNeural, Gender: Female
Name: zh-CN-shaanxi-XiaoniNeural, Gender: FemaleName: zh-HK-HiuGaaiNeural, Gender: Female
Name: zh-HK-HiuMaanNeural, Gender: FemaleName: zh-HK-WanLungNeural, Gender: Male
Name: zh-TW-HsiaoChenNeural, Gender: FemaleName: zh-TW-HsiaoYuNeural, Gender: Female
Name: zh-TW-YunJheNeural, Gender: MaleName: zu-ZA-ThandoNeural, Gender: Female
Name: zu-ZA-ThembaNeural, Gender: Male,
white printing paper with numbers

.csv to HTML Converter

This is a really simple but effective way to convert single column .csv files to HTML files.

Usage

Why would you use this? Well sometimes you may have a long list you want to render as a table on a website. It can be very time consuming to do this by hand, so this script does it automaticaly in the blink of an eye. I used it on this post to render the list of voices as a table.

Prerequisites

There are no prerequisites to install with this script.

How to use it

  1. Save the script below in the folder of your choice
  2. Rename the .csv file you want to convert as “input.csv” and save it in the same folder as the script
  3. Run the script
  4. The HTML file is saved as “output.html”
  5. If you want more than 2 columns, change “2” in the two places in the script indicated by comments
import csv

def csv_to_html(input_csv, output_html):
    with open(input_csv, 'r') as file:
        reader = csv.reader(file)
        data = [row[0] for row in reader]

    html_table = '<table>\n'
    for i in range(0, len(data), 2): # Change "2" to the number of columns you require
        html_table += '<tr>\n'
        for j in range(2):  # Change "2" to the number of columns you require
            if i + j < len(data):
                html_table += f'<td>{data[i + j]}</td>'
            else:
                html_table += '<td></td>'  # Add empty cell if data is exhausted
        html_table += '</tr>\n'
    html_table += '</table>'

    with open(output_html, 'w') as output_file:
        output_file.write(html_table)

if __name__ == "__main__":
    input_csv_file = "input.csv"
    output_html_file = "output.html"
    
    csv_to_html(input_csv_file, output_html_file)