如何正确返回来自 meteor 方法的响应和错误



我正在尝试从服务器端存在的 meteor 方法返回客户端的响应或错误。该方法本身用于在提交表单时向 API 提交发布请求,我希望在客户端能够返回 API 的响应,以便我可以向用户发送信息,例如,此电子邮件地址已经存在。这是我到目前为止所拥有的。

当我在成功调用该方法时,我在console.log(result);的客户端收到 201 响应,但我还想在 POST 提交给出 400 或任何错误时控制台记录错误。

服务器

/主服务器.js

Meteor.methods({
  addSub(email) {
  var emailAddress = {
      "lists": [
        {
          "id": "1395617465"
        }
      ],
      "email_addresses": [
        {
          "email_address": email
        }
      ]
    }
    HTTP.post('https://api.constantcontact.com/v2/contacts?api_key=<api-key>', {
      headers: {
          'Authorization': 'Bearer <token>',
          'Content-Type': 'application/json'
        },
        data: emailAddress
        }, function (error, response) {
          if ( error ) {
            console.log( error );
            throwError = error;
          } else {
            console.log(response);
            returnResponse = response;            
          }
    });
    return returnResponse;
  }
});

客户端/主.js

Template.info.events({
  'submit #cm-subscribe'(event){
    event.preventDefault();
    var form = event.target;
    var email = form.email.value;
    Meteor.call('addSub', email, (error, result) => { 
        console.log(result);
    });
  }
});

client/main.html

<template name="info">
  <h2>Signup For Our Newsletter</h2>
  <form id="cm-subscribe">
    <input field name="email" type="email" value="email">
    <input type="submit">
  </form>
</template>

1)您的代码当前包含争用条件。 returnResponse的值可能会也可能不会设置,具体取决于回调和 return 语句之间的执行顺序。 相反,您应该使用 Promises,它允许您以同步方式包装异步代码。流星博客上有一篇关于如何做到这一点的好文章。 请注意,这不会影响客户端行为 - 它仍然会像您一样使用回调。

2)您在客户端上看不到错误的原因是Meteor有意将所有正常的Javascript错误(如 HTTP.post 返回的错误)转换为内部服务器错误,以保护数据和源代码。 您应该做的是处理服务器端的所有错误,并为预期的无效响应返回新的Meteor.Error对象。 然后,客户端可以处理这些。 对此的详尽解释可以在流星指南中找到。

像这样:

Meteor.methods({
  addSub(email) {
   return new Promise((resolve, reject) => {
     HTTP.post('<URL>', ..., function(error, response){
        if(!error){
           resolve(response);
        } else if (error == <Some Error Comparison>){
           reject(new Meteor.Error("Email.subscription.invalidEmail", "You put in an invalid email"));
        }
     });
   }); 
  }
}

最新更新