Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Unique Constraints #286

Open
jcbdev opened this issue Jul 29, 2020 · 25 comments
Open

Unique Constraints #286

jcbdev opened this issue Jul 29, 2020 · 25 comments

Comments

@jcbdev
Copy link

jcbdev commented Jul 29, 2020

Is your feature request related to a problem? Please describe.
I have lots of scenarios where I require unique values on non-primary key fields. Especially for public apps where you might want to not allow two people with the same phone number or email address and so forth

Describe the solution you'd like
It would be possible to add a directive such as @unique to a field or @unique(fields: ["email", "phone"]) to the type

First of all this would cause resources to be added:

  • A table would have been created by the amplify CLI called "UniqueConstraints-APIID-env"

when this is added you can then change the VTL generated to do the following:

  • operation is changed from PutItem in the VTL to "TransactWriteItems"
  • The main record is a "Put" operation is added to the item list
  • for every unique constraint (up to 24 which is max transaction),
    ** another "Put" item is added the Items but for TableName - "UniqueConstraints-APIID-env" and the id(pk) of "table#field#value"
    ** attribute_not_exist(id) is added the condition for this item

This would then only succeed if the field is successful. You would also need the update and delete templates to follow the equivalent logic

If the "value" is a complex type you could potentially use a hash of the value

Describe alternatives you've considered
I have used custom lambda resolvers and custom resolvers to implement this myself but it tedious for something I have to do over and over again

@jamesone
Copy link

Does anyone know if this is currently being developed by the amplify team?

@half2me
Copy link

half2me commented Sep 19, 2020

I've only started out with Amplify, but it seems like a really basic requirement. There are plenty of times you would need to make sure a field(s) is unique. Having a directive like @unique seems to make sense.

@WillDent
Copy link

This is the best I could find specific to what you're asking for.

https://medium.com/@fullstackpho/ensuring-usernames-are-unique-in-your-aws-amplify-app-6fff963274

@connelhooley
Copy link

connelhooley commented Dec 12, 2020

A real shame this hasn't been picked up, the medium link posted above works by running a Lambda that validates the data before (or just after?) it is inserted into the database. Uniqueness should be handled by the data store so it is guaranteed to be enforced.

@jcbdev
Copy link
Author

jcbdev commented Dec 15, 2020

@WillDent I tried similar methods but the method I suggested works better. I have modified my VTL template to use the method and it works perfectly (and is not that diffficult to do). It just takes a long time to do for every template and field combination when we could easily add directive to the schema that changes the vtl template generation slightly

@matansagee
Copy link

+1

@droidgeek1
Copy link

droidgeek1 commented May 17, 2021

Even key field does not ensure uniqueness. I have following and duplicate email accepted even though I have indicated that as key. Worse part is that the backend corresponding dynamo table does not seem to take the duplicate record. So now, my client side sqlite is out of sync with dynamoDB. So I have to write code to check if the record exists..Sounds ugly unless someone can enlighten me please..

type AddUserToT
@model
@key(fields: ["userEmail"])
id: ID!
email: ID!

@mm87642
Copy link

mm87642 commented Aug 16, 2021

+1

7 similar comments
@shizhaojingszj
Copy link

+1

@DerekFei
Copy link

DerekFei commented Sep 4, 2021

+1

@graniczny
Copy link

+1

@Wyfy0107
Copy link

Wyfy0107 commented Dec 1, 2021

+1

@JohnLamHK
Copy link

+1

@iltumio
Copy link

iltumio commented Dec 23, 2021

+1

@hackmajoris
Copy link

+1

@ilkerburakkurt
Copy link

+1

@acusti
Copy link

acusti commented Apr 25, 2022

i notice this issue has the graphql-transformer-v1 label on it, but i want this feature for an amplify project i am working on that is using graphql transformer v2. should i create a new issue for that, or should this issue have the graphql-transformer-v2 label?

@alharris-at alharris-at transferred this issue from aws-amplify/amplify-cli May 17, 2022
@nxia416
Copy link

nxia416 commented Jun 10, 2022

+1

@acusti
Copy link

acusti commented Jun 17, 2022

did the behavior of attribute_exists / attribute_not_exists dynamodb condition expressions change at some point? i was using attribute_not_exists in a CreateItem mutation (via condition: { field: { attributeExists: false } }), and i thought i had verified that it worked to ensure that the new item’s value for the specified field would be unique amongst all other items in the table, but i can verify that is not the case right now.

according to the dynamodb docs on comparison operators and functions, attribute_exists and attribute_not_exists are used to check only whether an item has or doesn’t have an attribute. it seems to have nothing to do with whether any other items in the table have or don’t have the same value for that attribute.

