Meteor 錯(cuò)誤

2022-06-30 13:58 更新

錯(cuò)誤

僅使用瀏覽器標(biāo)準(zhǔn)的 alert() 對(duì)話窗去警告用戶他們的提交有錯(cuò)誤有那么一點(diǎn)不令人滿意,而且顯然不是一個(gè)良好的用戶體驗(yàn)。我們可以做得更好。

相反,讓我們建立一個(gè)更加靈活的錯(cuò)誤報(bào)告機(jī)制,來更好地在不打斷流程的情況下告訴用戶到底發(fā)生了什么。

我們要實(shí)現(xiàn)一個(gè)簡(jiǎn)單的系統(tǒng),在窗口右上角顯示新的錯(cuò)誤信息,類似于流行的 Mac OS 應(yīng)用程序 Growl

介紹本地集合

一開始,我們需要?jiǎng)?chuàng)建一個(gè)集合來存儲(chǔ)我們的錯(cuò)誤。既然錯(cuò)誤只與當(dāng)前會(huì)話相關(guān),而且不需要以任何方式長(zhǎng)久存在,我們要在這做點(diǎn)新鮮的事兒,創(chuàng)建一個(gè)本地集合(Local collection)。這意味著,錯(cuò)誤 Errors 集合將會(huì)只存在于瀏覽器中,并且將不作任何嘗試去同步回服務(wù)器。

為實(shí)現(xiàn)它,我們?cè)?client 文件夾中創(chuàng)建錯(cuò)誤(確保這集合只在客戶端存在),我們將它的 MongoDB 集合命名為 null (因?yàn)榧系臄?shù)據(jù)將不會(huì)保存在服務(wù)器端的數(shù)據(jù)庫(kù)中):

// 本地(僅客戶端)集合
Errors = new Mongo.Collection(null);

一開始,我們應(yīng)該建立一個(gè)可以儲(chǔ)存錯(cuò)誤的集合。介于錯(cuò)誤只是對(duì)于當(dāng)前的會(huì)話,我們將采用及時(shí)性集合。這就意味著錯(cuò)誤集合只存在于當(dāng)前的瀏覽器,該集合不會(huì)與服務(wù)端同步。

既然集合已經(jīng)建立了,我們可以創(chuàng)建一個(gè) throwError 函數(shù)用來添加新的錯(cuò)誤。我們不需要擔(dān)心 allowdeny 或其他任何的安全考慮,因?yàn)檫@個(gè)集合對(duì)于當(dāng)前用戶是“本地的”。

throwError = function(message) {
  Errors.insert({message: message});
};

使用本地集合去存儲(chǔ)錯(cuò)誤的優(yōu)勢(shì)在于,就像所有集合一樣,它是響應(yīng)性的————意味著我們可以以顯示其他任何集合數(shù)據(jù)的同樣的方式,去響應(yīng)性地顯示錯(cuò)誤。

顯示錯(cuò)誤

我們將在主布局的頂部插入錯(cuò)誤信息:

<template name="layout">
  <div class="container">
    {{> header}}
    {{> errors}}
    <div id="main" class="row-fluid">
      {{> yield}}
    </div>
  </div>
</template>

讓我們現(xiàn)在在 errors.html 中創(chuàng)建 errorserror 模版:

<template name="errors">
  <div class="errors">
    {{#each errors}}
      {{> error}}
    {{/each}}
  </div>
</template>

<template name="error">
  <div class="alert alert-danger" role="alert">
    <button type="button" class="close" data-dismiss="alert">&times;</button>
    {{message}}
  </div>
</template>

Twin 模版

你可能注意到我們?cè)谝粋€(gè)文件里面建立了兩個(gè)模板。直到現(xiàn)在我們一直在遵循“一個(gè)文件, 一個(gè)模板”的標(biāo)準(zhǔn),但對(duì)于 Meteor 而言,我們把所有模板放在同一個(gè)文件里也是一樣的(但是這會(huì)讓 main.html 的代碼變得非?;靵y!)。

在當(dāng)前情況下,因?yàn)檫@兩個(gè)錯(cuò)誤模板都比較小,我們破例將它們放在一個(gè)文件里,使我們的 repo 代碼庫(kù)更干凈些。

