[svn-inject] Installing original source of nanobloggertrackback upstream/0.2.0
authorgregor herrmann <gregoa@debian.org>
Sat, 25 Feb 2006 18:13:56 +0000 (18:13 -0000)
committergregor herrmann <gregoa@debian.org>
Sat, 25 Feb 2006 18:13:56 +0000 (18:13 -0000)
COPYING [new file with mode: 0644]
README [new file with mode: 0644]
entry.diff [new file with mode: 0644]
footer.txt [new file with mode: 0644]
header.txt [new file with mode: 0644]
nb_tb-setup [new file with mode: 0755]
nb_tb.css [new file with mode: 0644]
tb.cgi [new file with mode: 0755]
trackbacklist.sh [new file with mode: 0644]
trackbackping.sh [new file with mode: 0644]

diff --git a/COPYING b/COPYING
new file mode 100644 (file)
index 0000000..dc39cc4
--- /dev/null
+++ b/COPYING
@@ -0,0 +1,340 @@
+                   GNU GENERAL PUBLIC LICENSE
+                      Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.
+        51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+                           Preamble
+
+  The licenses for most software are designed to take away your
+freedom to share and change it.  By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users.  This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it.  (Some other Free Software Foundation software is covered by
+the GNU Library General Public License instead.)  You can apply it to
+your programs, too.
+
+  When we speak of free software, we are referring to freedom, not
+price.  Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+this service if you wish), that you receive source code or can get it
+if you want it, that you can change the software or use pieces of it
+in new free programs; and that you know you can do these things.
+
+  To protect your rights, we need to make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+  For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have.  You must make sure that they, too, receive or can get the
+source code.  And you must show them these terms so they know their
+rights.
+
+  We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+  Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software.  If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+  Finally, any free program is threatened constantly by software
+patents.  We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary.  To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+  The precise terms and conditions for copying, distribution and
+modification follow.
+\f
+                   GNU GENERAL PUBLIC LICENSE
+   TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+  0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License.  The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language.  (Hereinafter, translation is included without limitation in
+the term "modification".)  Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope.  The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+  1. You may copy and distribute verbatim copies of the Program's
+source code as you receive it, in any medium, provided that you
+conspicuously and appropriately publish on each copy an appropriate
+copyright notice and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+  2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+    a) You must cause the modified files to carry prominent notices
+    stating that you changed the files and the date of any change.
+
+    b) You must cause any work that you distribute or publish, that in
+    whole or in part contains or is derived from the Program or any
+    part thereof, to be licensed as a whole at no charge to all third
+    parties under the terms of this License.
+
+    c) If the modified program normally reads commands interactively
+    when run, you must cause it, when started running for such
+    interactive use in the most ordinary way, to print or display an
+    announcement including an appropriate copyright notice and a
+    notice that there is no warranty (or else, saying that you provide
+    a warranty) and that users may redistribute the program under
+    these conditions, and telling the user how to view a copy of this
+    License.  (Exception: if the Program itself is interactive but
+    does not normally print such an announcement, your work based on
+    the Program is not required to print an announcement.)
+\f
+These requirements apply to the modified work as a whole.  If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works.  But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+  3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+    a) Accompany it with the complete corresponding machine-readable
+    source code, which must be distributed under the terms of Sections
+    1 and 2 above on a medium customarily used for software interchange; or,
+
+    b) Accompany it with a written offer, valid for at least three
+    years, to give any third party, for a charge no more than your
+    cost of physically performing source distribution, a complete
+    machine-readable copy of the corresponding source code, to be
+    distributed under the terms of Sections 1 and 2 above on a medium
+    customarily used for software interchange; or,
+
+    c) Accompany it with the information you received as to the offer
+    to distribute corresponding source code.  (This alternative is
+    allowed only for noncommercial distribution and only if you
+    received the program in object code or executable form with such
+    an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it.  For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable.  However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+\f
+  4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License.  Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+  5. You are not required to accept this License, since you have not
+signed it.  However, nothing else grants you permission to modify or
+distribute the Program or its derivative works.  These actions are
+prohibited by law if you do not accept this License.  Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+  6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions.  You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+  7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License.  If you cannot
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all.  For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices.  Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+\f
+  8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded.  In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+  9. The Free Software Foundation may publish revised and/or new versions
+of the General Public License from time to time.  Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number.  If the Program
+specifies a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation.  If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+  10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission.  For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this.  Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+                           NO WARRANTY
+
+  11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
+FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW.  EXCEPT WHEN
+OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
+PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
+OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.  THE ENTIRE RISK AS
+TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU.  SHOULD THE
+PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
+REPAIR OR CORRECTION.
+
+  12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
+INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
+OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
+TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
+YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
+PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGES.
+
+                    END OF TERMS AND CONDITIONS
+\f
+           How to Apply These Terms to Your New Programs
+
+  If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+  To do so, attach the following notices to the program.  It is safest
+to attach them to the start of each source file to most effectively
+convey the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+    <one line to give the program's name and a brief idea of what it does.>
+    Copyright (C) <year>  <name of author>
+
+    This program is free software; you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation; either version 2 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License
+    along with this program; if not, write to the Free Software
+    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+
+
+Also add information on how to contact you by electronic and paper mail.
+
+If the program is interactive, make it output a short notice like this
+when it starts in an interactive mode:
+
+    Gnomovision version 69, Copyright (C) year  name of author
+    Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+    This is free software, and you are welcome to redistribute it
+    under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License.  Of course, the commands you use may
+be called something other than `show w' and `show c'; they could even be
+mouse-clicks or menu items--whatever suits your program.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the program, if
+necessary.  Here is a sample; alter the names:
+
+  Yoyodyne, Inc., hereby disclaims all copyright interest in the program
+  `Gnomovision' (which makes passes at compilers) written by James Hacker.
+
+  <signature of Ty Coon>, 1 April 1989
+  Ty Coon, President of Vice
+
+This General Public License does not permit incorporating your program into
+proprietary programs.  If your program is a subroutine library, you may
+consider it more useful to permit linking proprietary applications with the
+library.  If this is what you want to do, use the GNU Library General
+Public License instead of this License.
diff --git a/README b/README
new file mode 100644 (file)
index 0000000..3885bf0
--- /dev/null
+++ b/README
@@ -0,0 +1,116 @@
+nanobloggertrackback - adding trackback functionality to nanoblogger
+====================================================================
+
+Copyright: gregor herrmann <gregor+debian@comodo.priv.at>, 2005
+License: Released under the GNU GPL (cf. COPYING)
+
+Purpose
+-------
+
+nanobloggertrackback adds active and passive trackback functionality to
+nanoblogger, i.e. it let's the user (automatically or manually) send
+trackback pings to referenced blog entries and receive trackback pings from
+other sites.
+Additionally it shows received trackbacks in the blog pages and offers the
+possibility to delete them.
+
+Installation
+------------
+
+a) Automatic
+------------
+
+nb_tb-setup should automagically copy and patch all necessary files.
+If it breaks something: Don't get nervous, it creates a backup of your
+$BLOG_DIR in $HOME/nb.bak.
+It does _not_ copy the nanoblogger plugins though (they are supposed to be
+used systemwide, and nb_tb-setup just adjust one specific blog). - See
+below under "Manual".
+
+b) Manual
+---------
+
+* Create the following directories:
+  $BLOG_DIR/tb
+  $BLOG_DIR/tb/data
+  $BLOG_DIR/tb/rss
+  $BLOG_DIR/tb/pings
+  
+  ./data and ./rss must be writeable for the user under which the webserver
+  is running!
+
+* Change the beginning of tb.cgi:
+  Replace BLOGDIR, PASSWORD, and EMAIL with meaningful values
+
+* Copy tb.cgi to your webserver's cgi-bin directory or somewhere else, where
+  the webserver can execute it.
+
+* You may want to add the contents of nb_tb.css to your currently used
+  nanoblogger css file.
+
+* Copy header.txt and footer.txt to $BLOG_DIR/tb.
+
+* Change $BLOGDIR/templates/entry.htm
+  Use the patch in entry.diff or add the required lines manually:
+
+  | <a href="${TB_PATH}?__mode=list&amp;tb_id=$NB_EntryID" onclick="window.open(this.href, 'trackback', 'width=480,height=480,scrollbars=yes,status=yes'); return false">TrackBack</a>
+
+    e.g. between the categories and the comments line
+
+  <div class="tb">
+  $NB_TrackBacks
+  </div>
+
+    e.g. after the comments line, if you want to see the trackbacks directly
+    under each entry.
+
+  <!--
+  <rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+           xmlns:dc="http://purl.org/dc/elements/1.1/"
+           xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/">
+  <rdf:Description
+      rdf:about="${ARCHIVES_PATH}$NB_EntryPermalink"
+      dc:title="$NB_EntryTitle"
+      dc:identifier="${ARCHIVES_PATH}$NB_EntryPermalink" />
+      trackback:ping="${TB_PATH}/$NB_EntryID"
+  </rdf:RDF>
+
+    at the end of entry.htm
+
+* In order to activate the whole stuff you have to add two lines to your
+  blog.conf:
+  TB_PATH="$CGIURL/tb.cgi"
+    where $CGIURL is the full URL to tb.cgi, e.g. http://www.example.com/cgi-bin
+  SEND_TRACKBACK_PING="1"
+    if you want nanoblogger to send trackback pings automatically to each
+    URL found in a modified entry.
+  Absolute links have to be turned on in blog.conf:
+  ABSOLUTE_LINKS="1"
+
+* If nobody on your system has done this already: Copy trackbacklist.sh and
+  trackbackping.sh to nanobloggers plugins/entry/mod/ directory.
+
+Usage
+-----
+
+a) Receiving trackback pings
+----------------------------
+
+If everything is set up correctly other bloggers can send you trackback
+pings, either manually (the trackback URL can be found now in each entry of
+your blog) or automatically (that's what the RDF stuff in entry.htm is for).
+You can view trackbacks from others either directly under each entry or by
+clicking on the Trackbacks link; there you can (after logging in) also
+delete trackbacks.
+You are informed by a short email if someone else sends a trackback ping to
+one of your posts.
+(Now you know what the $PASSWORD and $EMAIL variables are for ;-).)
+
+b) Sending trackback pings
+--------------------------
+
+If you want to send a trackback ping manually go to
+$CGIURL/tb.cgi?__mode=send_form
+If you turn on SEND_TRACKBACK_PING in blog.conf trackback pings are sent
+automatically to each referenced URL in a modified blog entry (but only
+once; a list is kept in $BLOG_DIR/tb/pings).
diff --git a/entry.diff b/entry.diff
new file mode 100644 (file)
index 0000000..6d8196f
--- /dev/null
@@ -0,0 +1,23 @@
+--- entry.htm  2005-09-15 21:40:14.000000000 +0200
++++ entry_new.htm      2005-09-15 22:44:32.000000000 +0200
+@@ -14,4 +14,20 @@
+ $([ ! -z "$NB_EntryCategories" ] && echo "| Categories: $NB_EntryCategories" |sed -e '{$ s/\,$//; }')
++| <a href="${TB_PATH}?__mode=list&amp;tb_id=$NB_EntryID" onclick="window.open(this.href, 'trackback', 'width=480,height=480,scrollbars=yes,status=yes'); return false">TrackBack</a>
+ <!-- <br /><a href="${BASE_URL}$BLOG_URL_ACTION$ARCHIVES_DIR/$permalink_file#comments">Comments</a> -->
++<div class="tb">
++$NB_TrackBacks
+ </div>
+ </div>
++</div>
++
++<!--
++<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
++         xmlns:dc="http://purl.org/dc/elements/1.1/"
++         xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/">
++<rdf:Description
++    rdf:about="${ARCHIVES_PATH}$NB_EntryPermalink"
++    dc:title="$NB_EntryTitle"
++    dc:identifier="${ARCHIVES_PATH}$NB_EntryPermalink" />
++    trackback:ping="${TB_PATH}/$NB_EntryID"
++</rdf:RDF>
++-->
diff --git a/footer.txt b/footer.txt
new file mode 100644 (file)
index 0000000..c81b8a8
--- /dev/null
@@ -0,0 +1,7 @@
+<hr size="1" color="#CCC">     
+
+<div align="center" class="header"><a class="trackback" href="http://www.movabletype.org/docs/mtmanual_trackback.html#trackback" target="_blank">What is TrackBack?</a></a>
+</div>
+
+</body>
+</html>
diff --git a/header.txt b/header.txt
new file mode 100644 (file)
index 0000000..e810230
--- /dev/null
@@ -0,0 +1,88 @@
+<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
+
+<html>
+<head>
+<meta http-equiv="Content-Type" content="text/html; charset=iso-8859-1" />
+
+<title>TrackBack Display</title>
+
+<style type="text/css">
+
+body {
+       background-color:#FFF;
+       margin: 0px 0px 0px 0px;
+       padding: 0px 10px 0px 10px;
+       color:#333;
+       text-align:center;
+       }
+
+
+       A                       { font-family:verdana, arial; font-size:10px; color: #333; text-decoration: underline; background-color:#FFF;} 
+       A:link          { color: #333; text-decoration: underline; background-color:#FFF;} 
+       A:visited       { color: #333; text-decoration: underline; background-color:#FFF;} 
+       A:active        { color: #CC9966; background-color:#FFF;} 
+       A:hover         { color: #CC9966; background-color:#FFF;} 
+
+       A.delete                        { font-family:verdana, arial; font-size:10px; color: #CC3300;  background-color:#FFF; font-weight:bold;} 
+       A.delete:link           { color: #CC3300; background-color:#FFF;} 
+       A.delete:visited        { color: #CC3300; background-color:#FFF;} 
+       A.delete:active         { color: #CC9966; background-color:#FFF;} 
+       A.delete:hover          { color: #CC9966; background-color:#FFF;} 
+
+
+#body {
+       background-color:#FFF;  
+/*     font-family:georgia, verdana, arial, sans-serif;*/
+       font-family:verdana, arial, sans-serif;
+       font-size:11px;
+       line-height:15px;
+       color:#333;
+       padding:10px;   
+       text-align:left;
+       }
+
+.ping-url, .head {
+       font-family:verdana, arial, sans-serif;
+       color:#333;
+       font-size:10px;
+       padding-bottom:5px;
+       }
+
+.footer {
+       font-family:verdana, arial, sans-serif;
+       color:#333;
+       font-size:10px;
+       padding-bottom:5px;
+       margin-bottom: 15px;
+       }
+
+       
+.excerpt {
+/*     font-family:georgia, verdana, arial, sans-serif;*/
+       font-family:verdana, arial, sans-serif;
+       color:#333;
+       font-size:11px;
+       padding-bottom:5px;
+       padding-left:10px;
+       padding-right:10px;
+       }
+
+.url {
+/*     font-family:georgia, verdana, arial, sans-serif;*/
+       font-family:verdana, arial, sans-serif;
+       color:#333;
+       background-color:#EEE;
+       font-size:11px;
+       padding:5px;
+       margin-bottom: 15px;
+       border-top: 1px solid #CCC;
+       border-bottom: 1px solid #CCC;
+       }
+
+</style>
+
+</head>
+
+<body>
+
+<div id="body">                                
diff --git a/nb_tb-setup b/nb_tb-setup
new file mode 100755 (executable)
index 0000000..ad1d077
--- /dev/null
@@ -0,0 +1,147 @@
+#!/bin/sh
+
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+# 
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+# 
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301 USA
+#
+# Copyright: gregor herrmann <gregor+debian@comodo.priv.at>, 2005
+
+###debug
+###set -x
+
+set -e
+
+RPATH=$(/usr/bin/realpath $0)
+CURDIR=$(/usr/bin/dirname $RPATH)
+
+CGIDIR=
+CGIURL=
+BLOGDIR=
+PASSWORD=
+EMAIL=
+CSS=
+
+# get config
+while [ -z "$CGIDIR" ]; do
+       echo -n "cgi-bin directory (e.g. /home/sepp/public_html/cgi-bin): "
+       read CGIDIR
+done
+
+echo "You have to make sure that scripts can be executed there."
+read -p "Press return to continue" DUMMY
+
+while [ -z "$CGIURL" ]; do
+       echo -n "cgi-bin URL (e.g. http://info.example.org/~sepp/cgi-bin): "
+       read CGIURL
+done
+
+while [ -z "$BLOGDIR" ]; do
+       echo -n "nanoblogger directory (e.g. /home/sepp/public_html/blog): "
+       read BLOGDIR
+done
+
+echo "This script will create a /tb and /tb/data and /tb/rss directories
+there. The user under which your webserver is running must have write
+permissions in the latter two. Please check after the end of this script."
+read -p "Press return to continue" DUMMY
+
+while [ -z "$PASSWORD" ]; do
+       echo -n "password for managing trackback entries (use a new one; stored in plaintext): "
+       read -s PASSWORD; echo ""
+done
+
+while [ -z "$EMAIL" ]; do
+       echo -n "email address for mail notifies (e.g. <user@host.example>): "
+       read EMAIL
+done
+
+
+# test
+echo -n "Checking user input ... "
+( [ ! -d $CGIDIR ] || [ ! -w $CGIDIR ] ) && echo "$CGIDIR does not exist or is no directory or is not writeable." && exit 1
+( [ ! -d $BLOGDIR ] || [ ! -w $BLOGDIR ] ) && echo "$BLOGDIR does not exist or is no directory or is not writeable." && exit 1
+( [ ! -d $BLOGDIR/styles ] || [ ! -w $BLOGDIR/styles ] ) && echo "$BLOGDIR/styles does not exist or is no directory or is not writeable." && exit 1
+( [ ! -d $BLOGDIR/templates ] || [ ! -w $BLOGDIR/templates ] ) && echo "$BLOGDIR/templates does not exist or is no directory or is not writeable." && exit 1
+echo "OK"
+
+# backup
+echo -n "A backup of $BLOGDIR will be copied to $HOME/nb.bak. If this script breaks you may want to use it, otherwise please delete it afterwards. ... "
+cp -ar $BLOGDIR $HOME/nb.bak || exit 1
+echo "OK"
+
+# create dirs
+echo -n "Creating directories ... "
+mkdir -m 6775 -p $BLOGDIR/tb || exit 1
+mkdir -m 6777 -p $BLOGDIR/tb/data || exit 1
+mkdir -m 6777 -p $BLOGDIR/tb/rss || exit 1
+mkdir -p $BLOGDIR/tb/pings || exit 1
+echo "OK"
+
+# add css
+echo -n "Adding css ... "
+CSS=$(perl -ne 'print $1 if /BLOG_CSS="styles\/(.+)"/' $BLOGDIR/blog.conf) || exit 1
+cat $CURDIR/nb_tb.css >> $BLOGDIR/styles/$CSS || exit 1
+echo "OK"
+
+# copy cgi and change
+echo -n "Copying and changing cgi script ... "
+cp --preserve=mode $CURDIR/tb.cgi $CGIDIR/ || exit 1
+perl -pi -e "s~BLOGDIR~$BLOGDIR~; s~PASSWORD~$PASSWORD~; s~EMAIL~$EMAIL~;" $CGIDIR/tb.cgi || exit 1
+echo "OK"
+
+# copy footer/header
+echo -n "Copying header und footer ... "
+cp $CURDIR/header.txt $BLOGDIR/tb/ || exit 1
+cp $CURDIR/footer.txt $BLOGDIR/tb/ || exit 1
+echo "OK"
+
+# change entry.htm
+echo -n "Changing entry.htm ... "
+patch --quiet $BLOGDIR/templates/entry.htm $CURDIR/entry.diff || exit 1
+echo "OK"
+
+# change blog.conf
+echo -n "Adding TB_PATH to blog.conf ... "
+echo "# Trackback CGI" >> $BLOGDIR/blog.conf || exit 1
+echo "TB_PATH=\"$CGIURL/tb.cgi\"" >> $BLOGDIR/blog.conf || exit 1
+echo "OK"
+
+echo -n "Adding SEND_TRACKBACK_PING to blog.conf ... "
+echo "# Send trackback pings automatically" >> $BLOGDIR/blog.conf || exit 1
+echo "SEND_TRACKBACK_PING=\"1\"" >> $BLOGDIR/blog.conf || exit 1
+echo "OK"
+
+# change absolute links in blog.conf
+echo -n "Turning on absolute links in blog.conf ... "
+perl -pi -e "s/.*/ABSOLUTE_LINKS=\"1\"/ if /ABSOLUTE_LINKS/" $BLOGDIR/blog.conf || exit 1
+echo "OK"
+
+# the end
+echo ""
+echo "Finished!"
+echo ""
+echo "Enjoy the new possibilities - view the trackback links in your entries, send trackback pings with $CGIURL/tb.cgi?__mode=send_form, etc."
+echo ""
+echo "And don't forget to run 'nb -b $BLOGDIR -u all' NOW."
+echo ""
+echo "(Have you deleted $HOME/nb.bak already?)"
+
+CURDIR=
+CGIDIR=
+CGIURL=
+BLOGDIR=
+PASSWORD=
+EMAIL=
+CSS=
+
+exit 0
diff --git a/nb_tb.css b/nb_tb.css
new file mode 100644 (file)
index 0000000..c279707
--- /dev/null
+++ b/nb_tb.css
@@ -0,0 +1,23 @@
+       .tb     {
+               font-family:verdana, arial, sans-serif; 
+               font-size: x-small; 
+               color: #000000; 
+               background:#ccc;
+               margin-top:10px;
+               }
+       
+       .tb-title {
+               font-weight:bold;
+       }
+       
+       .tb-header {
+               border-top:1px dotted black;
+       }
+       
+       .tb-footer {
+               font-size:0.9em;
+       }
+       
+       .tb-excerpt {
+               padding-left: 10px;
+       }
diff --git a/tb.cgi b/tb.cgi
new file mode 100755 (executable)
index 0000000..d1b154a
--- /dev/null
+++ b/tb.cgi
@@ -0,0 +1,529 @@
+#!/usr/bin/perl -w
+# Copyright 2002 Benjamin Trott.
+# This code is released under the Artistic License.
+#
+# Changed by gregor herrmann <gregor+debian@comodo.priv.at>, 2005, 2006
+
+use strict;
+use warnings;
+
+my $DataDir = "BLOGDIR/tb/data";
+my $RSSDir = "BLOGDIR/tb/rss";
+my $GenerateRSS = 1;
+my $Header = "BLOGDIR/tb/header.txt";
+my $Footer = "BLOGDIR/tb/footer.txt";
+my $Password = "PASSWORD";  
+my $MailNotify = 'EMAIL';
+my $NBDataDir = "BLOGDIR/data";
+
+use vars qw( $VERSION );
+$VERSION = '1.02';
+
+use CGI qw( :standard );
+use File::Spec::Functions;
+
+my $mode = param('__mode');
+unless ($mode) {
+    my $tb_id = munge_tb_id(get_tb_id());
+    respond_exit("No TrackBack ID (tb_id)") unless $tb_id;
+    respond_exit("No valid TrackBack ID:" . $tb_id) unless is_valid_tb_id($tb_id);
+    my $i = { map { $_ => scalar param($_) } qw(title excerpt url blog_name) };
+    $i->{title} ||= $i->{url};
+    $i->{timestamp} = time;
+    respond_exit("No URL (url)") unless $i->{url};
+    my $data = load_data($tb_id);
+    unshift @$data, $i;
+    store_data($tb_id, $data);
+    if ($GenerateRSS && open(FH, ">" . catfile($RSSDir, $tb_id . '.xml'))) {
+        print FH generate_rss($tb_id, $data, 15);
+        close FH;
+    }
+    my $me = url();
+#    open(SENDMAIL, "|/usr/lib/sendmail -oi -t -odq") or die "Can't fork for sendmail: $!\n";
+    open(SENDMAIL, "| /usr/sbin/exim4 -ti") or respond_exit( "Can't fork for sendmail: $!\n");
+    print SENDMAIL <<"EOF";
+From: trackback $MailNotify
+To: $MailNotify
+Subject: new trackback ping
+
+You received a new trackback ping:
+$me?__mode=list&tb_id=$tb_id
+EOF
+    close(SENDMAIL)     or warn "sendmail didn't close nicely";
+    respond_exit();
+} elsif ($mode eq 'list') {
+    my $tb_id = munge_tb_id(get_tb_id());
+    die("No TrackBack ID (tb_id)") unless $tb_id;
+    my $me = url();
+    print header(), from_file($Header), <<URL;
+<div class="url">TrackBack URL for this entry:
+<div class="ping-url">$me/$tb_id</div>
+</div>
+URL
+    my $data = load_data($tb_id);
+    my $tmpl = <<TMPL;
+<a target="new" href="%s">%s</a><br />
+<div class="head">&#187;  %s</div>
+<div class="excerpt">"%s"</div>
+<div class="footer">Tracked: %s %s</div>
+TMPL
+    my $i = 0;
+    require POSIX;
+    my $logged_in = is_logged_in();
+    for my $item (@$data) {
+        my $ts = POSIX::strftime("%F %T %Z", localtime $item->{timestamp});
+        printf $tmpl,
+            $item->{url}, $item->{title},
+            $item->{blog_name} || "[No blog name]",
+            $item->{excerpt} || "[No excerpt]",
+            $ts,
+            $logged_in ? qq(<a class="delete" href="$me?__mode=delete&tb_id=$tb_id&index=$i">[DELETE]</a>) : '';
+        $i++;
+    }
+    unless ($logged_in) {
+        print <<HTML;
+<div align="right">[Is this your site? <a href="$me?__mode=login">Log in</a> to delete pings.]</div>
+HTML
+    } else {
+        print <<HTML;
+<div align="right">[<a href="$me?__mode=logout">Log out</a>]</div>
+HTML
+    }
+    print from_file($Footer);
+
+} elsif ($mode eq 'show') {
+    my $tb_id = munge_tb_id(get_tb_id());
+    die("No TrackBack ID (tb_id)") unless $tb_id;
+    my $data = load_data($tb_id);
+    print header();
+    if(@{$data} > 0) {
+      print <<URL;
+  <div class="tb-title">Trackbacks for this entry:</div>
+URL
+    }
+    my $tmpl = <<TMPL;
+<div class="tb-header">&#187;  %s:
+<a href="%s">%s</a>
+</div>
+<div class="tb-excerpt">"%s"</div>
+<div class="tb-footer">Tracked: %s</div>
+TMPL
+    my $i = 0;
+    require POSIX;
+    for my $item (@$data) {
+        my $ts = POSIX::strftime("%F %T %Z", localtime $item->{timestamp});
+        printf $tmpl,
+            $item->{blog_name} || "[No blog name]", 
+            $item->{url}, 
+            $item->{title},
+            $item->{excerpt} || "[No excerpt]",
+            $ts;
+        $i++;
+    }
+
+} elsif ($mode eq 'delete') {
+    die "You are not authorized" unless is_logged_in();
+    my $tb_id = munge_tb_id(get_tb_id());
+    die("No TrackBack ID (tb_id)") unless $tb_id;
+    my $data = load_data($tb_id);
+    my $index = param('index') || 0;
+    splice @$data, $index, 1;
+    store_data($tb_id, $data);
+    print redirect(url() . "?__mode=list&tb_id=$tb_id");
+} elsif ($mode eq 'rss') {
+    my $tb_id = munge_tb_id(get_tb_id());
+    respond_exit("No TrackBack ID (tb_id)") unless $tb_id;
+    my $data = load_data($tb_id);
+    respond_exit(undef, generate_rss($tb_id, $data));
+} elsif ($mode eq 'send_ping') {
+    require LWP::UserAgent;
+    my $ua = LWP::UserAgent->new;
+    $ua->agent("TrackBack/$VERSION");
+    my @qs = map $_ . '=' . encode_url(param($_) || ''),
+             qw( title url excerpt blog_name );
+    my $ping = param('ping_url') or ping_form_exit("No ping URL");
+    my $req;
+    if ($ping =~ /\?/) {
+        $req = HTTP::Request->new(GET => $ping . '&' . join('&', @qs));
+    } else {
+        $req = HTTP::Request->new(POST => $ping);
+        $req->content_type('application/x-www-form-urlencoded');
+        $req->content(join('&', @qs));
+    }
+    my $res = $ua->request($req);
+    ping_form_exit("HTTP error: " . $res->status_line) unless $res->is_success;
+    my($e, $msg) = $res->content =~ m!<error>(\d+).*<message>(.+?)</message>!s;
+    $e ? ping_form_exit("Error: $msg") : ping_form_exit("Ping successfuly sent");
+} elsif ($mode eq 'send_form') {
+    ping_form_exit();
+} elsif ($mode eq 'login') {
+    print header(), login_form();
+} elsif ($mode eq 'do_login') {
+    my $key = param('key');
+    unless ($key eq $Password) {
+        print header(), login_form("Invalid login");
+        exit;
+    }
+    require CGI::Cookie;
+    my @alpha = ('a'..'z', 'A'..'Z', 0..9);
+    my $salt = join '', map $alpha[rand @alpha], 1..2;
+    my $cookie = CGI::Cookie->new(-name => 'key',
+        -value => crypt($key, $salt));
+    print header(-cookie => $cookie), from_file($Header),
+        "Logged in", from_file($Footer);
+} elsif ($mode eq 'logout') {
+    require CGI::Cookie;
+    my $cookie = CGI::Cookie->new(-name => 'key', -value => '',
+        -expire => '-1y');
+    print header(-cookie => $cookie), login_form("Logged out");
+}
+
+sub get_tb_id {
+    my $tb_id = param('tb_id');
+    unless ($tb_id) {
+        if (my $pi = path_info()) {
+            ($tb_id = $pi) =~ s!^/!!;
+        }
+    }
+    $tb_id;
+}
+
+sub munge_tb_id {
+    my($id) = @_;
+    return '' unless $id;
+    $id =~ tr/a-zA-Z0-9/_/cs;
+    $id;
+}
+
+sub is_valid_tb_id {
+    my($id) = @_;   
+    return '' unless $id;
+    my @nb_files=<$NBDataDir/*>;
+    map { $_ =~ s/^$NBDataDir\/// } @nb_files;
+    map { $_ = munge_tb_id($_) } @nb_files;
+    print @nb_files;
+    $id =~ s/^e//;
+    return (grep(/$id/, @nb_files) ? '1' : '');
+}   
+
+sub is_logged_in {
+    require CGI::Cookie;
+    my %cookies = CGI::Cookie->fetch;
+    return unless $cookies{key};
+    my $key = $cookies{key}->value || return;
+    $key eq crypt $Password, substr $key, 0, 2;
+}
+
+sub load_data {
+    my($tb_id) = @_;
+    my $tb_file = catfile($DataDir, $tb_id . '.stor');
+    require Storable;
+    scalar eval { Storable::retrieve($tb_file) } || [];
+}
+
+sub store_data {
+    my($tb_id, $data) = @_;
+    my $tb_file = catfile($DataDir, $tb_id . '.stor');
+    require Storable;
+    Storable::store($data, $tb_file);
+}
+
+sub generate_rss {
+    my($tb_id, $data, $limit) = @_;
+    my $rss = qq(<rss version="0.91"><channel><title>TB: $tb_id</title>\n);
+    my $max = $limit ? $limit - 1 : $#$data;
+    for my $i (@{$data}[0..$max]) {
+        $rss .= sprintf "<item>%s%s%s</item>\n", xml('title', $i->{title}),
+                xml('link', $i->{url}), xml('description', $i->{excerpt}) if $i;
+    }
+    $rss . qq(</channel></rss>);
+}
+
+sub respond_exit {
+    print "Content-Type: text/xml\n\n";
+    print qq(<?xml version="1.0" encoding="iso-8859-1"?>\n<response>\n);
+    if ($_[0]) {
+        printf qq(<error>1</error>\n%s\n), xml('message', $_[0]);
+    } else {
+        print qq(<error>0</error>\n) . ($_[1] ? $_[1] : '');
+    }
+    print "</response>\n";
+    exit;
+}
+
+sub ping_form_exit {
+    print header(), from_file($Header);
+    print "@_" if @_;
+    print <<HTML;
+<h2>Send a TrackBack ping</h2>
+<form method="post"><input type="hidden" name="__mode" value="send_ping" />
+<table border="0" cellspacing="3" cellpadding="0">
+<tr><td>TrackBack Ping URL:</td><td><input name="ping_url" size="60" /></td></tr>
+<tr><td>&nbsp;</td></tr>
+<tr><td>Title:</td><td><input name="title" size="35" /></td></tr>
+<tr><td>Blog name:</td><td><input name="blog_name" size="35" /></td></tr>
+<tr><td>Excerpt:</td><td><input name="excerpt" size="60" /></td></tr>
+<tr><td>Permalink URL:</td><td><input name="url" size="60" /></td></tr>
+</table>
+<input type="submit" value="Send">
+</form>
+HTML
+    print from_file($Footer);
+    exit;
+}
+
+sub login_form {
+    my $str = from_file($Header);
+    $str .= "<p>@_</p>" if @_;
+    $str .= <<HTML . from_file($Footer);
+<form method="post">
+<input type="hidden" name="__mode" value="do_login" />
+Password: <input name="key" type="password" />
+<input type="submit" value="Log in" />
+</form>
+HTML
+    $str;
+}
+my(%Map, $RE);
+BEGIN {
+    %Map = ('&' => '&amp;', '"' => '&quot;', '<' => '&lt;', '>' => '&gt;');
+    $RE = join '|', keys %Map;
+}
+sub xml {
+    (my $s = defined $_[1] ? $_[1] : '') =~ s!($RE)!$Map{$1}!g;
+    "<$_[0]>$s</$_[0]>\n";
+}
+
+sub encode_url {
+    (my $str = $_[0]) =~ s!([^a-zA-Z0-9_.-])!uc sprintf "%%%02x", ord($1)!eg;
+    $str;
+}
+
+sub from_file {
+    my($file) = @_;
+    local *FH;
+    open FH, $file;
+    my $c;
+    { local $/; $c = <FH> }
+    close FH;
+    $c;
+}
+
+__END__
+
+=head1 NAME
+
+tb-standalone - Standalone TrackBack
+
+=head1 DESCRIPTION
+
+The standalone TrackBack tool serves two purposes: 1) it allows non-Movable
+Type users to use TrackBack with the tool of their choice, provided they meet
+the installation requirements; 2) it serves as a reference point to aid
+developers in implementing TrackBack in their own systems. This tool is a
+single CGI script that accepts TrackBack pings through HTTP requests, stores
+the pings locally in the filesystem, and can return a list of pings either
+in RSS or in a browser-viewable format. It can also be used to send pings
+to other sites.
+
+It is released under the Artistic License. The terms of the Artistic License
+are described at I<http://www.perl.com/language/misc/Artistic.html>.
+
+=head1 REQUIREMENTS
+
+You'll need a webserver capable of running CGI scripts (this means, for
+example, that this won't work with BlogSpot-hosted blogs). You'll also need
+perl, and the following Perl modules:
+
+=over 4
+
+=item * File::Spec
+
+=item * Storable
+
+=item * CGI
+
+=item * CGI::Cookie
+
+=item * LWP
+
+=back
+
+The first four are core modules as of perl 5.6.0, I believe, and LWP is
+installed on most hosts. Furthermore LWP is only required if you wish to
+B<send> TrackBack pings.
+
+=head1 INSTALLATION
+
+Installation of the standalone TrackBack tool is very simple. It's just one
+CGI script, F<tb.cgi>, along with two text files that define the header and
+footer HTML for the public list of TrackBack pings.
+
+=over 4
+
+=item 1. Configure tb.cgi
+
+You'll need to edit the script to change the I<$DataDir>, I<$RSSDir>,
+and I<$Password> settings.
+
+B<BE SURE TO CHANGE THE I<$Password> BEFORE INSTALLING THE TOOL.>
+
+I<$DataDir> is the path to the directory where the TrackBack data
+files will be stored; I<$RSSDir> is the path to the directory where the static
+RSS files will be generated; I<$Password> is your secret password that will
+allow you to delete TrackBack pings, when logged in.
+
+After setting I<$DataDir> and I<$RSSDir>, you'll need to create both of these
+directories and make them writeable by the user running the CGI scripts. In
+most cases, this means that you must set the permissions on these directories
+to 777.
+
+=item 2. Upload Files
+
+After editing the settings, upload F<tb.cgi>, F<header.txt>, and F<footer.txt>
+in ASCII mode to your webserver into a directory where you can run CGI
+scripts. Set the permissions on F<tb.cgi> to 755.
+
+=back
+
+=head1 USAGE
+
+=head2 Sending Pings
+
+To send pings from the tool, go to the following URL:
+
+    http://yourserver.com/cgi-bin/tb.cgi?__mode=send_form
+
+where I<http://yourserver.com/cgi-bin/tb.cgi> is the URL where you
+installed F<tb.cgi>. Fill out the fields in the form, then press I<Send>.
+
+=head2 Receiving Pings
+
+To use the tool in your existing pages, you'll need to do two things:
+
+=over 4
+
+=item 1. Link to TrackBack listing
+
+First, you'll need to add a link to each of your weblog entries with a
+link to the list of TrackBack pings for that entry. You can do this by
+adding the following HTML to your template:
+
+    <a href="http://yourserver.com/cgi-bin/tb.cgi?__mode=list&tb_id=[TrackBack ID]" onclick="window.open(this.href, 'trackback', 'width=480,height=480,scrollbars=yes,status=yes'); return false">TrackBack</a>
+
+You'll need to change C<http://yourserver.com/cgi-bin/tb.cgi> to the proper
+URL for I<tb.cgi> on your server. And, depending on the weblogging tool that
+you use, you'll need to change C<[TrackBack ID]> to a unique post ID. See
+the L<conversion table below|Conversion Table> to determine the proper tag to
+use for the tool that you use, to generate a unique post ID.
+
+=item 2. Add RDF
+
+TrackBack uses RDF embedded within your web page to auto-discover
+TrackBack-enabled entries on your pages. It also uses this information when
+building a threaded list of a cross-weblog "discussion". For these purposes,
+it is useful to embed the RDF into your page.
+
+Add the following to your weblog template so that it is displayed for each
+of the entries on your page:
+
+    <!--
+    <rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+             xmlns:dc="http://purl.org/dc/elements/1.1/"
+             xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/">
+    <rdf:Description
+        rdf:about="[Entry Permalink]"
+        dc:title="[Entry Title]"
+        dc:identifier="[Entry Permalink]" />
+        trackback:ping="http://yourserver.com/cgi-bin/tb.cgi/[TrackBack ID]"
+    </rdf:RDF>
+    -->
+
+As above, the tags that you should use for C<[TrackBack ID]>,
+C<[Entry Title]>, and C<[Entry Permalink]> all depend on the weblogging tool
+that you are using. See the L<conversion table below|Conversion Table>.
+
+=back
+
+=head2 Conversion Table
+
+=over 4
+
+=item * Blogger
+
+TrackBack ID = C<E<lt>$BlogItemNumber$E<gt>>
+
+Entry Title = C<E<lt>PostSubjectE<gt>E<lt>$BlogItemSubject$E<gt>E<lt>/PostSubjectE<gt>>
+
+Entry Permalink = C<E<lt>$BlogItemArchiveFileName$E<gt>#E<lt>$BlogItemNumber$E<gt>>
+
+=item * GreyMatter
+
+TrackBack ID = C<{{entrynumber}}>
+
+Entry Title = C<{{entrysubject}}>
+
+Entry Permalink = C<{{pagelink}}>
+
+=item * b2
+
+TrackBack ID = C<E<lt>?php the_ID() ?E<gt>>
+
+Entry Title = C<E<lt>?php the_title() ?E<gt>>
+
+Entry Permalink = C<E<lt>?php permalink_link() ?E<gt>>
+
+=item * pMachine
+
+TrackBack ID = C<%%id%%>
+
+Entry Title = C<%%title%%>
+
+Entry Permalink = C<%%comment_permalink%%>
+
+=item * Bloxsom
+
+TrackBack ID = C<$fn>
+
+Entry Title = C<$title>
+
+Entry Permalink = C<$url/$yr/$mo/$da#$fn>
+
+Thanks to Rael for this list of conversions.
+
+=back
+
+=head1 POSSIBLE USES
+
+=over 4
+
+=item 1. Content repository
+
+Like Movable Type's TrackBack implementation, this standalone script can
+be used to power a distributed content repository. The value of the I<tb_id>
+parameter does not necessarily have to be an integer, because all it is used
+for is a filename (B<note> that this is not true of most other TrackBack
+implementations). For example, if you run a site about cats, and want to have
+a way for users to ping your site with entries they write about their own
+cats, you could set up a TrackBack URL like
+F<http://www.foo.com/bar/tb.cgi?tb_id=cats>, then give that URL out on your
+site. End users could then associate this URL with a I<Cats> category in
+their own blog, and ping you whenever they wrote about cats.
+
+=item 2. Building block
+
+You can use this simple implementation as a building block, or a guide, for
+implementing TrackBack in your own system. It illustrates the core
+functionality of the TrackBack framework, onto which you could add bells
+and whistles (IP banning, password-protected TrackBacks, etc).
+
+=item 3. Centralized tool
+
+This TrackBack tool requires that the end user have the ability to run CGI
+scripts on their server. For many users (eg BlogSpot users), this is not
+an option. For such users, a centralized system (based on this tool, perhaps)
+would be ideal.
+
+=back
+
+=cut
diff --git a/trackbacklist.sh b/trackbacklist.sh
new file mode 100644 (file)
index 0000000..249af57
--- /dev/null
@@ -0,0 +1,16 @@
+# NanoBlogger trackback(list) Plugin, requires tb-standalone
+# creates a list of trackbacks to an entry
+#
+# sample code for template - based off default stylesheet with extensions
+#
+# <div class="tb">
+# $NB_TrackBacks
+# </div>
+
+if [ ! -z "$TB_PATH" ]; then
+       nb_msg "generating trackback lists ..."
+
+       PLUGIN_OUTFILE="$BLOG_DIR/$PARTS_DIR/tb.$NB_FILETYPE"
+       /usr/bin/wget -q -O $PLUGIN_OUTFILE "${TB_PATH}?__mode=show&tb_id=$NB_EntryID"
+       NB_TrackBacks=$(< "$PLUGIN_OUTFILE")
+fi
diff --git a/trackbackping.sh b/trackbackping.sh
new file mode 100644 (file)
index 0000000..ceb9930
--- /dev/null
@@ -0,0 +1,22 @@
+# sends trackback pings to URLs found in $NB_EntryBody
+# needs SEND_TRACKBACK_PING="1" und TB_PATH in blog.conf
+
+PINGCACHE=${BLOG_DIR}/tb/pings/${NB_EntryID}
+
+if [ "$SEND_TRACKBACK_PING" = "1" ] && [ ! -z "$TB_PATH" ]; then
+       if [ ! -e "$PINGCACHE" ]; then
+               for URL in $(echo $NB_EntryBody | perl -ne 'print "$1\n" if /href=\C(.*?)\C>/g'); do 
+                       nb_msg "sending trackback pings ..."
+                       PING_URL=$URL
+                       TITLE=$NB_EntryTitle
+                       BLOG_NAME=$BLOG_TITLE
+                       EXCERPT=$(echo $NB_EntryBody | perl -ne 'print "$1 ..." if /^(.{1,80})(\s+|$)/' | html2text -nobs | xargs)
+                       URL=${ARCHIVES_PATH}$NB_EntryPermalink
+                       TBURL="${TB_PATH}?__mode=send_ping&ping_url=${PING_URL}&title=${TITLE}&blog_name=${BLOG_NAME}&excerpt=${EXCERPT}&url=${URL}"
+                       TBURL=$(echo $TBURL | sed -e 's/!/\\!/')
+                       #nb_msg "++ $TBURL"
+                       /usr/bin/wget -q -O /dev/null "$TBURL"
+                       touch $PINGCACHE
+               done
+       fi
+fi