Data null after saving entity with Moongose and Gr

2019-07-27 03:27发布

问题:

When saving an entity with mongoose and graphql the following happens:

First method to save:

create(budgetProps){
   const budget = new Budget(budgetProps);
   return budget.save();
}

The result is as follows:

{
  "data": {
    "addBudget": {
      "_id": "59fbdefaa7b0a81180dd2c9c",
      "tiempoAproximado": 2245.5,
      "User": {
        "name": null,
        "organization": null
      },
      "Vehicle": {
        "name": null,
        "type": null
      }
    }
  }
}

Using this method:

create(budgetProps){
  const budget = new Budget(budgetProps);
  return budget.save().then((res)=>{
           Budget.findById(res._id)
              .populate('User')
              .populate('Vehicle')
              .exec((err, newBudget)=> {
                 return newBudget;
              });
        });
  },

I get the following:

{
  "data": {
    "addBudget": null
  }
}

This is the Schema:

 const typeDefs = `
   scalar Date

   input UserInput {
      _id: ID,
      name: String,
      organization: String,
      phones: [String],
      emails: [String],
      type: String,
      password: String,
      percentaje: String
   }

   input VehicleDescriptionInput {
      es: String,
      en: String
   }

   input VehicleInput{
      _id: ID,
      name: String,
      passengers: Int,
      largeBags: Int,
      smallBags: Int,
      doors: Int,
      type: String,
      status: Boolean,
      imagesUrls: [String],
      description: VehicleDescriptionInput
   }

   input FinalTotalCostInput {
      es: String,
      en: String   
   }

   input BudgetTotalCostInput {
      es: String,
      en: String
   }

   input BudgetInput {
      finalTotalCost: FinalTotalCostInput,
      budgetTotalCost: BudgetTotalCostInput,
      destinoInicial: String,
      destinoFinal: String,
      tiempoAproximado: Float,
      distancia: Float,
      tollCost: Float,
      tolls: [String],
      budgetDate: Date,
      aprove: Boolean,
      User: UserInput,
      Vehicle: VehicleInput
   }

   type Mutation {
      addBudget(data: BudgetInput): Budget
   }
`;

Here is the resolver:

Mutation: {
  addBudget: (_, {data}) =>{
    return BudgetController.create(data);  
  }
},

Finally here is the mutation with its variables:

mutation addBudget($budget: BudgetInput) {
  addBudget(data: $budget) {
    _id
    User{
      name
      organization
    }
    Vehicle{
      name
      type
    }
  }
}

{
  "budget": {
    "finalTotalCost": {
      "es": "100 peso",
      "en": "10 dolars"
    },
    "budgetTotalCost": {
      "es": "80 peso",
      "en": "8 dolars"
    },
    "destinoInicial": "Queretaro",
    "destinoFinal": "Sonora",
    "tiempoAproximado": 2245.5,
    "distancia": 100.565,
    "tollCost": 20.5,
    "tolls": [
      "GDL",
      "Marina",
      "Culap",
      "Malageña"
    ],
    "budgetDate": "2017/07/21",
    "aprove": false,
    "User": {
      "_id": "59fbcc42aa82460924e5fbad"
    },
    "Vehicle": {
      "_id": "59fbcbe4aa82460924e5fbac"
    }
  }
}

The entity is stored properly in the database, when Console.log the result of the populated search results are correct then I do not understand what is happening.

You can find the whole app in the following link: GitHub Repo

回答1:

You're mixing Promises and callbacks. exec() will return a Promise, but only if doesn't have any arguments passed to it. Additionally, you need to return the Promise that's returned by exec().

return budget.save().then((res) => {
  return Budget.findById(res._id) // missing return here
    .populate('User')
    .populate('Vehicle')
    .exec() // don't need anything else
})

You can clean this up a little more:

return budget.save()
  .then(res => Budget.findById(res._id)
    .populate('User')
    .populate('Vehicle')
    .exec())

If you need to transform the results returned by findById before turning them over to the client:

return budget.save()
  .then(res => Budget.findById(res._id)
    .populate('User')
    .populate('Vehicle')
    .exec())
  .then(res => {
    res.foo = 'Foo'
    return res
  })