Zend_TimeSync-Working.xml 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!-- Reviewed: no -->
  3. <!-- EN-Revision: 15103 -->
  4. <sect1 id="zend.timesync.working">
  5. <title>Zend_TimeSync の動作</title>
  6. <para>
  7. <classname>Zend_TimeSync</classname> は、指定した任意の
  8. <emphasis role="strong">NTP</emphasis> タイムサーバあるいは
  9. <emphasis role="strong">SNTP</emphasis> タイムサーバから実際の時刻を返します。
  10. 自動的に複数のサーバを処理することができ、シンプルなインターフェイスを提供します。
  11. </para>
  12. <note>
  13. <para>
  14. この章のすべてのサンプルでは、一般に公開されているタイムサーバのひとつを使用します。
  15. 私たちは <emphasis role="strong">0.europe.pool.ntp.org</emphasis>
  16. を使用しますが、あなたの環境にあわせて最も近いタイムサーバを使うことをお勧めします。
  17. 詳細は <ulink url="http://www.pool.ntp.org">http://www.pool.ntp.org</ulink>
  18. を参照ください。
  19. </para>
  20. </note>
  21. <sect2 id="zend.timesync.working.generic">
  22. <title>一般的なタイムサーバへのリクエスト</title>
  23. <para>
  24. タイムサーバから時刻を取得するのはきわめて簡単です。
  25. 単に時刻を取得したいタイムサーバを指定するだけでいいのです。
  26. </para>
  27. <programlisting role="php"><![CDATA[
  28. $server = new Zend_TimeSync('0.pool.ntp.org');
  29. print $server->getDate()->getIso();
  30. ]]>
  31. </programlisting>
  32. <para>
  33. このとき <classname>Zend_TimeSync</classname> の裏側では何が起こっているのでしょうか?
  34. まず、指定したサーバの構文をチェックします。つまり今回の例では、
  35. '<code>0.pool.ntp.org</code>' がタイムサーバのアドレスとして正しい形式かどうかを確認します。
  36. それから、<code>getDate()</code> をコールする際に実際にタイムサーバを設定し、
  37. そのサーバの時刻を返します。<classname>Zend_TimeSync</classname>
  38. は、スクリプトが動作しているサーバの時刻と返された時刻とを比較し、
  39. 修正済みの時刻を表す <classname>Zend_Date</classname> オブジェクトを返します。
  40. </para>
  41. <para>
  42. <classname>Zend_Date</classname> やそのメソッドについての詳細は
  43. <link linkend="zend.date.introduction"><classname>Zend_Date</classname></link>
  44. を参照ください。
  45. </para>
  46. </sect2>
  47. <sect2 id="zend.timesync.working.multiple">
  48. <title>複数のタイムサーバ</title>
  49. <para>
  50. すべてのタイムサーバが常に使用可能で時刻を返してくれるとは限りません。
  51. たとえばメンテナンス中などの理由でサーバに到達できないこともありえます。
  52. タイムサーバから時刻を取得できなかった場合は、例外が発生します。
  53. </para>
  54. <para>
  55. この問題への対策として、<classname>Zend_TimeSync</classname>
  56. では複数のタイムサーバを処理できるようになっており、
  57. ひとつのタイムサーバが使用できなくても代替サーバを使用できるようになっています。
  58. これを利用するには二通りの方法があります。
  59. インスタンスを作成する際にタイムサーバの配列を指定するか、
  60. 後から <code>addServer()</code> メソッドでタイムサーバを追加するかのいずれかです。
  61. </para>
  62. <programlisting role="php"><![CDATA[
  63. $server = new Zend_TimeSync(array('0.pool.ntp.org',
  64. '1.pool.ntp.org',
  65. '2.pool.ntp.org'));
  66. $server->addServer('3.pool.ntp.org');
  67. print $server->getDate()->getIso();
  68. ]]>
  69. </programlisting>
  70. <para>
  71. 追加できるタイムサーバの数には制限はありません。
  72. あるタイムサーバへの接続に失敗すると、
  73. <classname>Zend_TimeSync</classname> はその次に指定されたタイムサーバへの接続を試みます。
  74. </para>
  75. <para>
  76. タイムサーバは複数指定することをお勧めします。
  77. その場合は、タイムサーバに名前をつけなければなりません。
  78. 名前は配列のキーで指定することもできますが、
  79. 初期化やタイムサーバの追加の際に 2 番目のパラメータで指定することもできます。
  80. </para>
  81. <programlisting role="php"><![CDATA[
  82. $server = new Zend_TimeSync(array('generic' => '0.pool.ntp.org',
  83. 'fallback' => '1.pool.ntp.org',
  84. 'reserve' => '2.pool.ntp.org'));
  85. $server->addServer('3.pool.ntp.org', 'additional');
  86. print $server->getDate()->getIso();
  87. ]]>
  88. </programlisting>
  89. <para>
  90. タイムサーバの名前を設定することで、
  91. 特定のタイムサーバにリクエストを送ることができるようになります。
  92. その方法については本章の後半で説明します。
  93. </para>
  94. </sect2>
  95. <sect2 id="zend.timesync.working.protocol">
  96. <title>タイムサーバのプロトコル</title>
  97. <para>
  98. タイムサーバにはさまざまな形式があります。
  99. 公開されているタイムサーバのほとんどは、
  100. <emphasis role="strong">NTP</emphasis> プロトコルを使用するものです。
  101. しかし、それ以外のプロトコルを使用しているサーバもあります。
  102. </para>
  103. <para>
  104. タイムサーバのアドレスに、適切なプロトコルを指定することができます。
  105. <classname>Zend_TimeSync</classname> がサポートしているプロトコルは 2 種類で、
  106. デフォルトのプロトコルは <emphasis role="strong">NTP</emphasis> です。
  107. NTP しか使用しないのなら、先ほどまでの例のようにプロトコル指定を省略しても構いません。
  108. </para>
  109. <programlisting role="php"><![CDATA[
  110. $server = new Zend_TimeSync(array('generic' => 'ntp:\\0.pool.ntp.org',
  111. 'fallback' => 'ntp:\\1.pool.ntp.org',
  112. 'reserve' => 'ntp:\\2.pool.ntp.org'));
  113. $server->addServer('sntp:\\internal.myserver.com', 'additional');
  114. print $server->getDate()->getIso();
  115. ]]>
  116. </programlisting>
  117. <para>
  118. <classname>Zend_TimeSync</classname> は複数のタイムサーバを使用することができます。
  119. 特定のプロトコルだけでなく、個々のサーバに対して別のプロトコルも指定できるようになっています。
  120. </para>
  121. </sect2>
  122. <sect2 id="zend.timesync.working.ports">
  123. <title>タイムサーバのポートの指定</title>
  124. <para>
  125. ウェブにおけるさまざまな他のプロトコルと同様、<emphasis role="strong">NTP</emphasis> や
  126. <emphasis role="strong">SNTP</emphasis> にも標準のポート番号があります。
  127. NTP が使用するポートは <emphasis role="strong">123</emphasis> で、
  128. SNTP は <emphasis role="strong">37</emphasis> を使用します。
  129. </para>
  130. <para>
  131. しかし、実際には使われているポートが違うこともあります。
  132. 使用するポートは、サーバごとにアドレス内に指定することができます。
  133. ポートを指定するには、アドレスに続けてその数字を指定します。
  134. ポートを指定しなかった場合は、<classname>Zend_TimeSync</classname> は標準のポートを使用します。
  135. </para>
  136. <programlisting role="php"><![CDATA[
  137. $server = new Zend_TimeSync(array('generic' => 'ntp:\\0.pool.ntp.org:200',
  138. 'fallback' => 'ntp:\\1.pool.ntp.org'));
  139. $server->addServer('sntp:\\internal.myserver.com:399', 'additional');
  140. print $server->getDate()->getIso();
  141. ]]>
  142. </programlisting>
  143. </sect2>
  144. <sect2 id="zend.timesync.working.options">
  145. <title>タイムサーバ用のオプション</title>
  146. <para>
  147. 実際のところ <classname>Zend_TimeSync</classname> で使用するオプションはひとつだけで、
  148. それも内部的に使用するものです。しかし、必要に応じて任意のオプションを指定して
  149. リクエストを送信することができます。
  150. </para>
  151. <para>
  152. <emphasis role="strong">timeout</emphasis> オプションは、
  153. 応答がない状態が何秒続けば接続に失敗したとみなすのかを表します。
  154. デフォルトの値は <emphasis role="strong">1</emphasis> で、
  155. タイムサーバからの応答が 1 秒以内に返ってこなかった場合に
  156. <classname>Zend_TimeSync</classname> は別のタイムサーバに処理を移します。
  157. </para>
  158. <para>
  159. <code>setOptions()</code> メソッドを使用すれば、任意のオプションを指定することができます。
  160. このメソッドには、オプション名とその値を関連付けた配列を指定します。
  161. 既に設定済みのオプションは、新しい値で上書きされます。
  162. どんなオプションが設定されているのかを知りたい場合は
  163. <code>getOptions()</code> メソッドを使用します。
  164. このメソッドにキーを指定した場合はそのオプションの値を返します。
  165. 指定しなかった場合は、すべてのオプションを返します。
  166. </para>
  167. <programlisting role="php"><![CDATA[
  168. Zend_TimeSync::setOptions(array('timeout' => 3, 'myoption' => 'timesync'));
  169. $server = new Zend_TimeSync(array('generic' => 'ntp:\\0.pool.ntp.org',
  170. 'fallback' => 'ntp:\\1.pool.ntp.org'));
  171. $server->addServer('sntp:\\internal.myserver.com', 'additional');
  172. print $server->getDate()->getIso();
  173. print_r(Zend_TimeSync::getOptions();
  174. print "Timeout = " . Zend_TimeSync::getOptions('timeout');
  175. ]]>
  176. </programlisting>
  177. <para>
  178. ご覧のとおり、<classname>Zend_TimeSync</classname> のオプションは静的なものです。
  179. つまり、すべての <classname>Zend_TimeSync</classname> のインスタンスで同じ設定が共用されます。
  180. </para>
  181. </sect2>
  182. <sect2 id="zend.timesync.working.different">
  183. <title>別のタイムサーバの使用法</title>
  184. <para>
  185. 時間を取得したい場合、デフォルトでは最初に指定したサーバにリクエストを送信します。
  186. しかし、別のタイムサーバを指定してリクエストしたほうが便利なこともあります。
  187. その場合は <code>setServer()</code> メソッドを使用します。
  188. 使用するタイムサーバを指定するには、
  189. タイムサーバのエイリアスをこのメソッドのパラメータとして指定します。
  190. 実際に使われるタイムサーバを知りたい場合は、単に
  191. <code>getServer()</code> メソッドをコールします。
  192. タイムサーバのエイリアスを指定すると、そのタイムサーバを返します。
  193. パラメータを省略した場合は、現在のタイムサーバを返します。
  194. </para>
  195. <programlisting role="php"><![CDATA[
  196. $server = new Zend_TimeSync(array('generic' => 'ntp:\\0.pool.ntp.org',
  197. 'fallback' => 'ntp:\\1.pool.ntp.org'));
  198. $server->addServer('sntp:\\internal.myserver.com', 'additional');
  199. $actual = $server->getServer();
  200. $server = $server->setServer('additional');
  201. ]]>
  202. </programlisting>
  203. </sect2>
  204. <sect2 id="zend.timesync.working.informations">
  205. <title>タイムサーバからの情報</title>
  206. <para>
  207. タイムサーバは、時刻以外にも付加情報を提供します。
  208. この情報を取得するには <code>getInfo()</code> メソッドを使用します。
  209. </para>
  210. <programlisting role="php"><![CDATA[
  211. $server = new Zend_TimeSync(array('generic' => 'ntp:\\0.pool.ntp.org',
  212. 'fallback' => 'ntp:\\1.pool.ntp.org'));
  213. print_r ($server->getInfo());
  214. ]]>
  215. </programlisting>
  216. <para>
  217. どのような情報が得られるのかは、
  218. 使用しているプロトコルやサーバによってさまざまに異なります。
  219. </para>
  220. </sect2>
  221. <sect2 id="zend.timesync.working.exceptions">
  222. <title>例外処理</title>
  223. <para>
  224. すべてのタイムサーバで発生した例外は、ひとつの配列にまとめて返されます。
  225. 発生した例外をすべて処理するには、次の例のようにします。
  226. </para>
  227. <programlisting role="php"><![CDATA[
  228. $serverlist = array(
  229. // 無効なサーバを指定します
  230. 'invalid_a' => 'ntp://a.foo.bar.org',
  231. 'invalid_b' => 'sntp://b.foo.bar.org',
  232. );
  233. $server = new Zend_TimeSync($serverlist);
  234. try {
  235. $result = $server->getDate();
  236. echo $result->getIso();
  237. } catch (Zend_TimeSync_Exception $e) {
  238. $exceptions = $e->get();
  239. foreach ($exceptions as $key => $myException) {
  240. echo $myException->getMessage();
  241. echo '<br />';
  242. }
  243. }
  244. ]]>
  245. </programlisting>
  246. </sect2>
  247. </sect1>