The OP-Markt Bot is a versatile tool designed to enhance the Minecraft experience for the Opsucht community. This bot interacts with the server's market API to provide item prices, supports fuzzy matching to identify Minecraft items, and formats responses to be user-friendly.
- Price Retrieval: Fetches and displays item prices from the Opsucht server’s market.
- Fuzzy Matching: Utilizes a sophisticated algorithm to find the closest matching items based on user input.
- Embed Responses: Provides formatted and visually appealing responses, including item images and price details.
- Reputation Tracking: Allows users to view the reputation of a Minecraft player based on their UUID.
- Reputation Stats: Enables administrators to view how many positive and negative reputations a user has given.
- Reputation Management: Users can give positive or negative reputation to a Minecraft player, tracked by their UUID.
- Python 3.12 or higher
- Discord.py library (
discord.py
version 2.0 or later) aiohttp
for asynchronous HTTP requests- JSON files for configuration and item data
- Clone the repository:
https://github.com/david-star-git/op-markt.git
- Change directory:
cd op-markt
- Create a virtual environment:
python -m venv venv
- Activate the virtual environment:
- On Unix or MacOS:
source venv/bin/activate
- On Windows:
venv\Scripts\activate
- On Unix or MacOS:
Install the required dependencies: pip install discord.py aiohttp matplotlib pandas aiosqlite beautifulsoup4
- Create a file named
api.json
in the root directory with your API credentials.
{
"TOKEN": "DISCORD BOT TOKEN",
"API-KEY": "OPSCHT API KEY",
"API-UNAME": "OPSUCHT API USERNAME",
"API_URL": "https://api.opsucht.net/market"
}
- Create a file named
data/config.json
with your configuration details.
{
"name": "OP-Markt", <- bot username
"activity": "the market.", <- bot activity
"embed_hex": "0x60aefa", <- embed color
"last_refresh": 0 <- timestamp of the last data refresh
}
- Start the bot:
python bot.py
Fetches the price for a specified item. The bot returns both the buy and sell prices, along with an image of the item.
Example Usage:
- Command:
/price diamond
Response:
The bot will provide an embed message with:
- Item Name
- Buy Price
- Sell Price
- Item Image
Displays the reputation of a specified Minecraft player. The bot retrieves the player's UUID and shows positive, negative, and overall reputation in an embed.
Example Usage:
- Command:
/view_rep Notch
Response:
The bot will provide an embed message with:
- Player Name
- Positive Reputations
- Negative Reputations
- Overall Reputation
- Player's Avatar Image
Allows users to check the reputation stats of a specified Discord user. The bot will display how many positive and negative reputations the user has given.
Example Usage:
- Command:
/stats @Steve
Response:
The bot will provide a message with:
- Positive Reputations Given
- Negative Reputations Given
Users can give positive or negative reputation to a Minecraft player. The bot checks the player’s UUID and adds the reputation to the database.
Example Usage:
- Command:
/give_rep Notch positive
Response:
The bot will confirm the action by replying with a message, indicating the reputation has been added to the player.
The fuzzy matching algorithm in this bot is designed to handle imperfect user input and still find the most relevant Minecraft item. The algorithm incorporates Levenshtein distance and length-based prioritization to ensure accurate and useful results.
Levenshtein distance was chosen for fuzzy matching in this bot due to its effectiveness in handling small differences between strings. Here are the key reasons:
-
Error Tolerance: Levenshtein distance measures how many single-character edits (insertions, deletions, substitutions) are required to transform one string into another. This makes it particularly useful for dealing with common typographical errors or slight misspellings.
-
Simplicity: The algorithm is relatively straightforward to implement and understand, making it suitable for the bot's needs without requiring complex setup or computation.
-
Versatility: It can be used in various contexts beyond just matching item names, allowing for flexible application in different scenarios where string similarity is important.
-
Adaptability: It effectively balances between exact matches and approximate matches, ensuring that users receive relevant results even if their input is not perfectly accurate.
While other algorithms like Jaccard similarity or cosine similarity could also be used for fuzzy matching, Levenshtein distance is favored here due to its direct measurement of edit distance and ease of use for our specific requirements.
-
Levenshtein Distance Calculation: The algorithm computes the Levenshtein distance between the user’s query and each item name. This distance measures how many single-character edits (insertions, deletions, substitutions) are needed to transform one string into another.
-
Prioritization Logic:
- Substring Matches: Initially, it looks for items that contain the user’s query as a substring.
- Levenshtein Distance: For substring matches, it calculates the distance and selects the closest match.
-
Handling Special Cases:
- If there are no substring matches, the algorithm falls back to a general fuzzy match using Levenshtein distance.
- Specificity is prioritized when lengths and distances are close.
The bot refreshes its data every 60 minutes to ensure that it provides up-to-date information. The timestamp of the last refresh is stored in data/config.json
under "last_refresh"
. This approach is designed to avoid excessive API calls and reduce server resource usage, enhancing efficiency. By keeping track of when the last refresh occurred, the bot ensures that it only refreshes data when necessary, thereby minimizing the load on the server and conserving resources. If the bot is restarted, it will remember when the last refresh took place and continue operating efficiently.
The bot formats prices to be user-friendly and consistent. Prices are displayed with thousand separators and one decimal point. For instance, 1000.00
is formatted as 1.000,0 $
.
The function:
- Rounds prices to one decimal point.
- Formats with thousand separators.
- Appends a currency symbol.
The bot includes item images and formatted responses in Discord embeds. Images are sourced from a specific URL pattern.
For an item like diamond
, the image URL is:
https://mc.nerothe.com/img/1.21/minecraft_diamond.png
Embedded Responses:
- The embed title is the item name.
- The description includes price information.
- The thumbnail is the item’s image.
The OP-Markt Bot includes a price history graph feature that provides visual insights into the price trends of items over the past 30 days. This feature is integrated into the /price command, enhancing the user experience by offering a dynamic representation of price fluctuations.
- Data Collection:
- Daily Refresh: The bot refreshes item prices daily and saves the data into data/prices/ directory.
- File Management: Each day’s price data is stored in a file named with the date (e.g., 10.08.2024.json).
- Graph Generation:
- Data Aggregation: When generating the graph, the bot aggregates price data from the last 30 days. Prices are extracted from the daily JSON files.
- Handling Missing Data: If an item’s price is not available for a particular day, it is defaulted to zero to maintain continuity in the graph.
- Graph Construction:
- X-Axis: Represents the days over the past 30 days.
- Y-Axis: Shows the price values, formatted with thousands (k) and millions (M) for large numbers.
- Data Series:
- Buy Prices: Displayed as a blue line.
- Sell Prices: Displayed as a red line.
- Graph Formatting: The graph is dynamically created with a transparent background to seamlessly integrate with Discord embeds.
- Title and Labels:
- Graph Title: Uses the formatted item name to clearly identify the item being analyzed.
- Axis Labels: Include date and price labels formatted for readability.
- Embed Integration:
- Image Embedding: The generated graph image is embedded in the Discord response for the /price command.
- Visual Presentation: The graph is displayed below the item’s price details, providing a clear visual representation of price trends.
- Visual Insight: Allows users to quickly understand price trends and fluctuations over time.
- Enhanced User Experience: Provides a more engaging way to view price data compared to text-based reports.
- Dynamic Updates: Ensures that the graph reflects the most recent 30 days of data, keeping the information relevant and up-to-date.
- Price History Graph: The /price command now includes a graph showing the price history for the past 30 days. The graph displays buy prices in blue and sell prices in red, with formatted y-axis labels and transparent background. The graph title uses the formatted item name.
You need to install matplotlib
and pandas
for the price graph to work.
- Reputation System: The bot now includes a reputation tracking system for Minecraft players. Users can give positive or negative reputation to a player based on their Minecraft UUID. The bot keeps track of reputations and allows users to view the overall reputation of any player using the
/view_rep
command. Additionally, users can check the reputation stats of a Discord user to see how many positive and negative reputations they have given using the/stats
command.
You need to install aiosqlite
so you the bot can create and acces the database.
- Reputation Commands:
- /view_rep: Displays the total positive, negative, and overall reputation for a Minecraft player, complete with their avatar image.
- /give_rep: Allows users to give a positive or negative reputation to a Minecraft player. This command ensures that each user can only give reputation to another player once.
- /stats: Enables users to check how many positive and negative reputations a Discord user has given.
- Add image to embed.
- Write a markdown.
- Add categories.
- Add graph to show item price changes.
- Add support for german itemnames.
- Translate to german.
- Add tab completion to
/price
command. - Add a reputation system.
- Keep it efficient.
- KISS
If you’d like to contribute to this project, please fork the repository and submit a pull request with your changes. Ensure to follow the coding standards and provide documentation for new features.
This project is licensed under the GNU General Public License. See the LICENSE file for details.