]> git.cworth.org Git - sup/blob - lib/sup.rb
many many changes. this is what happens when i have 5 hours on an airplane
[sup] / lib / sup.rb
1 require 'rubygems'
2 require 'yaml'
3 require 'zlib'
4 require 'thread'
5 require 'fileutils'
6 require 'curses'
7
8 class Object
9   ## this is for debugging purposes because i keep calling #id on the
10   ## wrong object and i want it to throw an exception
11   def id
12     raise "wrong id called on #{self.inspect}"
13   end
14 end
15
16 class Module
17   def yaml_properties *props
18     props = props.map { |p| p.to_s }
19     vars = props.map { |p| "@#{p}" }
20     klass = self
21     path = klass.name.gsub(/::/, "/")
22     
23     klass.instance_eval do
24       define_method(:to_yaml_properties) { vars }
25       define_method(:to_yaml_type) { "!#{Redwood::YAML_DOMAIN},#{Redwood::YAML_DATE}/#{path}" }
26     end
27
28     YAML.add_domain_type("#{Redwood::YAML_DOMAIN},#{Redwood::YAML_DATE}", path) do |type, val|
29       klass.new(*props.map { |p| val[p] })
30     end
31   end
32 end
33
34 module Redwood
35   VERSION = "0.2"
36
37   BASE_DIR   = ENV["SUP_BASE"] || File.join(ENV["HOME"], ".sup")
38   CONFIG_FN  = File.join(BASE_DIR, "config.yaml")
39   SOURCE_FN  = File.join(BASE_DIR, "sources.yaml")
40   LABEL_FN   = File.join(BASE_DIR, "labels.txt")
41   PERSON_FN  = File.join(BASE_DIR, "people.txt")
42   CONTACT_FN = File.join(BASE_DIR, "contacts.txt")
43   DRAFT_DIR  = File.join(BASE_DIR, "drafts")
44   SENT_FN    = File.join(BASE_DIR, "sent.mbox")
45   LOCK_FN    = File.join(BASE_DIR, "lock")
46   SUICIDE_FN = File.join(BASE_DIR, "please-kill-yourself")
47   HOOK_DIR   = File.join(BASE_DIR, "hooks")
48
49   YAML_DOMAIN = "masanjin.net"
50   YAML_DATE = "2006-10-01"
51
52 ## determine encoding and character set
53 ## probably a better way to do this
54   $ctype = ENV["LC_CTYPE"] || ENV["LANG"] || "en-US.utf-8"
55   $encoding =
56     if $ctype =~ /\.(.*)?/
57       $1
58     else
59       "utf-8"
60     end
61
62 ## record exceptions thrown in threads nicely
63   def reporting_thread name
64     if $opts[:no_threads]
65       yield
66     else
67       ::Thread.new do
68         begin
69           yield
70         rescue Exception => e
71           $exceptions ||= []
72           $exceptions << [e, name]
73           raise
74         end
75       end
76     end
77   end
78   module_function :reporting_thread
79
80 ## one-stop shop for yamliciousness
81   def save_yaml_obj object, fn, safe=false
82     if safe
83       safe_fn = "#{File.dirname fn}/safe_#{File.basename fn}"
84       mode = File.stat(fn) if File.exists? fn
85       File.open(safe_fn, "w", mode) { |f| f.puts object.to_yaml }
86       FileUtils.mv safe_fn, fn
87     else
88       File.open(fn, "w") { |f| f.puts object.to_yaml }
89     end
90   end
91
92   def load_yaml_obj fn, compress=false
93     if File.exists? fn
94       if compress
95         Zlib::GzipReader.open(fn) { |f| YAML::load f }
96       else
97         YAML::load_file fn
98       end
99     end
100   end
101
102   def start
103     Redwood::PersonManager.new Redwood::PERSON_FN
104     Redwood::SentManager.new Redwood::SENT_FN
105     Redwood::ContactManager.new Redwood::CONTACT_FN
106     Redwood::LabelManager.new Redwood::LABEL_FN
107     Redwood::AccountManager.new $config[:accounts]
108     Redwood::DraftManager.new Redwood::DRAFT_DIR
109     Redwood::UpdateManager.new
110     Redwood::PollManager.new
111     Redwood::SuicideManager.new Redwood::SUICIDE_FN
112     Redwood::CryptoManager.new
113   end
114
115   def finish
116     Redwood::LabelManager.save if Redwood::LabelManager.instantiated?
117     Redwood::ContactManager.save if Redwood::ContactManager.instantiated?
118     Redwood::PersonManager.save if Redwood::PersonManager.instantiated?
119     Redwood::BufferManager.deinstantiate! if Redwood::BufferManager.instantiated?
120   end
121
122   ## not really a good place for this, so I'll just dump it here.
123   ##
124   ## a source error is either a FatalSourceError or an OutOfSyncSourceError.
125   ## the superclass SourceError is just a generic.
126   def report_broken_sources opts={}
127     return unless BufferManager.instantiated?
128
129     broken_sources = Index.sources.select { |s| s.error.is_a? FatalSourceError }
130     unless broken_sources.empty?
131       BufferManager.spawn_unless_exists("Broken source notification for #{broken_sources.join(',')}", opts) do
132         TextMode.new(<<EOM)
133 Source error notification
134 -------------------------
135
136 Hi there. It looks like one or more message sources is reporting
137 errors. Until this is corrected, messages from these sources cannot
138 be viewed, and new messages will not be detected.
139
140 #{broken_sources.map { |s| "Source: " + s.to_s + "\n Error: " + s.error.message.wrap(70).join("\n        ")}.join("\n\n")}
141 EOM
142 #' stupid ruby-mode
143       end
144     end
145
146     desynced_sources = Index.sources.select { |s| s.error.is_a? OutOfSyncSourceError }
147     unless desynced_sources.empty?
148       BufferManager.spawn_unless_exists("Out-of-sync source notification for #{broken_sources.join(',')}", opts) do
149         TextMode.new(<<EOM)
150 Out-of-sync source notification
151 -------------------------------
152
153 Hi there. It looks like one or more sources has fallen out of sync
154 with my index. This can happen when you modify these sources with
155 other email clients. (Sorry, I don't play well with others.)
156
157 Until this is corrected, messages from these sources cannot be viewed,
158 and new messages will not be detected. Luckily, this is easy to correct!
159
160 #{desynced_sources.map do |s|
161   "Source: " + s.to_s + 
162    "\n Error: " + s.error.message.wrap(70).join("\n        ") + 
163    "\n   Fix: sup-sync --changed #{s.to_s}"
164   end}
165 EOM
166 #' stupid ruby-mode
167       end
168     end
169   end
170
171   module_function :save_yaml_obj, :load_yaml_obj, :start, :finish,
172                   :report_broken_sources
173 end
174
175 ## set up default configuration file
176 if File.exists? Redwood::CONFIG_FN
177   $config = Redwood::load_yaml_obj Redwood::CONFIG_FN
178 else
179   require 'etc'
180   require 'socket'
181   name = Etc.getpwnam(ENV["USER"]).gecos.split(/,/).first
182   email = ENV["USER"] + "@" + 
183     begin
184       Socket.gethostbyname(Socket.gethostname).first
185     rescue SocketError
186       Socket.gethostname
187     end
188
189   $config = {
190     :accounts => {
191       :default => {
192         :name => name,
193         :email => email,
194         :alternates => [],
195         :sendmail => "/usr/sbin/sendmail -oem -ti",
196         :signature => File.join(ENV["HOME"], ".signature")
197       }
198     },
199     :editor => ENV["EDITOR"] || "/usr/bin/vim -f -c 'setlocal spell spelllang=en_us' -c 'set filetype=mail'",
200     :thread_by_subject => false,
201     :edit_signature => false,
202     :ask_for_cc => true,
203     :ask_for_bcc => false,
204     :ask_for_subject => true,
205     :confirm_no_attachments => true,
206     :confirm_top_posting => true,
207   }
208   begin
209     FileUtils.mkdir_p Redwood::BASE_DIR
210     Redwood::save_yaml_obj $config, Redwood::CONFIG_FN
211   rescue StandardError => e
212     $stderr.puts "warning: #{e.message}"
213   end
214 end
215
216 require "sup/util"
217 require "sup/hook"
218
219 ## we have to initialize this guy first, because other classes must
220 ## reference it in order to register hooks, and they do that at parse
221 ## time.
222 Redwood::HookManager.new Redwood::HOOK_DIR
223
224 ## everything we need to get logging working
225 require "sup/buffer"
226 require "sup/keymap"
227 require "sup/mode"
228 require "sup/modes/scroll-mode"
229 require "sup/modes/text-mode"
230 require "sup/modes/log-mode"
231 require "sup/logger"
232 module Redwood
233   def log s; Logger.log s; end
234   module_function :log
235 end
236
237 ## now everything else (which can feel free to call Redwood::log at load time)
238 require "sup/update"
239 require "sup/suicide"
240 require "sup/message-chunks"
241 require "sup/message"
242 require "sup/source"
243 require "sup/mbox"
244 require "sup/maildir"
245 require "sup/imap"
246 require "sup/person"
247 require "sup/account"
248 require "sup/thread"
249 require "sup/index"
250 require "sup/textfield"
251 require "sup/colormap"
252 require "sup/label"
253 require "sup/contact"
254 require "sup/tagger"
255 require "sup/draft"
256 require "sup/poll"
257 require "sup/crypto"
258 require "sup/modes/line-cursor-mode"
259 require "sup/modes/help-mode"
260 require "sup/modes/edit-message-mode"
261 require "sup/modes/compose-mode"
262 require "sup/modes/resume-mode"
263 require "sup/modes/forward-mode"
264 require "sup/modes/reply-mode"
265 require "sup/modes/label-list-mode"
266 require "sup/modes/contact-list-mode"
267 require "sup/modes/thread-view-mode"
268 require "sup/modes/thread-index-mode"
269 require "sup/modes/label-search-results-mode"
270 require "sup/modes/search-results-mode"
271 require "sup/modes/person-search-results-mode"
272 require "sup/modes/inbox-mode"
273 require "sup/modes/buffer-list-mode"
274 require "sup/modes/poll-mode"
275 require "sup/modes/file-browser-mode"
276 require "sup/modes/completion-mode"
277 require "sup/sent"
278
279 $:.each do |base|
280   d = File.join base, "sup/share/modes/"
281   Redwood::Mode.load_all_modes d if File.directory? d
282 end