我們只需要加上我們的模板 helper 就可以大功告成了!

Template.errors.helpers({
  errors: function() {
    return Errors.find();
  }
});

你可以嘗試手動(dòng)測(cè)試我們的新錯(cuò)誤消息了。打開瀏覽器控制臺(tái),并輸入:

throwError("我就是一個(gè)錯(cuò)誤!");

兩種類型的錯(cuò)誤

在這一點(diǎn)上,重要的是要把“應(yīng)用級(jí)(app-level)”的錯(cuò)誤和“代碼級(jí)(code-level)”的錯(cuò)誤區(qū)別開來。

應(yīng)用級(jí)錯(cuò)誤一般是由用戶觸發(fā),用戶從而能夠?qū)ΠY采取行動(dòng)。這些包括像驗(yàn)證錯(cuò)誤、權(quán)限錯(cuò)誤、“未找到”錯(cuò)誤,等等。這是是那種你希望展現(xiàn)給用戶,以幫助他們解決他們剛剛遇到的任何問題的錯(cuò)誤。

代碼級(jí)錯(cuò)誤,作為另一種類型,是實(shí)際的代碼 bug 非期待情況下觸發(fā)的,你可能不希望將錯(cuò)誤直接呈現(xiàn)給用戶,而是通過比如第三方錯(cuò)誤跟蹤服務(wù)(比如 Kadira)去跟蹤錯(cuò)誤。

在本章中,我們將重點(diǎn)放在處理第一種類型的錯(cuò)誤,而不是去抓蟲子(bug)。

創(chuàng)建錯(cuò)誤

我們知道怎樣顯示錯(cuò)誤,但我們還需要在發(fā)現(xiàn)之前去觸發(fā)錯(cuò)誤。實(shí)際上我們已經(jīng)建立了良好的錯(cuò)誤情境:重復(fù)帖子的警告。我們簡(jiǎn)單地用新的 throwError 函數(shù)去替代 postSubmit 事件 helper 中的 alert 調(diào)用:

Template.postSubmit.events({
  'submit form': function(e) {
    e.preventDefault();

    var post = {
      url: $(e.target).find('[name=url]').val(),
      title: $(e.target).find('[name=title]').val()
    };

    Meteor.call('postInsert', post, function(error, result) {
      // display the error to the user and abort
      if (error)
        return throwError(error.reason);

      // show this result but route anyway
      if (result.postExists)
        throwError('This link has already been posted');

      Router.go('postPage', {_id: result._id});
    });
  }
});

既然到此,我們也針對(duì) postEdit 事件 helper 做同樣的事情:

Template.postEdit.events({
  'submit form': function(e) {
    e.preventDefault();

    var currentPostId = this._id;

    var postProperties = {
      url: $(e.target).find('[name=url]').val(),
      title: $(e.target).find('[name=title]').val()
    }

    Posts.update(currentPostId, {$set: postProperties}, function(error) {
      if (error) {
        // display the error to the user
        throwError(error.reason);
      } else {
        Router.go('postPage', {_id: currentPostId});
      }
    });
  },
  //...
});

親自試一試:嘗試建立一個(gè)帖子并輸入 URL http://meteor.com。因?yàn)檫@個(gè) URL 已經(jīng)存在了,你可以看到:

清理錯(cuò)誤

你會(huì)注意到錯(cuò)誤消息在幾秒鐘后自動(dòng)消失。這是因?yàn)楸緯_頭我們往樣式表中添加的一些 CSS 而產(chǎn)生的魔力:

@keyframes fadeOut {
  0% {opacity: 0;}
  10% {opacity: 1;}
  90% {opacity: 1;}
  100% {opacity: 0;}
}

//...

.alert {
  animation: fadeOut 2700ms ease-in 0s 1 forwards;
  //...
}

我們定義了一個(gè)有四幀透明度屬性變化(分別是 0%、10%、90% 和 100% 貫穿整個(gè)動(dòng)畫過程)的 fadeOut CSS 動(dòng)畫,并附在了 .alert class 樣式。

動(dòng)畫時(shí)長(zhǎng)為 2700 毫秒,使用 ease-in 效果,有 0 秒延遲,運(yùn)行一次,當(dāng)動(dòng)畫完成時(shí),最后停留在最后一幀。

