Prisma Datamodel: Primary key as a combination of two relational models











up vote
1
down vote

favorite












I have a problem in Prisma data modeling where I have to constrain that a user can submit only one review for a product. I have following design for the non-constrained situation.




Should Customer and Product be combined into a primary key in ProductReview model, or should this constraint be imposed at the application server level, and not at the database level?




Datamodel for now (non-constrained version):



type Product {
id: ID! @unique
title: String!
reviews: [ProductReview!]! @relation(name: "ProductReviews", onDelete: CASCADE)
}

type Customer {
id: ID! @unique
email: String @unique
}

type ProductReview {
id: ID! @unique
forProduct: Product! @relation(name: "ProductReviews", onDelete: SET_NULL)
byCustomer: Customer!
review: String!
ratinng: Float!
}









share|improve this question


























    up vote
    1
    down vote

    favorite












    I have a problem in Prisma data modeling where I have to constrain that a user can submit only one review for a product. I have following design for the non-constrained situation.




    Should Customer and Product be combined into a primary key in ProductReview model, or should this constraint be imposed at the application server level, and not at the database level?




    Datamodel for now (non-constrained version):



    type Product {
    id: ID! @unique
    title: String!
    reviews: [ProductReview!]! @relation(name: "ProductReviews", onDelete: CASCADE)
    }

    type Customer {
    id: ID! @unique
    email: String @unique
    }

    type ProductReview {
    id: ID! @unique
    forProduct: Product! @relation(name: "ProductReviews", onDelete: SET_NULL)
    byCustomer: Customer!
    review: String!
    ratinng: Float!
    }









    share|improve this question
























      up vote
      1
      down vote

      favorite









      up vote
      1
      down vote

      favorite











      I have a problem in Prisma data modeling where I have to constrain that a user can submit only one review for a product. I have following design for the non-constrained situation.




      Should Customer and Product be combined into a primary key in ProductReview model, or should this constraint be imposed at the application server level, and not at the database level?




      Datamodel for now (non-constrained version):



      type Product {
      id: ID! @unique
      title: String!
      reviews: [ProductReview!]! @relation(name: "ProductReviews", onDelete: CASCADE)
      }

      type Customer {
      id: ID! @unique
      email: String @unique
      }

      type ProductReview {
      id: ID! @unique
      forProduct: Product! @relation(name: "ProductReviews", onDelete: SET_NULL)
      byCustomer: Customer!
      review: String!
      ratinng: Float!
      }









      share|improve this question













      I have a problem in Prisma data modeling where I have to constrain that a user can submit only one review for a product. I have following design for the non-constrained situation.




      Should Customer and Product be combined into a primary key in ProductReview model, or should this constraint be imposed at the application server level, and not at the database level?




      Datamodel for now (non-constrained version):



      type Product {
      id: ID! @unique
      title: String!
      reviews: [ProductReview!]! @relation(name: "ProductReviews", onDelete: CASCADE)
      }

      type Customer {
      id: ID! @unique
      email: String @unique
      }

      type ProductReview {
      id: ID! @unique
      forProduct: Product! @relation(name: "ProductReviews", onDelete: SET_NULL)
      byCustomer: Customer!
      review: String!
      ratinng: Float!
      }






      mysql graphql prisma






      share|improve this question













      share|improve this question











      share|improve this question




      share|improve this question










      asked 2 days ago









      devautor

      8411616




      8411616
























          2 Answers
          2






          active

          oldest

          votes

















          up vote
          2
          down vote



          accepted











          I have to constrain that a user can submit only one review for a product. I have following design for the non-constrained situation.




          Unfortunately, this is currently not possible with Prisma. There already is an open feature request asking for this functionality, please leave your 👍 on the issue!



          To get that functionality in your application, you'll need to implement that constraint manually on the application layer (e.g. express, apollo-server or graphql-yoga).



          You can take a look at this page of How to GraphQL where there's a similar situation with the User, Link and Vote types. Here's how the resolver to create a Vote and ensurs no votes from that user already exist is implemented with graphql-yoga:



          async function vote(parent, args, context, info) {
          // 1
          const userId = getUserId(context)

          // 2
          const linkExists = await context.db.exists.Vote({
          user: { id: userId },
          link: { id: args.linkId },
          })
          if (linkExists) {
          throw new Error(`Already voted for link: ${args.linkId}`)
          }

          // 3
          return context.db.mutation.createVote(
          {
          data: {
          user: { connect: { id: userId } },
          link: { connect: { id: args.linkId } },
          },
          },
          info,
          )
          }





          share|improve this answer



















          • 1




            Hi @nburk, so the idea is to handle this at the application server level for now. Got it! Have raised a feature request with the same body too. Thanks!
            – devautor
            2 days ago


















          up vote
          1
          down vote













          I will answer from the MySQL perspective. If you want to enforce that a given customer can only be associated with a given product once, then you should make (cusotmer_id, product_id) a unique key (maybe primary) in the ProductReview table:



          ALTER TABLE ProductReview ADD UNIQUE KEY uk_cust_prod (customer_id, product_id);


          This means that any attempt to insert a record for a given customer and product, when such a relation already exists, would fail at the database level.



          If you also want to add an application level check for this, you of course may do so, and perhaps handle it there first.






          share|improve this answer





















          • Thanks @tim, since Prisma doesn't support this, application level check seems the only way. Would you suggest a "right way" to handle this there?
            – devautor
            2 days ago











          Your Answer






          StackExchange.ifUsing("editor", function () {
          StackExchange.using("externalEditor", function () {
          StackExchange.using("snippets", function () {
          StackExchange.snippets.init();
          });
          });
          }, "code-snippets");

          StackExchange.ready(function() {
          var channelOptions = {
          tags: "".split(" "),
          id: "1"
          };
          initTagRenderer("".split(" "), "".split(" "), channelOptions);

          StackExchange.using("externalEditor", function() {
          // Have to fire editor after snippets, if snippets enabled
          if (StackExchange.settings.snippets.snippetsEnabled) {
          StackExchange.using("snippets", function() {
          createEditor();
          });
          }
          else {
          createEditor();
          }
          });

          function createEditor() {
          StackExchange.prepareEditor({
          heartbeatType: 'answer',
          convertImagesToLinks: true,
          noModals: true,
          showLowRepImageUploadWarning: true,
          reputationToPostImages: 10,
          bindNavPrevention: true,
          postfix: "",
          imageUploader: {
          brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
          contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
          allowUrls: true
          },
          onDemand: true,
          discardSelector: ".discard-answer"
          ,immediatelyShowMarkdownHelp:true
          });


          }
          });














           

          draft saved


          draft discarded


















          StackExchange.ready(
          function () {
          StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53373101%2fprisma-datamodel-primary-key-as-a-combination-of-two-relational-models%23new-answer', 'question_page');
          }
          );

          Post as a guest















          Required, but never shown

























          2 Answers
          2






          active

          oldest

          votes








          2 Answers
          2






          active

          oldest

          votes









          active

          oldest

          votes






          active

          oldest

          votes








          up vote
          2
          down vote



          accepted











          I have to constrain that a user can submit only one review for a product. I have following design for the non-constrained situation.




          Unfortunately, this is currently not possible with Prisma. There already is an open feature request asking for this functionality, please leave your 👍 on the issue!



          To get that functionality in your application, you'll need to implement that constraint manually on the application layer (e.g. express, apollo-server or graphql-yoga).



          You can take a look at this page of How to GraphQL where there's a similar situation with the User, Link and Vote types. Here's how the resolver to create a Vote and ensurs no votes from that user already exist is implemented with graphql-yoga:



          async function vote(parent, args, context, info) {
          // 1
          const userId = getUserId(context)

          // 2
          const linkExists = await context.db.exists.Vote({
          user: { id: userId },
          link: { id: args.linkId },
          })
          if (linkExists) {
          throw new Error(`Already voted for link: ${args.linkId}`)
          }

          // 3
          return context.db.mutation.createVote(
          {
          data: {
          user: { connect: { id: userId } },
          link: { connect: { id: args.linkId } },
          },
          },
          info,
          )
          }





          share|improve this answer



















          • 1




            Hi @nburk, so the idea is to handle this at the application server level for now. Got it! Have raised a feature request with the same body too. Thanks!
            – devautor
            2 days ago















          up vote
          2
          down vote



          accepted











          I have to constrain that a user can submit only one review for a product. I have following design for the non-constrained situation.




          Unfortunately, this is currently not possible with Prisma. There already is an open feature request asking for this functionality, please leave your 👍 on the issue!



          To get that functionality in your application, you'll need to implement that constraint manually on the application layer (e.g. express, apollo-server or graphql-yoga).



          You can take a look at this page of How to GraphQL where there's a similar situation with the User, Link and Vote types. Here's how the resolver to create a Vote and ensurs no votes from that user already exist is implemented with graphql-yoga:



          async function vote(parent, args, context, info) {
          // 1
          const userId = getUserId(context)

          // 2
          const linkExists = await context.db.exists.Vote({
          user: { id: userId },
          link: { id: args.linkId },
          })
          if (linkExists) {
          throw new Error(`Already voted for link: ${args.linkId}`)
          }

          // 3
          return context.db.mutation.createVote(
          {
          data: {
          user: { connect: { id: userId } },
          link: { connect: { id: args.linkId } },
          },
          },
          info,
          )
          }





          share|improve this answer



















          • 1




            Hi @nburk, so the idea is to handle this at the application server level for now. Got it! Have raised a feature request with the same body too. Thanks!
            – devautor
            2 days ago













          up vote
          2
          down vote



          accepted







          up vote
          2
          down vote



          accepted







          I have to constrain that a user can submit only one review for a product. I have following design for the non-constrained situation.




          Unfortunately, this is currently not possible with Prisma. There already is an open feature request asking for this functionality, please leave your 👍 on the issue!



          To get that functionality in your application, you'll need to implement that constraint manually on the application layer (e.g. express, apollo-server or graphql-yoga).



          You can take a look at this page of How to GraphQL where there's a similar situation with the User, Link and Vote types. Here's how the resolver to create a Vote and ensurs no votes from that user already exist is implemented with graphql-yoga:



          async function vote(parent, args, context, info) {
          // 1
          const userId = getUserId(context)

          // 2
          const linkExists = await context.db.exists.Vote({
          user: { id: userId },
          link: { id: args.linkId },
          })
          if (linkExists) {
          throw new Error(`Already voted for link: ${args.linkId}`)
          }

          // 3
          return context.db.mutation.createVote(
          {
          data: {
          user: { connect: { id: userId } },
          link: { connect: { id: args.linkId } },
          },
          },
          info,
          )
          }





          share|improve this answer















          I have to constrain that a user can submit only one review for a product. I have following design for the non-constrained situation.




          Unfortunately, this is currently not possible with Prisma. There already is an open feature request asking for this functionality, please leave your 👍 on the issue!



          To get that functionality in your application, you'll need to implement that constraint manually on the application layer (e.g. express, apollo-server or graphql-yoga).



          You can take a look at this page of How to GraphQL where there's a similar situation with the User, Link and Vote types. Here's how the resolver to create a Vote and ensurs no votes from that user already exist is implemented with graphql-yoga:



          async function vote(parent, args, context, info) {
          // 1
          const userId = getUserId(context)

          // 2
          const linkExists = await context.db.exists.Vote({
          user: { id: userId },
          link: { id: args.linkId },
          })
          if (linkExists) {
          throw new Error(`Already voted for link: ${args.linkId}`)
          }

          // 3
          return context.db.mutation.createVote(
          {
          data: {
          user: { connect: { id: userId } },
          link: { connect: { id: args.linkId } },
          },
          },
          info,
          )
          }






          share|improve this answer














          share|improve this answer



          share|improve this answer








          edited yesterday

























          answered 2 days ago









          nburk

          14.2k106093




          14.2k106093








          • 1




            Hi @nburk, so the idea is to handle this at the application server level for now. Got it! Have raised a feature request with the same body too. Thanks!
            – devautor
            2 days ago














          • 1




            Hi @nburk, so the idea is to handle this at the application server level for now. Got it! Have raised a feature request with the same body too. Thanks!
            – devautor
            2 days ago








          1




          1




          Hi @nburk, so the idea is to handle this at the application server level for now. Got it! Have raised a feature request with the same body too. Thanks!
          – devautor
          2 days ago




          Hi @nburk, so the idea is to handle this at the application server level for now. Got it! Have raised a feature request with the same body too. Thanks!
          – devautor
          2 days ago












          up vote
          1
          down vote













          I will answer from the MySQL perspective. If you want to enforce that a given customer can only be associated with a given product once, then you should make (cusotmer_id, product_id) a unique key (maybe primary) in the ProductReview table:



          ALTER TABLE ProductReview ADD UNIQUE KEY uk_cust_prod (customer_id, product_id);


          This means that any attempt to insert a record for a given customer and product, when such a relation already exists, would fail at the database level.



          If you also want to add an application level check for this, you of course may do so, and perhaps handle it there first.






          share|improve this answer





















          • Thanks @tim, since Prisma doesn't support this, application level check seems the only way. Would you suggest a "right way" to handle this there?
            – devautor
            2 days ago















          up vote
          1
          down vote













          I will answer from the MySQL perspective. If you want to enforce that a given customer can only be associated with a given product once, then you should make (cusotmer_id, product_id) a unique key (maybe primary) in the ProductReview table:



          ALTER TABLE ProductReview ADD UNIQUE KEY uk_cust_prod (customer_id, product_id);


          This means that any attempt to insert a record for a given customer and product, when such a relation already exists, would fail at the database level.



          If you also want to add an application level check for this, you of course may do so, and perhaps handle it there first.






          share|improve this answer





















          • Thanks @tim, since Prisma doesn't support this, application level check seems the only way. Would you suggest a "right way" to handle this there?
            – devautor
            2 days ago













          up vote
          1
          down vote










          up vote
          1
          down vote









          I will answer from the MySQL perspective. If you want to enforce that a given customer can only be associated with a given product once, then you should make (cusotmer_id, product_id) a unique key (maybe primary) in the ProductReview table:



          ALTER TABLE ProductReview ADD UNIQUE KEY uk_cust_prod (customer_id, product_id);


          This means that any attempt to insert a record for a given customer and product, when such a relation already exists, would fail at the database level.



          If you also want to add an application level check for this, you of course may do so, and perhaps handle it there first.






          share|improve this answer












          I will answer from the MySQL perspective. If you want to enforce that a given customer can only be associated with a given product once, then you should make (cusotmer_id, product_id) a unique key (maybe primary) in the ProductReview table:



          ALTER TABLE ProductReview ADD UNIQUE KEY uk_cust_prod (customer_id, product_id);


          This means that any attempt to insert a record for a given customer and product, when such a relation already exists, would fail at the database level.



          If you also want to add an application level check for this, you of course may do so, and perhaps handle it there first.







          share|improve this answer












          share|improve this answer



          share|improve this answer










          answered 2 days ago









          Tim Biegeleisen

          209k1380129




          209k1380129












          • Thanks @tim, since Prisma doesn't support this, application level check seems the only way. Would you suggest a "right way" to handle this there?
            – devautor
            2 days ago


















          • Thanks @tim, since Prisma doesn't support this, application level check seems the only way. Would you suggest a "right way" to handle this there?
            – devautor
            2 days ago
















          Thanks @tim, since Prisma doesn't support this, application level check seems the only way. Would you suggest a "right way" to handle this there?
          – devautor
          2 days ago




          Thanks @tim, since Prisma doesn't support this, application level check seems the only way. Would you suggest a "right way" to handle this there?
          – devautor
          2 days ago


















           

          draft saved


          draft discarded



















































           


          draft saved


          draft discarded














          StackExchange.ready(
          function () {
          StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53373101%2fprisma-datamodel-primary-key-as-a-combination-of-two-relational-models%23new-answer', 'question_page');
          }
          );

          Post as a guest















          Required, but never shown





















































          Required, but never shown














          Required, but never shown












          Required, but never shown







          Required, but never shown

































          Required, but never shown














          Required, but never shown












          Required, but never shown







          Required, but never shown







          Popular posts from this blog

          MongoDB - Not Authorized To Execute Command

          in spring boot 2.1 many test slices are not allowed anymore due to multiple @BootstrapWith

          How to fix TextFormField cause rebuild widget in Flutter