<div dir="ltr">If what you need is not exactly a down-to-the-microsecond time stamp but a temporally ordered globally unique ID, this<div><br></div><div> <a href="https://github.com/boundary/flake">https://github.com/boundary/flake</a></div><div><br></div><div>could be part of the solution, I suppose. </div><div><br></div></div><div class="gmail_extra"><br clear="all"><div><div class="gmail_signature">Regards,<br>Michael Turner<br>Executive Director<br>Project Persephone<br>K-1 bldg 3F<br>7-2-6 Nishishinjuku<br>Shinjuku-ku Tokyo 160-0023<br>Tel: +81 (3) 6890-1140<br>Fax: +81 (3) 6890-1158<br>Mobile: +81 (90) 5203-8682<br><a href="mailto:turner@projectpersephone.org" target="_blank">turner@projectpersephone.org</a><br><a href="http://www.projectpersephone.org/" target="_blank">http://www.projectpersephone.org/</a><br><br>"Love does not consist in gazing at each other, but in looking outward together in the same direction." -- Antoine de Saint-Exupéry</div></div>
<br><div class="gmail_quote">On Wed, Apr 22, 2015 at 10:34 AM, Daniel <span dir="ltr"><<a href="mailto:liudanking@gmail.com" target="_blank">liudanking@gmail.com</a>></span> wrote:<br><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">The background of my problem is as follows:<br>
<br>
I use AWS dynamoDB (<a href="http://aws.amazon.com/cn/dynamodb/" target="_blank">http://aws.amazon.com/cn/dynamodb/</a>) to store ejabbed chat messages. DynamoDB is a NoSQL storage and needs a *key* to look up a record. In my application, I construct the *key* as <jid, erlang:now()> where jid is the hash key and erlang:now() is the range key (<a href="http://docs.aws.amazon.com/amazondynamodb/latest/developerguide/WorkingWithTables.html" target="_blank">http://docs.aws.amazon.com/amazondynamodb/latest/developerguide/WorkingWithTables.html</a>). The reason that I prefer to use erlang:now() as the range key is that it will be convenient to query history messages according to timestamp. Unfortunately, <jid, erlang:now()> may be not unique in erlang cluster, so I am looking for a global *erlang:now()* function that generate global unique timestamp in erlang cluster.<br>
<br>
BTW, small probability of <jid, erlang:now()> collision is not a big problem in my application, so I am still using <jid, erlang:now()> as the *key*.<br>
<div class="HOEnZb"><div class="h5"><br>
<br>
> On Apr 22, 2015, at 8:59 AM, Michael Turner <<a href="mailto:michael.eugene.turner@gmail.com">michael.eugene.turner@gmail.com</a>> wrote:<br>
><br>
> "slapping "lamport clock" on it is reductive."<br>
><br>
> -- it was just a suggestion, in case something like that might work. My suggestion was not intended to be all-inclusive or a panacea. Slapping "reductive" on my suggestion is ... well, reductive?<br>
><br>
> "Not come with a pet solution to push through."<br>
><br>
> It's not my "pet solution". I don't have a "pet solution." I don't even know what this guy's problem is, and you don't either. So how can I have a pet solution if there's no way to know what the solution is in the first place?<br>
><br>
> I just happen to know that Erlang/OTP has this feature that's been in "beta" for what seems to be a decade or more, one that exposes a kind of Lamport clock functionality (even though the documentation fails to call it that, which might be why it's still waiting for enough user input to refine the interface -- people who go looking for something like that in Erlang/OTP are not finding it.) seq_trace might be part of /a/ solution to his problem. But since we don't know what his problem really is, I'm just making suggestions.<br>
><br>
> Understand?<br>
><br>
><br>
><br>
> Regards,<br>
> Michael Turner<br>
> Executive Director<br>
> Project Persephone<br>
> K-1 bldg 3F<br>
> 7-2-6 Nishishinjuku<br>
> Shinjuku-ku Tokyo 160-0023<br>
> Tel: <a href="tel:%2B81%20%283%29%206890-1140" value="+81368901140">+81 (3) 6890-1140</a><br>
> Fax: <a href="tel:%2B81%20%283%29%206890-1158" value="+81368901158">+81 (3) 6890-1158</a><br>
> Mobile: <a href="tel:%2B81%20%2890%29%205203-8682" value="+819052038682">+81 (90) 5203-8682</a><br>
> <a href="mailto:turner@projectpersephone.org">turner@projectpersephone.org</a><br>
> <a href="http://www.projectpersephone.org/" target="_blank">http://www.projectpersephone.org/</a><br>
><br>
> "Love does not consist in gazing at each other, but in looking outward together in the same direction." -- Antoine de Saint-Exupéry<br>
><br>
> On Tue, Apr 21, 2015 at 9:20 PM, Fred Hebert <<a href="mailto:mononcqc@ferd.ca">mononcqc@ferd.ca</a>> wrote:<br>
> On 04/21, Michael Turner wrote:<br>
> "Lamport/vector clocks and other similar ones operate on *causality*, but<br>
> this partial ordering is not the only one available or workable."<br>
><br>
> Whether it's "workable" depends on what's desired. Sorting by {Node,<br>
> Timestamp} is not accurate if causality matters and clocks have drifted out<br>
> of synch. As they will. Hence Lamport's work, and the work of others. And<br>
> if causality doesn't matter, well, I wonder: why bother? Unless you just<br>
> want a rough idea of when certain things happened, in which case {Node,<br>
> Timestamp} can give you a /total/ order that's, if anything, more accurate<br>
> than what you need.<br>
><br>
><br>
> That's not necessarily true. Let's see for different options and when they can be useful.<br>
><br>
> - Lamport/vector clocks: causality. I wan to track the logical dependencies of changes.<br>
> - `{Node, Timestamp}`: I have lots of local events (say HTTP requests and responses in logs) and want to see *when* they happen and how far apart. The timestmap might need to be monotonic, but the per-node value lets me impose a logical order, track some density over time (assuming I at least have NTP working), and so on.<br>
> - {Shard, Timestamp}: I require a total order, but for events within a sharded data set.<br>
> - {Cluster, Timestamp}: Each cluster I run might belong to specific customers or whatever, or run a specific set of hardware, or be a logical division. In any case, it's possible they have their own time or id service and I may want a partial or total order based on the events within that cluster, without worrying I might want to compare cross-cluster activity.<br>
> - {Region, Timestamp}: Similar to the above, but by geographical area. I might decide that I need a total order on some form of transactions and will run a service, but for latency (and if real world allows it), I won't try to synchronize my time across data-centers or large geographical areas.<br>
><br>
> All of these 'labelled timestamps' *are* a partial order. They only define it on some label. I.e. you can sort all timestamps within a node/shard/cluster/region, but can't do it across boundaries.<br>
><br>
> There are other avenues that even combine some of them; One interesting case is inspired by Google's Chubby and CRDTs: You use a timestamp synchronized by NTP, guaranteeing you a maximal drift interval. You then add in a lamport clock whenever two events happen within too close of an interval that we cannot guarantee from the system clocks they truly happened apart.<br>
><br>
> The lamport clock is mergeable in a deterministic way that is also commutative and idempotent (that's a CRDT!), and acts as a tie-breaker between events that happen at too close together.<br>
><br>
> This way you get reliable timestamps when you can, and when you suddenly can't, you get a form of causality (or global monotonicity) to break things up.<br>
><br>
> slapping "lamport clock" on it is reductive. It's a good way to track some levels of causality, but has its limitations. If you only *need* node-local accuracy and you have access to a monotonic clock, it might be far less work to just slap the monotonic clock into things than weave the logical clock through everything, and obtain the same logical result in the end (plus more information). Maybe it's not the best solution either.<br>
><br>
> But really, if we want to make good recommendations, we have to ask what the user needs. Not come with a pet solution to push through.<br>
><br>
<br>
</div></div></blockquote></div><br></div>