動(dòng)畫 vs 動(dòng)畫

你也許在想為什么我們使用基于 CSS 的動(dòng)畫(預(yù)先定義,并且在我們應(yīng)用控制以外),而不用 Meteor 本身來控制動(dòng)畫。

雖然 Meteor 的確提供插入動(dòng)畫的支持,但是我們想在本章專注于錯(cuò)誤。所以我們現(xiàn)在使用“笨”CSS 動(dòng)畫,我們把比較炫麗的東西留在以后的動(dòng)畫章節(jié)。

這可以工作了,但是如果你要觸發(fā)多個(gè)錯(cuò)誤(比如,通過提交三次同一個(gè)連接),你會(huì)看到錯(cuò)誤信息會(huì)堆疊在一起。

這是因?yàn)殡m然 .alert 元素在視覺上消失了,但仍存留在 DOM 中。我們需要修正這個(gè)問題。

這正是 Meteor 發(fā)光的情形。由于 Errors 集合是響應(yīng)性的,我們要做的就是將舊的錯(cuò)誤從集合中刪除!

我們用 Meteor.setTimeout 指定在一定時(shí)間(當(dāng)前情形,3000毫秒)后執(zhí)行一個(gè)回調(diào)函數(shù)。

Template.errors.helpers({
  errors: function() {
    return Errors.find();
  }
});

Template.error.onRendered(function() {
  var error = this.data;
  Meteor.setTimeout(function () {
    Errors.remove(error._id);
  }, 3000);
});

一旦模板在瀏覽器中渲染完畢,onRendered 回調(diào)函數(shù)被觸發(fā)。其中,this 是指當(dāng)前模板實(shí)例,而 this.data 是當(dāng)前被渲染的對(duì)象的數(shù)據(jù)(這種情況下是,一個(gè)錯(cuò)誤)。

尋求驗(yàn)證

到現(xiàn)在為止,我們還沒有對(duì)表單進(jìn)行任何驗(yàn)證。至少,我們想讓用戶為新帖子提供 URL 和標(biāo)題。那么我們確保他們這么做。

我們要做兩件事:第一,我們給任何有問題的表單字段的父 div 標(biāo)簽一個(gè)特別的 has-error CSS class。第二,我們?cè)谧侄蜗路斤@示一個(gè)有用的錯(cuò)誤消息。

首先,我們要準(zhǔn)備 postSubmit 模板來包含這些新 helper:

<template name="postSubmit">
  <form class="main form">
    <div class="form-group {{errorClass 'url'}}">
      <label class="control-label" for="url">URL</label>
      <div class="controls">
          <input name="url" id="url" type="text" value="" placeholder="Your URL" class="form-control"/>
          <span class="help-block">{{errorMessage 'url'}}</span>
      </div>
    </div>
    <div class="form-group {{errorClass 'title'}}">
      <label class="control-label" for="title">Title</label>
      <div class="controls">
          <input name="title" id="title" type="text" value="" placeholder="Name your post" class="form-control"/>
          <span class="help-block">{{errorMessage 'title'}}</span>
      </div>
    </div>
    <input type="submit" value="Submit" class="btn btn-primary"/>
  </form>
</template>

注意我們傳遞參數(shù)(分別是 urltitle)到每個(gè) helper。這讓我們兩次重復(fù)使用同一個(gè) helper,基于參數(shù)修改它的行為。

現(xiàn)在到了有趣的部分:使這些 helper 真正做點(diǎn)什么事情。

我們會(huì)用會(huì)話 Session 去存儲(chǔ)包含任何潛在錯(cuò)誤的 postSubmitErrors 對(duì)象。當(dāng)用戶使用表單時(shí),這個(gè)對(duì)象會(huì)改變,也就是響應(yīng)性地更新表單代碼和內(nèi)容。

首先,當(dāng) postSubmit 模板被創(chuàng)建時(shí),我們初始化對(duì)象。這確保用戶不會(huì)看到上次訪問該頁(yè)面時(shí)遺留下的舊的錯(cuò)誤消息。