@jcbdev do you know if your technique of maintaining a UniqueConstraints table with the id of each item being a hash of table + fieldname + value still works? or maybe it only works because it’s being used as the primary key, and primary keys enforce uniqueness, so it has nothing to do with the attribute_not_exists(id) condition? in which case i suppose that condition passes because the item hasn’t been created yet so it doesn’t have an id? and lastly, have you added aDelete item for the previous value of the unique field on Update and Delete operations? it would be great to see a more complete example of how i could implement this manually by modifying the VTL templates.

@Voyager-Two
Copy link

Voyager-Two commented Jul 7, 2022

This seems like major missing feature IMO.

For future readers:
One work-around is to set unique field as primary key:

username: String! @primaryKey

You can also do something like this (e.g. Discord usernames Bob#0001):

discordUsername: String! @primaryKey(sortKeyFields: ["discordNumber"])
discordNumber: Int!

@jcbdev
Copy link
Author

jcbdev commented Jul 7, 2022

did the behavior of attribute_exists / attribute_not_exists dynamodb condition expressions change at some point? i was using attribute_not_exists in a CreateItem mutation (via condition: { field: { attributeExists: false } }), and i thought i had verified that it worked to ensure that the new item’s value for the specified field would be unique amongst all other items in the table, but i can verify that is not the case right now.

according to the dynamodb docs on comparison operators and functions, attribute_exists and attribute_not_exists are used to check only whether an item has or doesn’t have an attribute. it seems to have nothing to do with whether any other items in the table have or don’t have the same value for that attribute.

@jcbdev do you know if your technique of maintaining a UniqueConstraints table with the id of each item being a hash of table + fieldname + value still works? or maybe it only works because it’s being used as the primary key, and primary keys enforce uniqueness, so it has nothing to do with the attribute_not_exists(id) condition? in which case i suppose that condition passes because the item hasn’t been created yet so it doesn’t have an id? and lastly, have you added aDelete item for the previous value of the unique field on Update and Delete operations? it would be great to see a more complete example of how i could implement this manually by modifying the VTL templates.

@acusti It did when I was using it but yes i think its because the column is the primary key. The trick was to create a uniqueconstraints table where all the unique constraints are written to and not to write the unique constraints to the same table your data is in (the dynamodb put requests can go to different tables in the transaction). But I do not use amplify anymore. I rewrote the platform on cdk and have never looked back. I found aws amplify terrible at scaling and just couldn't get to work on any project that wasn't a toy project. Too many limitations, compromises and assumptions that I spent more time maintaining and working around amplify in the project than developing new features.

@lewisdonovan
Copy link

lewisdonovan commented Aug 12, 2022

Hi all. Just stumbled across this and think I might have found the answer. If you look at the generated schemas in AppSync, you'll see the schemas for the condition expression objects as well. For example, the schema to create a condition on a String input is:

input ModelStringInput {
	ne: String
	eq: String
	le: String
	lt: String
	ge: String
	gt: String
	contains: String
	notContains: String
	between: [String]
	beginsWith: String
	attributeExists: Boolean
	attributeType: ModelAttributeTypes
	size: ModelSizeInput
}

The first item in that schema ne means not equal to, which can be used to run a uniqueness comparison when putting an item. An example mutation would look like this:

mutation CreateTodoGraphQL($input: CreateTodoInput!, $condition: ModelTodoConditionInput) {
  createTodoGraphQL(input: $input, condition: $condition) {
    id
    name
    priority
    completedAt
  }
}

and the input variables would look like:

{
  "input": {
    "name": "test",
    "priority": "1",
    "completedAt": "2022-09-11T09:00:00.000Z"
  },
  "condition": { 
    "name": {
      "ne": "test"
    }
  }
}

If an item already exists with the name test then the condition expression would fail, and the item would not be added to the DB.

@Meerkov
Copy link

Meerkov commented Jun 10, 2023

@lewisdonovan , isn't the condition specified on the client side? In other words, a client can cheat your system by removing that part of the request.

@lewisdonovan
Copy link

@lewisdonovan , isn't the condition specified on the client side? In other words, a client can cheat your system by removing that part of the request.

Yes, absolutely. But AppSync can be run on the back-end too. You could create a custom mutation, pass name as a parameter (rather than a filter variable), map the mutation to a Lambda and run the filter logic there.

@gavmck
Copy link

gavmck commented Aug 15, 2023

@lewisdonovan I'm not sure the condition method works, just given it a spin and it still lets me create/update the user.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests