To summarize, I\'m a little confused as to how I would design such a database that allows indefinite badge-rule creation without requiring structu
Given that the badge criteria can be arbitrarily complex, I don't think you can store it in a database table broken down into "simple" data elements. Trying to write a "rules engine" that can handle arbitrarily complex criteria is going to take you down the path of basically re-writing all the tools that you have in your programming language.
If you know in advance you want the badges limited to only certain fields (i.e. badges are only based off reputation or number of edits or something), then you can store those in a simple table like:
ReputationBadgeCriteria
BadgeId
BadgeName
MinReputation
Alternatively, you could use some kind of DSL to write your "rules" but you end up having to also create a parser to parse the rules when you read them as well as something to execute these rules. Depending on the complexity you want in your DSL, this may not be a trivial task. This looks like the path you are going in your question with having a Criteria column (presumably plain text) that has something like "[Reputation] > 1000" or "[Posts] > 5" in it. You still have to parse and execute those rules and the complexity of writing something to do so is dependent on how complex you want those rules to be.
I would recommend you read these Daily WTF articles for information on why this approach leads to pain.
I'm approaching it like this: Create a table to store all the badges, and have one column reference a function that is run to see if the badge is awarded. That way the table remains simple and the logic to determine the badges can be kept in the code, where it is best suited.
With this method, badge requirements could also be linked together, to form more complex dependencies. For example, user must receive three separate, specific badges w/in a certain timeframe in order to get this badge.
I would not create an increment for the edit badge. I think you should have a job running in the background and count() the numbero of post edited for members that haven't the badge of editing yet. When you see that the count is over the range you want, you add an entry in the database that tell that the user have the badge.
I think it's about the same for other badges. Try to limit the number of writting and do not write directly the count of the badge information in the user table. Use a table that will contain badge information and link it to the user table.
I apologize for being brief.
To implement a system such as this, I might create a table which stores either stored procedure names or actual queries that would be used to determine whether a particular user has earned a badge.
badge_criteria
badge_key int
badge_criteria varchar(max)
You can extract and execute the queries for badges which the user has not earned from your middle tier, but you would not have to make any code or structural changes to add new badges going forward.
You'd track your unique users in one table and unique badges in another then create a cross reference table to relate them.
A user can have many badges and a badge can have many users.
create table users (
id int,
name varchar
)
create table badges (
id int,
badge_name varchar
)
create table user_badges_xref (
user_id int,
badge_id int
)
Statistics that might affect whether a user earns a badge are tracked as a part of the administration of the site. so something like an answer being accepted would be in a schema that relates questions & answers. in order to display the answer and the owner of the answer, there would be a relationship to the user table and triggers that would check for badge conditions whenever a change was made.
I'm not asking how to award badges. I'm asking how to store criteria within the database.
So you want to store the logical operation required to determine if a badge is earned in a field somewhere?
I think agree with the other poster that criteria should be a part of the business logic. That logic can be on the app side or within a trigger. I think that's a matter of style.
If you were really married to the idea of storing the criteria in a field, I'd store it as parameterized SQL and run it dynamically.
So this sort of thing would be in your criteria field:
select "Badge Earned"
from all_posts
where user_id = @user_id
having count(*) > 10000
Depending on how far you want to go with it, your schema can get pretty complicated. It seems to me that the base elements you need to track are:
Badges awarded
Points earned
Pretty simple so far, but you want to be able to dynamically create new badges and new points categories. Badge awards would depend on earning points in one or more point categories that would add up to a certain amount. So you need to track the relationship between point categories (and points earned) and badges:
Point categories
Badge categories
So the key would be your user points table, which would link to point categories, which link to badges. Users earn points in a particular category, which would contribute to earning points towards one or more badges.
badges:
badge_id
badge_name
required_points
....
point_categories:
point_id
category_name
weighting (optional)
...
point_groups:
badge_id
point_id
weighting (optional)
...
user_points:
user_id
point_id
points
...
user_badges:
user_id
badge_id
points_earned
badge_awarded (yes/no)
...
Your "admin" interface would allow someone to create a new badge and pick which point categories are required to earn that badge (point_groups). Whenever a user earns points (user_points), you update the user_points table, then determine which badges those points would could contribute to (point_groups). You then recompile the points for the badges that were affected by the points earned and update the user_badges table with the point_earned. Then check the points_earned field in user_badges against the required_points in the badges table.
You can get much fancier by assigning different weights to different point categories, or even different weights for point categories for particular badges. But this setup would allow an unlimited amount of badges and point categories to be created and managed fairly easily without changing tables structures.
If that is completely not what you are looking for, then I think I should at least get a vote or two for a lot of typing.