然后定義我們的兩個(gè)模板 helper,緊盯 Session.get('postSubmitErrors')field 屬性(fieldurltitle 取決于我們?nèi)绾握{(diào)用 helper)。

errorMessage 只是返回消息本身,而 errorClass 檢查消息是否存在,如果為真返回 has-error。

Template.postSubmit.onCreated(function() {
  Session.set('postSubmitErrors', {});
});

Template.postSubmit.helpers({
  errorMessage: function(field) {
    return Session.get('postSubmitErrors')[field];
  },
  errorClass: function (field) {
    return !!Session.get('postSubmitErrors')[field] ? 'has-error' : '';
  }
});

//...

你可以測(cè)試 helper 是否工作正常,打開瀏覽器控制臺(tái)并輸入以下代碼:

Session.set('postSubmitErrors', {title: 'Warning! Intruder detected. Now releasing robo-dogs.'});

下一步將 postSubmitErrors Session 會(huì)話對(duì)象綁在表單上。

開始之前,我們?cè)?posts.js 中添加一個(gè)新的 validatePost 函數(shù)來監(jiān)視 post 對(duì)象,返回一個(gè)包含任何錯(cuò)誤相關(guān)消息的(即,titleurl 字段是否未填寫)errors 對(duì)象:

//...

validatePost = function (post) {
  var errors = {};

  if (!post.title)
    errors.title = "請(qǐng)?zhí)顚憳?biāo)題";

  if (!post.url)
    errors.url =  "請(qǐng)?zhí)顚?URL";

  return errors;
}

//...

我們通過 postSubmit 事件 helper 去調(diào)用這個(gè)函數(shù):

Template.postSubmit.events({
  'submit form': function(e) {
    e.preventDefault();

    var post = {
      url: $(e.target).find('[name=url]').val(),
      title: $(e.target).find('[name=title]').val()
    };

    var errors = validatePost(post);
    if (errors.title || errors.url)
      return Session.set('postSubmitErrors', errors);

    Meteor.call('postInsert', post, function(error, result) {
      // 向用戶顯示錯(cuò)誤信息并終止
      if (error)
        return throwError(error.reason);

      // 顯示這個(gè)結(jié)果且繼續(xù)跳轉(zhuǎn)
      if (result.postExists)
        throwError('This link has already been posted');

      Router.go('postPage', {_id: result._id});
    });
  }
});

注意如果出現(xiàn)任何錯(cuò)誤,我們用 return 終止 helper 執(zhí)行,而不是我們要實(shí)際地返回這個(gè)值。

服務(wù)器端驗(yàn)證

我們還沒有完成。我們?cè)?em>客戶端驗(yàn)證 URL 和標(biāo)題是否存在,但是在服務(wù)器端呢?畢竟,還會(huì)有人仍然嘗試通過瀏覽器控制臺(tái)輸入一個(gè)空帖子來手動(dòng)調(diào)用 postInsert 方法。

即使我們不需要在服務(wù)器端顯示任何錯(cuò)誤消息,但是我們依然要利用好那個(gè) validatePost 函數(shù)。除了這次我們?cè)?postInsert 方法內(nèi)調(diào)用它,而不只是在事件 helper:

Meteor.methods({
  postInsert: function(postAttributes) {
    check(this.userId, String);
    check(postAttributes, {
      title: String,
      url: String
    });

    var errors = validatePost(postAttributes);
    if (errors.title || errors.url)
      throw new Meteor.Error('invalid-post', "你必須為你的帖子填寫標(biāo)題和 URL");

    var postWithSameLink = Posts.findOne({url: postAttributes.url});
    if (postWithSameLink) {
      return {
        postExists: true,
        _id: postWithSameLink._id
      }
    }

    var user = Meteor.user();
    var post = _.extend(postAttributes, {
      userId: user._id,
      author: user.username,
      submitted: new Date()
    });

    var postId = Posts.insert(post);

    return {
      _id: postId
    };
  }
});

再次,用戶正常情況下不必看到“你必須 為你的帖子填寫標(biāo)題和 URL”的消息。這僅會(huì)在當(dāng)用戶想繞過我們煞費(fèi)苦心創(chuàng)建的用戶界面而直接使用瀏覽器的情況下,才會(huì)顯示。

