python/ext_build.py (70 lines of code) (raw):

# # Licensed to the Apache Software Foundation (ASF) under one # or more contributor license agreements. See the NOTICE file # distributed with this work for additional information # regarding copyright ownership. The ASF licenses this file # to you under the Apache License, Version 2.0 (the # "License"); you may not use this file except in compliance # with the License. You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, # software distributed under the License is distributed on an # "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY # KIND, either express or implied. See the License for the # specific language governing permissions and limitations # under the License. # import os import cffi.pkgconfig from cffi import FFI ffibuilder = FFI() # cdef() expects a single string declaring the C types, functions and # globals needed to use the shared object. It must be in valid C syntax # with cffi extensions cdefs = open('cproton.h').read() ffibuilder.cdef(cdefs) proton_base = '.' proton_c_src = os.path.join(proton_base, 'src') proton_core_src = os.path.join(proton_c_src, 'core') proton_c_include = os.path.join(proton_base, 'include') sources = [] extra = [] libraries = [] for root, _, files in os.walk(proton_core_src): for file_ in files: if file_.endswith(('.c', '.cpp')): sources.append(os.path.join(root, file_)) if os.name == 'nt': sources += [ os.path.join(proton_c_src, 'compiler', 'msvc', 'start.c') ] elif os.name == 'posix': sources += [ os.path.join(proton_c_src, 'compiler', 'gcc', 'start.c') ] extra += ['-std=c99'] sources.append(os.path.join(proton_c_src, 'sasl', 'sasl.c')) sources.append(os.path.join(proton_c_src, 'sasl', 'default_sasl.c')) pkgconfig = [] if os.name == 'nt': libraries += ['crypt32', 'secur32'] sources.append(os.path.join(proton_c_src, 'ssl', 'schannel.cpp')) else: try: ssl_pkgcfg = cffi.pkgconfig.flags_from_pkgconfig(['openssl']) sources.append(os.path.join(proton_c_src, 'ssl', 'openssl.c')) pkgconfig.append('openssl') except cffi.pkgconfig.PkgConfigError: # Stub ssl sources.append(os.path.join(proton_c_src, 'ssl', 'ssl_stub.c')) # Stub sasl try: sasl_pkgcfg = cffi.pkgconfig.flags_from_pkgconfig(['libsasl2']) sources.append(os.path.join(proton_c_src, 'sasl', 'cyrus_sasl.c')) pkgconfig.append('libsasl2') except cffi.pkgconfig.PkgConfigError: sources.append(os.path.join(proton_c_src, 'sasl', 'cyrus_stub.c')) include_dirs = [proton_c_include, proton_c_src] macros = [('PROTON_DECLARE_STATIC', None)] c_code = r""" #include "proton/version.h" #include "proton/types.h" #include "proton/object.h" #include "proton/error.h" #include "proton/condition.h" #include "proton/connection.h" #include "proton/session.h" #include "proton/link.h" #include "proton/terminus.h" #include "proton/delivery.h" #include "proton/disposition.h" #include "proton/transport.h" #include "proton/event.h" #include "proton/message.h" #include "proton/sasl.h" #include "proton/ssl.h" #include "proton/codec.h" #include "proton/connection_driver.h" #include "proton/cid.h" static void pn_pyref_incref(void *object); static void pn_pyref_decref(void *object); static int pn_pyref_refcount(void *object) { return 1; } pn_connection_t *pn_cast_pn_connection(void *x) { return (pn_connection_t *) x; } pn_session_t *pn_cast_pn_session(void *x) { return (pn_session_t *) x; } pn_link_t *pn_cast_pn_link(void *x) { return (pn_link_t *) x; } pn_delivery_t *pn_cast_pn_delivery(void *x) { return (pn_delivery_t *) x; } pn_transport_t *pn_cast_pn_transport(void *x) { return (pn_transport_t *) x; } static pn_class_t* PN_PYREF; PN_HANDLE(PN_PYCTX); static pn_class_t* pn_create_pyref() { return pn_class_create("pn_pyref", NULL, NULL, pn_pyref_incref, pn_pyref_decref, pn_pyref_refcount); } pn_event_t *pn_collector_put_py(pn_collector_t *collector, void *context, pn_event_type_t type) { return pn_collector_put(collector, PN_PYREF, context, type); } void pn_record_def_py(pn_record_t *record) { pn_record_def(record, PN_PYCTX, PN_PYREF); } void *pn_record_get_py(pn_record_t *record) { return pn_record_get(record, PN_PYCTX); } void pn_record_set_py(pn_record_t *record, void *value) { pn_record_set(record, PN_PYCTX, value); } ssize_t pn_message_encode_py(pn_message_t *msg, char *bytes, size_t size) { int err = pn_message_encode(msg, bytes, &size); if (err == 0) return size; else return err; } ssize_t pn_data_format_py(pn_data_t *data, char *bytes, size_t size) { int err = pn_data_format(data, bytes, &size); if (err == 0) return size; else return err; } int pn_ssl_get_peer_hostname_py(pn_ssl_t *ssl, char *hostname, size_t size) { return pn_ssl_get_peer_hostname(ssl, hostname, &size); } const char *pn_event_class_name_py(pn_event_t *event) { const pn_class_t *class = pn_event_class(event); return class ? pn_class_name(class) : 0; } void init() { PN_PYREF = pn_create_pyref(); } """ if len(pkgconfig) == 0: ffibuilder.set_source( "cproton_ffi", c_code, define_macros=macros, extra_compile_args=extra, sources=sources, include_dirs=include_dirs, libraries=libraries ) else: ffibuilder.set_source_pkgconfig( "cproton_ffi", pkgconfig, c_code, define_macros=macros, extra_compile_args=extra, sources=sources, include_dirs=include_dirs ) if __name__ == "__main__": ffibuilder.compile(verbose=True)