#!/usr/bin/env python
# -*- coding: utf-8 -*-
+##############################################################################
+#
+# OpenERP, Open Source Management Solution
+# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
+#
+# This program is free software: you can redistribute it and/or modify
+# it under the terms of the GNU Affero General Public License as
+# published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
+#
+# You should have received a copy of the GNU Affero General Public License
+# along with this program. If not, see <http://www.gnu.org/licenses/>.
+#
+##############################################################################
+
# setup from TinERP
# taken from straw http://www.nongnu.org/straw/index.html
# taken from gnomolicious http://www.nongnu.org/gnomolicious/
# adapted by Nicolas Évrard <nicoe@altern.org>
#
-# $Id$
-import imp
import sys
import os
+from os.path import join, isfile, basename
import glob
-from distutils.core import setup, Command
-from distutils.command.install_scripts import install_scripts
-from distutils.file_util import copy_file
+from pprint import pprint as pp
+
+from setuptools import setup, find_packages
+from setuptools.command.install import install
+from distutils.sysconfig import get_python_lib
-from stat import ST_MODE
+has_py2exe = False
+py2exe_keywords = {}
+if os.name == 'nt':
+ import py2exe
+ has_py2exe = True
+ py2exe_keywords['console'] = [
+ { "script": join("bin", "openerp-server.py"),
+ "icon_resources": [(1, join("pixmaps","openerp-icon.ico"))]
+ }]
+ py2exe_keywords['options'] = {
+ "py2exe": {
+ "compressed": 1,
+ "optimize": 2,
+ "dist_dir": 'dist',
+ "packages": [
+ "lxml", "lxml.builder", "lxml._elementpath", "lxml.etree",
+ "lxml.objectify", "decimal", "xml", "xml", "xml.dom", "xml.xpath",
+ "encodings", "dateutil", "wizard", "pychart", "PIL", "pyparsing",
+ "pydot", "asyncore","asynchat", "reportlab", "vobject",
+ "HTMLParser", "select", "mako", "poplib",
+ "imaplib", "smtplib", "email", "yaml", "DAV",
+ "uuid", "commands",
+ ],
+ "excludes" : ["Tkconstants","Tkinter","tcl"],
+ }
+ }
-opj = os.path.join
+sys.path.append(join(os.path.abspath(os.path.dirname(__file__)), "bin"))
-name = 'tinyerp-server'
-version = '4.0.0'
+execfile(join('bin', 'release.py'))
+
+if 'bdist_rpm' in sys.argv:
+ version = version.split('-')[0]
# get python short version
py_short_version = '%s.%s' % sys.version_info[:2]
-included_addons = [
- 'account', 'account_followup', 'account_tax_include', 'airport', 'audittrail',
- 'base','base_partner_relation', 'base_setup', 'crm', 'custom', 'delivery',
- 'edi', 'esale_ez', 'esale_joomla', 'esale_osc',
- 'hr', 'hr_evaluation', 'hr_expense', 'hr_skill', 'hr_timesheet',
- 'hr_timesheet_ical', 'hr_timesheet_invoice', 'hr_timesheet_project',
- 'letter', 'marketing', 'mrp', 'network', 'partner_ldap',
- 'product','product_electronic', 'product_expiry', 'product_extended',
- 'productivity_analysis', 'product_variant', 'profile_accounting',
- 'profile_manufacturing', 'profile_service', 'project', 'purchase',
- 'purchase_tax_include', 'report_analytic_line', 'report_crm',
- 'report_project', 'report_purchase', 'report_sale', 'sale', 'sale_crm',
- 'sale_journal', 'sale_rebate', 'sale_tax_include', 'sandwich', 'scrum',
- 'stock', 'subscription', 'travel',
- 'l10n_be', 'l10n_ca-qc', 'l10n_ch', 'l10n_ch_pcpbl_association',
- 'l10n_ch_pcpbl_independant', 'l10n_ch_pcpbl_menage',
- 'l10n_ch_pcpbl_plangen', 'l10n_ch_pcpbl_plangensimpl', 'l10n_ch_vat_brut',
- 'l10n_ch_vat_forfait', 'l10n_ch_vat_net', 'l10n_fr', 'l10n_se',
- 'l10n_simple', 'l10n_chart_at', 'l10n_chart_au', 'l10n_chart_be_frnl',
- 'l10n_chart_br', 'l10n_chart_ca_en', 'l10n_chart_ca_fr',
- 'l10n_chart_ch_german', 'l10n_chart_cn', 'l10n_chart_cn_traditional',
- 'l10n_chart_co', 'l10n_chart_cz', 'l10n_chart_da', 'l10n_chart_de_skr03',
- 'l10n_chart_hu', 'l10n_chart_id', 'l10n_chart_it', 'l10n_chart_it_cc2424',
- 'l10n_chart_la', 'l10n_chart_nl', 'l10n_chart_nl_standard', 'l10n_chart_no',
- 'l10n_chart_pa', 'l10n_chart_pl', 'l10n_chart_sp', 'l10n_chart_sw',
- 'l10n_chart_sw_church', 'l10n_chart_sw_food', 'l10n_chart_uk',
- 'l10n_chart_us_general', 'l10n_chart_us_manufacturing',
- 'l10n_chart_us_service', 'l10n_chart_us_ucoa', 'l10n_chart_us_ucoa_ez',
- 'l10n_chart_ve',]
-
-required_modules = [('psycopg', 'PostgreSQL module'),
- ('xml', 'XML Tools for python'),
- ('libxml2', 'libxml2 python bindings'),
- ('libxslt', 'libxslt python bindings')]
-
-def check_modules():
- ok = True
- for modname, desc in required_modules:
- try:
- exec('import %s' % modname)
- except ImportError:
- ok = False
- print 'Error: python module %s (%s) is required' % (modname, desc)
-
- if not ok:
- sys.exit(1)
+# backports os.walk with followlinks from python 2.6
+def walk_followlinks(top, topdown=True, onerror=None, followlinks=False):
+ from os.path import join, isdir, islink
+ from os import listdir, error
+
+ try:
+ names = listdir(top)
+ except error, err:
+ if onerror is not None:
+ onerror(err)
+ return
+
+ dirs, nondirs = [], []
+ for name in names:
+ if isdir(join(top, name)):
+ dirs.append(name)
+ else:
+ nondirs.append(name)
+
+ if topdown:
+ yield top, dirs, nondirs
+ for name in dirs:
+ path = join(top, name)
+ if followlinks or not islink(path):
+ for x in walk_followlinks(path, topdown, onerror, followlinks):
+ yield x
+ if not topdown:
+ yield top, dirs, nondirs
+
+if sys.version_info < (2, 6):
+ os.walk = walk_followlinks
def find_addons():
- for addon in included_addons:
- path = opj('bin', 'addons', addon)
- for dirpath, dirnames, filenames in os.walk(path):
- if '__init__.py' in filenames:
- modname = dirpath.replace(os.path.sep, '.')
- yield modname.replace('bin', 'tinyerp-server', 1)
+ for root, _, names in os.walk(join('bin', 'addons'), followlinks=True):
+ if '__openerp__.py' in names or '__terp__.py' in names:
+ yield basename(root), root
+ #look for extra modules
+ try:
+ empath = os.getenv('EXTRA_MODULES_PATH', '../addons/')
+ for mname in open(join(empath, 'server_modules.list')):
+ mname = mname.strip()
+ if not mname:
+ continue
+
+ terp = join(empath, mname, '__openerp__.py')
+ if not os.path.exists(terp):
+ terp = join(empath, mname, '__terp__.py')
+
+ if os.path.exists(terp):
+ yield mname, join(empath, mname)
+ else:
+ print "Module %s specified, but no valid path." % mname
+ except Exception:
+ pass
def data_files():
'''Build list of data files to be installed'''
- files = [(opj('share', 'man', 'man1'),
- ['man/tinyerp-server.1']),
- (opj('share', 'man', 'man5'),
- ['man/terp_serverrc.5']),
- (opj('share','doc', 'tinyerp-server-%s' % version),
- [f for f in glob.glob('doc/*') if os.path.isfile(f)]),
- (opj('lib','python%s' % py_short_version, 'site-packagess', 'tinyerp-server', 'i18n'),
- glob.glob('bin/i18n/*')),
- (opj('lib', 'python%s' % py_short_version, 'site-packages', 'tinyerp-server', 'addons', 'custom'),
- glob.glob('bin/addons/custom/*xml') +
- glob.glob('bin/addons/custom/*rml') +
- glob.glob('bin/addons/custom/*xsl'))]
- for addon in find_addons():
- add_path = addon.replace('.', os.path.sep).replace('tinyerp-server', 'bin',
- 1)
- pathfiles = [(opj('lib', 'python%s' % py_short_version, 'site-packages',
- add_path.replace('bin', 'tinyerp-server', 1)),
- glob.glob(opj(add_path, '*xml')) +
- glob.glob(opj(add_path, '*csv')) +
- glob.glob(opj(add_path, '*sql'))),
- (opj('lib', 'python%s' % py_short_version, 'site-packages',
- add_path.replace('bin', 'tinyerp-server', 1), 'data'),
- glob.glob(opj(add_path, 'data', '*xml'))),
- (opj('lib', 'python%s' % py_short_version, 'site-packages',
- add_path.replace('bin', 'tinyerp-server', 1), 'report'),
- glob.glob(opj(add_path, 'report', '*xml')) +
- glob.glob(opj(add_path, 'report', '*rml')) +
- glob.glob(opj(add_path, 'report', '*xsl')))]
- files.extend(pathfiles)
+ files = []
+ if os.name == 'nt':
+ os.chdir('bin')
+ for (dp, dn, names) in os.walk('addons'):
+ files.append((dp, map(lambda x: join('bin', dp, x), names)))
+ os.chdir('..')
+ #for root, _, names in os.walk(join('bin','addons')):
+ # files.append((root, [join(root, name) for name in names]))
+ for root, _, names in os.walk('doc'):
+ files.append((root, [join(root, name) for name in names]))
+ #for root, _, names in os.walk('pixmaps'):
+ # files.append((root, [join(root, name) for name in names]))
+ files.append(('.', [join('bin', 'import_xml.rng'),]))
+ else:
+ man_directory = join('share', 'man')
+ files.append((join(man_directory, 'man1'), ['man/openerp-server.1']))
+ files.append((join(man_directory, 'man5'), ['man/openerp_serverrc.5']))
+
+ doc_directory = join('share', 'doc', 'openerp-server-%s' % version)
+ files.append((doc_directory, filter(isfile, glob.glob('doc/*'))))
+ files.append((join(doc_directory, 'migrate', '3.3.0-3.4.0'),
+ filter(isfile, glob.glob('doc/migrate/3.3.0-3.4.0/*'))))
+ files.append((join(doc_directory, 'migrate', '3.4.0-4.0.0'),
+ filter(isfile, glob.glob('doc/migrate/3.4.0-4.0.0/*'))))
+
+ openerp_site_packages = join(get_python_lib(prefix=''), 'openerp-server')
+
+ files.append((openerp_site_packages, [join('bin', 'import_xml.rng'),]))
+
+ if sys.version_info[0:2] == (2,5):
+ files.append((openerp_site_packages, [ join('python25-compat','BaseHTTPServer.py'),
+ join('python25-compat','SimpleXMLRPCServer.py'),
+ join('python25-compat','SocketServer.py')]))
+
+ for addonname, add_path in find_addons():
+ addon_path = join(get_python_lib(prefix=''), 'openerp-server','addons', addonname)
+ for root, dirs, innerfiles in os.walk(add_path):
+ innerfiles = filter(lambda fil: os.path.splitext(fil)[1] not in ('.pyc', '.pyd', '.pyo'), innerfiles)
+ if innerfiles:
+ res = os.path.normpath(join(addon_path, root.replace(join(add_path), '.')))
+ files.extend(((res, map(lambda fil: join(root, fil),
+ innerfiles)),))
+
return files
-long_desc = '''\
-Tiny ERP is a complete ERP and CRM. The main features are accounting (analytic
-and financial), stock management, sales and purchases management, tasks
-automation, marketing campaigns, help desk, POS, etc. Technical features include
-a distributed server, flexible workflows, an object database, a dynamic GUI,
-customizable reports, and SOAP and XML-RPC interfaces.
-'''
-
-classifiers = """\
-Development Status :: 5 - Production/Stable
-License :: OSI Approved :: GNU General Public License (GPL)
-Programming Language :: Python
-"""
-
-check_modules()
-
-# create startup script
-start_script = \
-"#!/bin/sh\n\
-cd %s/lib/python%s/site-packages/tinyerp-server\n\
-exec %s ./tinyerp-server.py $@" % (sys.prefix, py_short_version, sys.executable)
-# write script
-f = open('tinyerp-server', 'w')
-f.write(start_script)
+f = file('openerp-server','w')
+f.write("""#!/bin/sh
+echo "Error: the content of this file should have been replaced during "
+echo "installation\n"
+exit 1
+""")
f.close()
+def find_package_dirs():
+ package_dirs = {'openerp-server': 'bin'}
+ for mod, path in find_addons():
+ package_dirs['openerp-server.addons.' + mod] = path
+ return package_dirs
+
+class openerp_server_install(install):
+ def run(self):
+ # create startup script
+ start_script = "#!/bin/sh\ncd %s\nexec %s ./openerp-server.py $@\n"\
+ % (join(self.install_libbase, "openerp-server"), sys.executable)
+ # write script
+ f = open('openerp-server', 'w')
+ f.write(start_script)
+ f.close()
+ install.run(self)
+
+
+
+
setup(name = name,
version = version,
- description = "Tiny's Enterprise Resource Planning",
+ description = description,
long_description = long_desc,
- url = 'http://tinyerp.com',
- author = 'Tiny.be',
- author_email = 'info@tiny.be',
+ url = url,
+ author = author,
+ author_email = author_email,
classifiers = filter(None, classifiers.split("\n")),
- license = 'GPL',
+ license = license,
data_files = data_files(),
- packages = ['tinyerp-server', 'tinyerp-server.addons',
- 'tinyerp-server.ir',
- 'tinyerp-server.osv',
- 'tinyerp-server.ssl',
- 'tinyerp-server.service', 'tinyerp-server.tools',
- 'tinyerp-server.pychart', 'tinyerp-server.pychart.afm',
- 'tinyerp-server.report',
- 'tinyerp-server.report.printscreen',
- 'tinyerp-server.report.render',
- 'tinyerp-server.report.render.rml2pdf',
- 'tinyerp-server.report.render.rml2html',
- 'tinyerp-server.wizard', 'tinyerp-server.workflow'] + \
- list(find_addons()),
- package_dir = {'tinyerp-server': 'bin'},
- scripts = ['tinyerp-server']
- )
-
-# vim:expandtab:tw=80
+ cmdclass = {
+ 'install' : openerp_server_install,
+ },
+ scripts = ['openerp-server'],
+ packages = [
+ '.'.join(['openerp-server'] + package.split('.')[1:])
+ for package in find_packages()
+ ],
+ include_package_data = True,
+ package_data = {
+ '': ['*.yml', '*.xml', '*.po', '*.pot', '*.csv'],
+ },
+ package_dir = find_package_dirs(),
+ install_requires = [
+ 'lxml',
+ 'mako',
+ 'python-dateutil',
+ 'psycopg2',
+ 'pychart',
+ 'pydot',
+ 'pytz',
+ 'reportlab',
+ 'caldav',
+ 'pyyaml',
+ 'pywebdav',
+ 'feedparser',
+ ],
+ extras_require={
+ 'SSL' : ['pyopenssl'],
+ },
+ **py2exe_keywords
+)
+
+if has_py2exe:
+ # Sometime between pytz-2008a and pytz-2008i common_timezones started to
+ # include only names of zones with a corresponding data file in zoneinfo.
+ # pytz installs the zoneinfo directory tree in the same directory
+ # as the pytz/__init__.py file. These data files are loaded using
+ # pkg_resources.resource_stream. py2exe does not copy this to library.zip so
+ # resource_stream can't find the files and common_timezones is empty when
+ # read in the py2exe executable.
+ # This manually copies zoneinfo into the zip. See also
+ # http://code.google.com/p/googletransitdatafeed/issues/detail?id=121
+ import pytz
+ import zipfile
+ # Make sure the layout of pytz hasn't changed
+ assert (pytz.__file__.endswith('__init__.pyc') or
+ pytz.__file__.endswith('__init__.py')), pytz.__file__
+ zoneinfo_dir = os.path.join(os.path.dirname(pytz.__file__), 'zoneinfo')
+ # '..\\Lib\\pytz\\__init__.py' -> '..\\Lib'
+ disk_basedir = os.path.dirname(os.path.dirname(pytz.__file__))
+ zipfile_path = os.path.join(py2exe_keywords['options']['py2exe']['dist_dir'], 'library.zip')
+ z = zipfile.ZipFile(zipfile_path, 'a')
+
+ for absdir, directories, filenames in os.walk(zoneinfo_dir):
+ assert absdir.startswith(disk_basedir), (absdir, disk_basedir)
+ zip_dir = absdir[len(disk_basedir):]
+ for f in filenames:
+ z.write(os.path.join(absdir, f), os.path.join(zip_dir, f))
+
+ z.close()