*** empty log message ***
[freeside.git] / TODO
1 $Id: TODO,v 1.28 1999-04-14 08:58:59 ivan Exp $
2
3 If you are interested in helping with any of these, please join the mailing
4 list (send a blank message to ivan-freeside-subscribe@sisd.com) to avoid 
5 duplication of effort.
6
7 ---
8
9 hmm... maybe svc_acct__shell should check off the legal shells list if
10 applicable?  yeah... cool.
11
12 payinfo field should me much larger than 16
13
14 default (and ordering) state/county/country config file
15
16 [Mon Apr 12 20:31:21 1999] [error] [Mon Apr 12 20:31:21 1999] null: Error closing true: Broken pipe at /usr/local/lib/site_perl/FS/cust_main.pm line 615.
17
18 configuration option to show passwords
19
20 javascript (yuck!) "are you sure?" confirmation on cancelations, etc.
21 (view/cust_pkg and view/svc_*)
22
23 get rid of time2str("%D") which formats dates in a non-y2k-safe looking fashion
24 (all the actual date handling uses UNIX timestamps and is fine)
25
26 uncomment expire in view/cust_pkg.cgi and find the expire cron from fsold
27
28 (Test this)
29 one-time/per-customer/? changes in rates and descriptions ('remembered
30 invoices'): implement by creating a new package on the fly... but it isn't 
31 associated with any agent types so it won't show up for other customers to buy.
32 (but also... make sure they go away when the customer does! - need this? :
33  one-off package edits! : need a cust_pkgs or cust_part_pkgs or something table,
34  with custnum and partpkg (like type_pkgs)
35 (what happens if you hit "custom pricing" but the pricing is already custom?)
36
37 Lay out any remaining ugly forms better.
38
39 remove "records identical" warning?  gets in the way of more important stuff.
40 or fix logic which tries to update identical records??
41 1.2 should be quiet enough that the error log is useful, hopefully.
42
43 Postgres has a maximum column length of 31 characters (but see NAMEDATALEN in
44 postgres_ext.h).  part_svc has columns like: svc_acct__radius_Attribute_flag
45 (22 characters!)  It seems that stuff over the limit is silently ignored,
46 so we get 4 characters back.  So, Radius_Attributes are max 13 characters with
47 stock Postgres.  see rfc2138 for what's affected
48 What's a good fix?  (besides recompiling postgres with NAMEDATALEN 64)
49 (mysql has a 64 character max column length.  others?)
50
51 [Mon Mar 29 06:57:56 1999] -e: Use of uninitialized value at /usr/lib/perl5/Date/Format.pm line 333.
52 (when sending mail in cust_main.pm::bill or svc_domain.pm)
53
54 look at DBIx::Recordset!  (and Tie::DBI, and...)
55
56 undefined conf/lpr gives this uninfomative error:
57 [Fri Feb 26 16:42:36 1999] bill.cgi: Can't do bidirectional pipe at
58 /usr/lib/per
59 l5/site_perl/FS/cust_main.pm line 629.
60 [Fri Feb 26 16:42:38 1999] bill.cgi: Error closing : Broken pipe at
61 /usr/lib/per
62 l5/site_perl/FS/cust_main.pm line 631.
63 So give a meaningful error!
64
65 password and slipip stuff in svc_acct.pm store need to be split into two fields or something, so the silliness in svc_acct.pm and svc_acct.export with looking at the data to decide what to do with it can be fixed
66
67 i10n: Apache::Language
68
69 Apache::Session?  Other useful Apache::* ?
70
71 email invoices are only sent for the BILL payby.  If setup, should statements
72 (since they're not invoices) be sent for COMP and CARD as well?
73
74 $cgi->keywords is causing the (hard to trace) error:
75         Use of uninitialized value at (eval 5) line 5
76
77 edit/cust_main.cgi gives an uninformative error message:
78 > Can't call method "agentnum" without a package or object reference at   
79 > /usr/local/apache-ssl/htdocs/freeside/edit/cust_main.cgi line 116.
80 if there are no agents.
81
82 (is this missing on any web screens?  (easy with $cust_svc->label)
83 Add the ability for services to filter information up to the package level
84 for web screens, so you can select a particlar package based
85 on username or domain name, etc.
86
87 Allow a cancelled/suspended/active status from packages to bubble up to
88 the customer lists.  Put active, then suspended, then cancelled accounts.
89 Similar ordering on the package listing inside a single customer.
90
91 false laziness: edit/cust_main.cgi got some parts copied from edit/svc_acct.cgi
92 the web interface in general needs to be redone in a more abstract way.
93
94 false laziness: some of search/svc_acct_sm.cgi was copied to search/svc_domain.cgi.  but web interface in general needs to be rewritten in a mucho cleaner way.
95
96 Portability: in FS::Record, $sth->execute does not return a number of rows for all DBD's.  see man DBI
97
98 subroutine the where clause (eventually all SQL) as OO perhaps (has anyone done this?)
99
100 add a select method to FS::Record?
101
102 one-time/per-customer/? changes in rates and descriptions ('remembered
103 invoices'): implement by creating a new package on the fly... but it isn't 
104 associated with any agent types so it won't show up for other customers to buy.
105 (but also... make sure they go away when the customer does! - need this? :
106  one-off package edits! : need a cust_pkgs or cust_part_pkgs or something table,
107  with custnum and partpkg (like type_pkgs)
108 (what happens if you hit "custom pricing" but the pricing is already custom?)
109
110 You can't delete the stuff under administration yet.  Add this,
111 _including_ making sure the thing you are deleting is not in use!
112
113 add links on view/cust_main.cgi to setup services, like view/cust_pkg.cgi
114
115 FS::cust_pkg _require_'s FS::$svc, but this won't work with %FS::UID::callback
116 loading of configuration.  (pry need same idea, but will run immediately if
117 context allows).  Looks like error is masked by 'use FS::cust_svc' which in
118 turn 'use's FS::{svc_acct, svc_acct_sm, svc_domain}' which is now explicit
119 w/comments in source
120
121 Allow a cancelled/suspended/active status from packages to bubble up to
122 the customer lists.  Put active, then suspended, then cancelled accounts.
123 Similar ordering on the package listing inside a single customer.
124
125 svc_domain.pm mail sending uses Date::Format which doesn't seem to pick up 
126 correct timezone.
127
128 view/svc_domain.cgi needs to know the domain might be unaudited (cosmetic)
129
130 remove whois_hack set to 1 for svc_domain.pm?  add all known registries and
131 whois accordingly.
132 .us domains and others!
133 site_perl/svc_domain.cgi (hmm... or maybe should have a button?  or maybe svc_domain.pm should handle this) should set $whois_hack for non-internic domains, so you can add them...
134
135 turn on the depriciation warnings for [e]idiot in FS::CGI. Stop using [e]idiot
136 the last places it is (htdocs/search/ htdocs/misc/ htdocs/misc/process)
137
138 (test cust_main.pm with cybercash v2 and v3, especially with the callback
139  stuff AND with mod_perl w/cybercash v2 kludge in package main)
140 (callback stuff should be eliminated by now)
141
142 bah, table/itable/*table in FS::CGI is silly.
143
144 doc Apache::AuthDBI as well
145 ..
146 Provide sample httpd.conf files.
147
148 hey look: Tie::DBI!  Check that out.  Override its commit with something that
149 does perl-side caching for ? a performance improvement and as an emulation
150 layer to plug in f.ex mysql's atomic transactions
151 ..
152 Record.pm uses does some non-portable DBI things.  MySQL and Pg seem fine.
153 Fix it anyway unless we migrate to Tie::DBI.
154
155 faq
156
157 cust_bill.pm uses '==' comparison on dates because they're currently ints
158
159 config file for allowed card types
160
161 write instructions for adding new services w/svc_Common.pm.  Get rid of all
162 places where svc_* tables are hardcoded (rename svc_acct_pop to part_pop so
163 we can do that)
164
165 test and document libapache-dbi-logger (woo!)
166
167 radius logfile parsing and perl expression check.
168
169 Fix in cust_bill BUGS: 
170 There is an off-by-one error in print_text which causes a visual error (Page 1
171 of 2 printed on some single-page invoices).
172
173 fields should be a method against a FS::Record or derived object, as well as
174 being something you can call as FS::Record::fields('tablename').  Might
175 even be able to handle both in the same routine (that would be neato).
176
177 Immediate removal of incorrectly entered check payments (can't take too
178 long to do this, or accounting is fubared).
179
180 Add code to move from one service to another (POP to SLIP/PPP, etc.).
181 This _should_ be possible by working off the rules in part_svc rather than
182 hardcoding anything in.  The rules in part_svc may need some elaboration,
183 perhaps.
184
185 Use ut_ FS::Record methods in all derived classes (possibly some from dbdef?... eventually all from dbdef??? - but then `dbdef-create' would be impossible as there would be metadata we couldn't ask the backend for.  hmm.)
186
187 (bring back from fsold, ) Generalize config-sending stuff and make more configurable.
188 Expand the HylaFAX interface (also possibly generalize for other fax
189 softwar ie .comfaxe); allow things like arbitrary faxes of sales
190 literature, specific troubleshooting documents and so on.  Maybe even
191 allow users to do this (though that might not belong in Freeside).
192 misc/sendconfig.cgi
193 misc/process/sendconfig.cgi
194 Configure fax recipients via a separate box rather than using the finger
195 name or first+last from cust_main.
196
197 move all phone number logic out of Freeside - let HylaFAX or whatever
198 handle it.
199
200 soundex searches for customer name and company?  where are free soundex tools? (standard Text::Soundex duh) - I could have sworn I saw Text::Soundex on CPAN?!
201
202 should be able to link on some field in email alias (right now you can link
203 on username or domain with a fallback to svcnum)
204
205 generalize and make configurable new invoice printing scheme in FS::cust_main::collect (past due)
206
207 deleting an svc_domain should delete all associated svc_acct_sm records.
208 same with a svc_acct.
209
210 periodic password encrypter
211
212 Automated, configurable notification, suspension and cancellation of
213 defunct accounts.
214 ...
215 expire cron job
216 ...
217 Allow for a future setup date on accounts.
218
219 sub AUTOLOAD in FS::Record should warn? die? if used with a non-existant column
220 name?
221
222 edit (not just import, export and allow default/fixed) arbitrary radius stuff
223 in svc_acct
224 edit/svc_acct.cgi and edit/process/svc_acct.cgi should deal with arbitrary radius stuff
225
226 radius import should take DEFAULT entry and put it in /var/spool/freeside/conf/radius-default ; svc_acct.export should use it (and doc)
227
228 in UI, s/State/State\/Provence/go and s/County/County\/Locality/go
229
230 what else (besides l10n) for i18n? (money!)
231
232 audit htdocs/* for things that should be libraried and things that should be
233 new methods on the objects (need to do this before implementing a new UI)
234 all the big things are done
235
236 some places we die() where we should &FS::CGI::idiot (and perhaps vice-versa).
237 Decide based on whether or not the "error" should show up in logs.
238
239 all .cgi's should use standard header/footer and idiot() subroutines.  maybe HTML:: perl modules
240 for HTML creation.  Maybe Embperl or something along those lines.  ?
241
242 When running bin/bill, Fix this (Annoying but harmless):
243 Use of uninitialized value at /usr/local/lib/site_perl/FS/cust_pkg.pm line 99, <ADDRESS> chunk 4.
244 Use of uninitialized value at /usr/local/lib/site_perl/FS/cust_pkg.pm line 102, <ADDRESS> chunk 4.
245 Use of uninitialized value at /usr/local/lib/site_perl/FS/cust_pkg.pm line 105, <ADDRESS> chunk 4.
246
247 all cgi (but internal to the isp) places where package names are listed should also have
248 comment (like agent_type)
249
250 clean up $recref and other silliness and use -> calls where possible, or
251 one other alternative.  clean up everything else.
252 should FS::Record use Tie::Hash?  That would be very clean, but where do we
253 store the other information?  Maybe you could ask any FS::Record object for a
254 tied hash?
255
256 change all file access from regular open(FILE,) stuff to OO, because of 
257 problems scoping and passing filehandles like that.
258
259 the web interface should create a new object and use it instead of a blank
260 form for new records.  the create method of svc_ objects should set defaults
261 (from part_svc).
262
263 sub check in man FS::table_name should be rewriteen.  Get rid of $recref
264 stuff.  Make sure all fields that refer to other database are checked.
265
266 Integration with signup disks (are there any free ones?  Netscape?).
267
268 One-button cancel (+refund) for lusers who can't get online.
269
270 Keep information on virtual web servers (hostname, IP, host machine,
271 directory, etc.) and export this information for importation into the ISPs
272 web farm. 
273
274 Remove requirement that the first mail alias be the catchall?  Still make
275 sure only one catchall per domain is defined in any case, of course. 
276
277 Ability to move cust_pkg records from one customer to another? (proably
278 will need to cancel the old and create a new like when we move services
279 between packages). 
280
281 Auto-increment expired cards one year, and try again?
282
283 More non-US stuff - zip codes, country codes, foreign currencies, etc.
284
285 cust_refund.{cgi.pm} need to do cards xaxtions.  (now we only have cust_credit)
286
287 Nicer set of integrated reporting possibilities, like weekly sales totals
288 by customer, package, agent, referral, etc., aging reports sorted by lots
289 of different things, and so on.
290
291 Client/server setup for users to modify their own passwords, shells, etc,
292 via passwd or secure web interface (prelminary passwd/chfn/chsh
293 replacement done).  Complicated by the fact that we don't want to allow
294 incoming connections to the machine running Freeside, so we probably need
295 to have a daemon on each external shell or web machine that is contacted
296 by the Freeside machine.  Be very very careful for both traditional
297 security issues and DoS problems. 
298
299 An extension of the above to allow users to modify selected parts of their
300 own information, order and cancel services.  A web interface for new
301 customers.
302
303 Expand domain name stuff to house all domain information.  Export
304 named.boot/named.conf (primary and secondary) and named.{domain} files.
305 Add more registries (not just InterNIC's com org net edu)
306
307 Nice postscript paper invoices, rather than current ASCII invoices.
308
309 think about race-condititions in FS::Record and derived ->check ->insert
310 and so on, uid and username checks in svc_acct, etc.
311
312 Move to rsync over ssh file exportation rather than scp.
313
314 check 'n fix the proactive password checker. (cracklib?)
315
316 refunds of "BILL" payments: generate pseudo-check.
317
318 write batch senders and batch parsers for the different credit card processors
319 people use/
320 More CC processors/methods.
321
322 you should be able to fiddle the setup date in cust_pkg. (at least initially)
323
324 delete options in administration section
325
326 write a generic batch senders and batch parsers.
327
328 need a way to override svc_acct export on a per-machine basis; just use config files based on machine name i suppose; document that. (no, import desync_hosts
329 type stuff from cerkit)
330 ...
331 add a table with column of export services (passwd, shadow, master.passwd, .qmail file update, dns update, etc.) and rows machine groups and whether or not to export that (and any necessary parameters).  wasn't matt (vunderkid, not matt@michweb) working on this?  find him?  each machine goes in a group of its own as well as a group based on function.  add a table with only svcpart and machine group.  now, when you import from each machine, it can get its own accounts with one svcpart and universal accounts with another svcpart.  (though that does make the username duplicate checking more interesting)
332
333 you should be able to get column types as a method against an FS::Record object
334 as well as dbdef->table($table)->column($column)->type
335
336 move to perl module for fuzzy and soundex searching.
337
338 package view needs to list extraneous services; we need to prevent the
339 creation of them so this never happens (and mark it as such in the source)
340 (the creation problem should be fixed - though they will still happen if people
341 fsck around in the data manually, so list them anyway)
342
343 add attribute dictionary to fs-setup as a menu, plus analyze users file to
344 decide automatically
345
346 Check for and report on duplicate billing accounts (cust_main, though many
347 will have a need for these so probably don't disable them outright.)
348
349 create a ->warn as well as a ->check method for all FS::table classes?
350 (see above)
351
352 something to automate making a release and updating the web demo
353
354 export a debian-style (also redhat and?) /etc/group file aswell!
355
356 svc_acct_sm.import qmail import should pull in recipientmap people too.
357
358 .pm's like svc_acct.pm which need to do time-consuming things like ssh remotely
359 should fork and do them in a child.
360
361 i18n/l10n: take ALL messages and catalog them in english.txt or in database or something, so we can eventually go int'l.  int'l currency support would be a help aswell.
362
363 get some of { city, county, state, zip } from the missing bits if
364 possible (where can i get the data to do this?  usps.gov?)
365
366 additional interfaces (perltk?  java?)
367
368 Put the GPL notice in all files.
369
370 integrate w/IDEA's signup server
371
372 $cust_bill->owed database field to be eliminated, replaced by a method call
373 that calculates on the fly.  make sure to grep for ->(get|set)field('owed') 
374 same for cust_credit->credited
375
376 Export quota information.
377
378 move all configuration to a central place.  maybe in blob's in the
379 database.  maybe even things like the code to execute when a username is
380 changed can be in there, so less of the distributed scripts change between
381 different sites.
382
383 Implement setup and recurring fees as Safe perl expressions rather than
384 numbers, to allow for variable-rate services.  Backwards compatibility is
385 obtained because { 43 } in perl is still 43.  :)  Define API to pass
386 starting and ending dates and any other necessary data to expression
387 (fees are currently evaluated as Safe expressions but more work needs to
388 be done to define an opmask for various needs, write examples
389 (usage-based billing, etc.) and so on).
390 ...
391 Add the ability to modify the next billing date in cust_pkg, and take
392 appropriate action.  This will allow the implementation of pro-rate/1st of
393 the month billing as well as the ability to manually fiddle with
394 anniversary dates in cust_pkg, so you can sync a customer's anniversary
395 date even if you're using anniversary billing (manually or automatically).
396 (now with above, we need to have a way to automatically pro-rate /^(\d+)$/
397 charges - anything more complicated should figure it out itself given
398 starting and ending dates [document that!])
399 ...
400 Daily Radius log parsing into database; other logfile formats?
401 ...
402 Callbacks to enforce hourly limits on accounts (suspend until the end of
403 the billing period?), for those who limit customers rather than tack on
404 extra charges.
405
406 Flag packages (part_pkg) as taxable or non/taxable as some ISPs (for
407 example) need to charge tax on equipment but not service (separate flags
408 for setup and recurring fee... or perhaps a setup_tax, setup_notax,
409 recur_tax and recur_notax fees, and possibly something more flexible if
410 there is need).
411
412 Allow for a variable number of invoices for customers who need multiple
413 copies.
414
415 Add a mail alias service with table svc_acct (not domain mail aliasing
416 which is domain with svc_acct_sm)
417
418 (bring back from fsold) Change customer comment field from its current kludge to something more
419 workable.
420
421 Better work orders with more information.  Should eventually open a ticket
422 when we have such a thing.
423 edit/svc_wo.cgi
424 edit/process/svc_wo.cgi
425 Call tracking and trouble tickets.
426
427 More accoutability for complimentary accounts: approval, expiration, term
428 (no more than x months in advance) and notification.
429 Flag particular users (or all users, for that matter) as having their
430 passwords hidden and/or locked from users of Freeside (maybe need Freeside
431 security levels first?). 
432 ...
433 Better Freeside-level configurable access, for those ISP's who have
434 employees they can't trust.  Right now you're "stuck" with setting up
435 .htaccess stuff yourself.  This should really just be integrated. 
436
437 configuration/setup should get web interface
438 ...
439 /usr/local/etc/freeside should be configurable
440 ...
441 (probably as part of some automated installation process?)
442
443 This requires some serious magic in FS::Record:
444 ok, if date_type in fs-setup is to be something besides int,
445 now we need to create wrappers
446 for them so they behave identically across RDBMS's, ie date pops out as as
447 UNIX timestamp (or an object of some sort? maybe even a blessed $obj which
448 is a string not a hashref for backwards compatibility?) and so on. (remember
449 to treat '0' as Not a Date instead of 1/1/70.
450
451 Add Freeside-level transactions for RDBMS's which don't support
452 transcations?  (Currently we assume a minimal RDBMS which has no rollback,
453 transactions or atomic updates).  Or just require a RDBMS that supports
454 rollback and/or atomic updates and get rid of the work-arounds?  The /rdb
455 interface had this kludge on top of it but is a technical dead-end in most
456 other ways, unless it can gain an SQL parser and DBD interface.
457 ...
458 if i'm really bored, find the /rdb interface in fsold and port it to NoSQL,
459 and while I'm add it add interfaces for AnyDBM_File tied hash.. hmm.  Shouldn't
460 an FS::Record have something to do with a tied hash?  But we don't want
461 performance to go gaga... maybe something with commit to help out here?
462 ...
463 Ok: FS::Record gives you a tied hash, and you get methods for commit, etc.
464
465 Better automated comparison of our CC records with processors (CyberCash,
466 at least, has not always had 100% accuracy, though recent versions are
467 much better) 
468
469 Expect or other pty based login check, where we actually connect to a
470 terminal server or shell machine and test logging in as the user (if we
471 are keeping a cleartext password for that user)  (This is something tech
472 support often needs for new customers)
473
474 Use cust_main table for pre-sales tracking as well?
475
476 Automatic commision report and check generation via freq and prog (to
477 become a Safe perl expression) fields in agent table, and possibly others.
478
479 Database and add a mailed-out date and method for disk/CD mailing, so a
480 customer can call and you can say, "sent on xx/xx/xx via {US Mail, Fedex,
481 UPS, etc}" 
482
483 Inventory tracking for physical items such as routers (for sale or
484 lease... probably doesn't make a difference in the ordering... but if you
485 cancel a router lease the inventory should come back.  hmm.)
486
487 -- Matt's wishlist ---
488
489 From matt@michweb.net Fri Feb 20 16:39:53 1998
490 Date: Thu, 19 Feb 1998 23:20:11 -0500
491 From: Matt Simerson <matt@michweb.net>
492 Reply-To: quadran-developer@netgoth.com
493 To: quadran-developer@netgoth.com
494 Subject: Re: Welcome to quadran-developer
495
496 >Whats it based on and what is it supposed to do?  I'm interested, but
497 >unfortunatly, I don't have that much time to help on the project (I'm busily
498 >working on one of my own based around MySQL and Qt right now -- don't know
499 >if it will be GPL'ed or not yet -- we'll probably just use it in house since
500 >it is designed around our system)...
501
502 That's what I set out to find, but didn't find anything on the web site.
503 I'm looking for something that will do the following:
504
505 Single point of entry for users on a secure system:
506         Creates account on user (public) systems
507                 update /etc/passwd/master.passwd file
508                 update radius database (if necessary)
509         Set up up disk quotas (although I hacked adduser to do this)
510         Option for adding user to a mailing list(s)
511         Export of new user info to customizable report (for automated entry
512 into
513                 accounting software, etc...)
514
515 Automated billing:
516         Export credit card info for batch processing and have hooks built
517                 in for other forms of electronic processing.
518         Batch-Payment (apply payments from formatted text file).
519         Customizable reports for manual entry/importing into Accounting
520 software
521         Email or laser print invoices
522         Sanity checks credit card numbers before processing (code available)
523
524 Simple method for disabling an account.
525         Arbitrary Expiration Dates (on a given day, in x days)
526         Remove from radius.
527         Changing password to '*'
528         Virtual customers disabling dns, http server, log processing, etc..
529
530 Billing for different account types:
531         Dialup monthly flat rate. Prorates for partial months.
532         Dialup monthly flat rate for x hours + hourly usage.
533         Dialup email only
534         Email only accounts
535         Virtual Web accounts - w/multiple mailboxes
536         Leased line accounts
537         Disk space used over quota.
538         Tech support minimum + hourly charges
539         Other for misc stuff (modem, RAM, etc...)
540
541 Per user definable RADIUS attributes (ties in with above)
542         Fixed IP
543         Simultaneous Use
544         IP filters (for dialup email only)
545
546 Keep logs of modem usage generated daily from radius accounting logs stored
547 on multiple radius servers.
548
549 Keep logs of disk usage generated from quota.
550
551 Method of adding virtual domains to your system:
552         Automatically grabs an IP address from a preassigned pool.
553         Creates a domain.com database file from database fields
554         Updates /etc/named.conf or /etc/named.boot and reloads named.
555         Add's virtual.com to /etc/sendmail.cw or qmail control files.
556         Edits your web servers httpd.conf file and restarts http server.
557         An optional section for adding vif's can be added if the users OS
558                 supports adding them on the fly. Otherwise it's up to the end
559                 user. Make a hook that can run a custom script that the user
560                 tweaks for his system.
561         Update or create the config file your web stats analyzer needs. I've
562                 done this for analog (free) and http-analyze. Probably
563                 should only officially support analog and let users hack
564                 it to their hearts desire.
565 I've already written scripts that do most of the virtual web stuff on my
566 system...in bash. Shouldn't be hard for a perlmeister to convert. In fact,
567 as long as all the info was stored in the database (username, domain name,
568 and ip pool) this could easily just be run as an external script that the
569 user tweaks to match his system.
570
571 We use a great accounting software (M.Y.O.B) that does all the AP, AR,
572 Payroll, Tax stuff, and most everything else we could need. It's already
573 set up for the type of checks we have, etc, etc... I just need something to
574 do the billing part. I can import/export sales and payments directly once
575 the billing part is done. You can't write accounting software as good as
576 M.Y.O.B. for $120.
577
578