sql >> Databáze >  >> NoSQL >> Redis

Použijte redis k vytvoření chatu v reálném čase se socket.io a NodeJs

Redis je více než úložiště párů klíč-hodnota.

Takže chcete následující:

  • chatové zprávy,
  • dvoučlenné diskuse,
  • nezmínili jste se o časovém omezení, takže předpokládejme, že zprávy archivujete po chvíli,
  • také neříkáte, zda chcete oddělená „vlákna“ mezi dvěma lidmi, jako jsou fóra nebo nepřetržité zprávy, jako je Facebook. Předpokládám kontinuální.

Pro každého uživatele musíte uložit zprávy, které odesílá. Řekněme APP_NAMESPACE:MESSAGES:<USER_ID>:<MESSAGE_ID> . Zde přidáváme userId, abychom mohli snadno načíst všechny zprávy odeslané jedním uživatelem.

A pro každé dva uživatele musíte sledovat jejich konverzace. Jako klíč můžete jednoduše použít jejich uživatelské ID APP_NAMESPACE:CONVERSATIONS:<USER1_ID>-<USER2_ID> . Abyste měli jistotu, že budete mít vždy stejnou sdílenou konverzaci pro dva uživatele, můžete jejich ID seřadit abecedně, takže uživatelé 132 a 145 budou mít oba jako klíč konverzace 132:145

Co tedy ukládat do „rozhovorů“? Použijme seznam:[messageKey, messageKey, messageKey] .

Dobře, ale co je teď messageKey? Kombinace userId výše a messageId (abychom mohli získat skutečnou zprávu).

V zásadě tedy potřebujete dvě věci:

  1. Uložte zprávu a přidělte jí ID
  2. Uložte odkaz na tuto zprávu do příslušné konverzace.

S nodem a standardním klientem redis/hiredis by to bylo něco jako (přeskočím kontroly zjevných chyb atd., a napíšu ES6. Pokud ES6 ještě neumíte číst, stačí to vložit do babelu):

 // assuming the init connects to redis and exports a redisClient
import redisClient from './redis-init';
import uuid from `node-uuid`;


export function storeMessage(userId, toUserId, message) {

  return new Promise(function(resolve, reject) {

    // give it an id.
    let messageId = uuid.v4(); // gets us a random uid.
    let messageKey = `${userId}:${messageId}`;
    let key = `MY_APP:MESSAGES:${messageKey}`;
    client.hmset(key, [
      "message", message,
      "timestamp", new Date(),
      "toUserId", toUserId
    ], function(err) {
      if (err) { return reject(err); }

      // Now we stored the message. But we also want to store a reference to the messageKey
      let convoKey = `MY_APP:CONVERSATIONS:${userId}-${toUserId}`; 
      client.lpush(convoKey, messageKey, function(err) {
        if (err) { return reject(err); }
        return resolve();
      });
    });
  });
}

// We also need to retreive the messages for the users.

export function getConversation(userId, otherUserId, page = 1, limit = 10) {
  return new Promise(function(resolve, reject) {
    let [userId1, userId2] = [userId, otherUserId].sort();
    let convoKey = `MY_APP:CONVERSATIONS:${userId1}-${userId2}`;
    // lets sort out paging stuff. 
    let start = (page - 1) * limit; // we're zero-based here.
    let stop = page * limit - 1;
    client.lrange(convoKey, start, stop, function(err, messageKeys) {

      if (err) { return reject(err); }
      // we have message keys, now get all messages.
      let keys = messageKeys.map(key => `MY_APP:MESSAGES:${key}`);
      let promises = keys.map(key => getMessage(key));
      Promise.all(promises)
      .then(function(messages) {
         // now we have them. We can sort them too
         return resolve(messages.sort((m1, m2) => m1.timestamp - m2.timestamp));
      })
      .catch(reject);
    }); 
  });
}

// we also need the getMessage here as a promise. We could also have used some Promisify implementation but hey.
export function getMessage(key) {
  return new Promise(function(resolve, reject)  {
    client.hgetall(key, function(err, message) {
      if (err) { return reject(err); }
      resolve(message);
    });
  });
}

Teď je to hrubé a nevyzkoušené, ale to je podstata toho, jak to můžete udělat.



  1. Rozdíl mezi protokolem Redis AOF a Tarantool WAL

  2. Jak vytvořit vnořený index v MongoDB?

  3. Mohu upravit existující index v MongoDB, aniž bych jej zrušil?

  4. Nejjednodušší způsob, jak zkopírovat/klonovat instanci mongoose dokumentu?