為了測(cè)試,打開瀏覽器控制臺(tái),輸入一個(gè)沒有 URL 的帖子:

Meteor.call('postInsert', {url: '', title: 'No URL here!'});

如果我們完成得順利的話,你會(huì)得到一堆嚇人的代碼 和“你必須為你的帖子填寫標(biāo)題和 URL”的消息。

編輯驗(yàn)證

為了更加完善,我們?yōu)樘?em>編輯表單添加相同的驗(yàn)證。代碼看起來十分相似。首先,是模板:

<template name="postEdit">
  <form class="main form">
    <div class="form-group {{errorClass 'url'}}">
      <label class="control-label" for="url">URL</label>
      <div class="controls">
          <input name="url" id="url" type="text" value="{{url}}" placeholder="Your URL" class="form-control"/>
          <span class="help-block">{{errorMessage 'url'}}</span>
      </div>
    </div>
    <div class="form-group {{errorClass 'title'}}">
      <label class="control-label" for="title">Title</label>
      <div class="controls">
          <input name="title" id="title" type="text" value="{{title}}" placeholder="Name your post" class="form-control"/>
          <span class="help-block">{{errorMessage 'title'}}</span>
      </div>
    </div>
    <input type="submit" value="Submit" class="btn btn-primary submit"/>
    <hr/>
    <a class="btn btn-danger delete" href="#">Delete post</a>
  </form>
</template>

然后是模板 helper:

Template.postEdit.onCreated(function() {
  Session.set('postEditErrors', {});
});

Template.postEdit.helpers({
  errorMessage: function(field) {
    return Session.get('postEditErrors')[field];
  },
  errorClass: function (field) {
    return !!Session.get('postEditErrors')[field] ? 'has-error' : '';
  }
});

Template.postEdit.events({
  'submit form': function(e) {
    e.preventDefault();

    var currentPostId = this._id;

    var postProperties = {
      url: $(e.target).find('[name=url]').val(),
      title: $(e.target).find('[name=title]').val()
    }

    var errors = validatePost(postProperties);
    if (errors.title || errors.url)
      return Session.set('postEditErrors', errors);

    Posts.update(currentPostId, {$set: postProperties}, function(error) {
      if (error) {
        // 向用戶顯示錯(cuò)誤消息
        throwError(error.reason);
      } else {
        Router.go('postPage', {_id: currentPostId});
      }
    });
  },

  'click .delete': function(e) {
    e.preventDefault();

    if (confirm("Delete this post?")) {
      var currentPostId = this._id;
      Posts.remove(currentPostId);
      Router.go('postsList');
    }
  }
});

就像我們?yōu)樘犹峤槐韱嗡龅?,我們也想在服?wù)器端驗(yàn)證帖子。請(qǐng)記住我們不是在用一個(gè)方法去編輯帖子,而是直接從客戶端的 update 調(diào)用。

這意味著我們必須添加一個(gè)新的 deny 回調(diào)函數(shù):

//...

Posts.deny({
  update: function(userId, post, fieldNames, modifier) {
    var errors = validatePost(modifier.$set);
    return errors.title || errors.url;
  }
});

//...

注意的是參數(shù) post 是指已存在的帖子。我們想驗(yàn)證更新,所以我們?cè)?modifier$set 屬性中調(diào)用 validatePost(就像是 Posts.update({$set: {title: ..., url: ...}}))。

這會(huì)正常運(yùn)行,因?yàn)?modifier.$set 像整個(gè) post 對(duì)象那樣包含同樣兩個(gè) titleurl 屬性。當(dāng)然,這也的確意味著只部分更新 title 或者 url 是不行的,但是實(shí)踐中不應(yīng)有問題。

你也許注意到,這是我們第二個(gè) deny 回調(diào)。當(dāng)添加多個(gè) deny 回調(diào)時(shí),如果任何一個(gè)回調(diào)返回 true,運(yùn)行就會(huì)失敗。在此例中,這意味著 update 只有在面向 titleurl 兩個(gè)字段時(shí)才會(huì)成功,并且這些字段不能為空。

以上內(nèi)容是否對(duì)您有幫助:
在線筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號(hào)
微信公眾號(hào)

編程獅公眾號(hào)