/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* 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 .
*/
#include "pyuno_impl.hxx"
#include <typelib/typedescription.hxx>
#include <com/sun/star/script/CannotConvertException.hpp>
using com::sun::star::uno::RuntimeException;
using com::sun::star::uno::XInterface;
using com::sun::star::uno::TypeDescription;
namespace pyuno
{
void raisePyExceptionWithAny(
const css::uno::Any &anyExc )
{
try
{
Runtime runtime;
PyRef exc = runtime.any2PyObject( anyExc );
if ( exc.is() )
{
PyRef type( getClass( anyExc.getValueTypeName(),runtime ) );
PyErr_SetObject( type.get(), exc.get());
}
else
{
css::uno::Exception e;
anyExc >>= e;
OUString buf =
"Couldn't convert uno exception to a python exception (" +
anyExc.getValueTypeName() +
": " + e.Message +
")" ;
PyErr_SetString(
PyExc_SystemError,
OUStringToOString(buf,RTL_TEXTENCODING_ASCII_US).getStr() );
}
}
catch (
const css::lang::IllegalArgumentException & e)
{
PyErr_SetString( PyExc_SystemError,
OUStringToOString( e.Message, RTL_TEXTENCODING_ASCII_US).getStr() );
}
catch (
const css::script::CannotConvertException & e)
{
PyErr_SetString( PyExc_SystemError,
OUStringToOString( e.Message, RTL_TEXTENCODING_ASCII_US).getStr() );
}
catch (
const RuntimeException & e)
{
PyErr_SetString( PyExc_SystemError,
OUStringToOString( e.Message, RTL_TEXTENCODING_ASCII_US).getStr() );
}
}
/// @throws RuntimeException
static PyRef createClass(
const OUString & name,
const Runtime &runtime )
{
// assuming that this is never deleted !
// note I don't have the knowledge how to initialize these type objects correctly !
TypeDescription desc( name );
if ( ! desc.is() )
{
throw RuntimeException(
"pyuno.getClass: uno exception " + name +
" is unknown" );
}
bool isStruct = desc.get()->eTypeClass == typelib_TypeClass_STRUCT;
bool isExc = desc.get()->eTypeClass == typelib_TypeClass_EXCEPTION;
bool isInterface = desc.get()->eTypeClass == typelib_TypeClass_INTERFACE;
if ( !isStruct && !isExc && ! isInterface )
{
throw RuntimeException(
"pyuno.getClass: " + name +
"is a " +
OUString::createFromAscii( typeClassToString(
static_cast <css::uno::TypeClass>(desc.
get()->eTypeClass)) ) +
", expected EXCEPTION, STRUCT or INTERFACE" );
}
// retrieve base class
PyRef base;
if ( isInterface )
{
typelib_InterfaceTypeDescription *pDesc = reinterpret_cast <typelib_InterfaceTypeDescription *>(desc.get());
if ( pDesc->pBaseTypeDescription )
{
base = getClass( pDesc->pBaseTypeDescription->aBase.pTypeName, runtime );
}
else
{
// must be XInterface !
}
}
else
{
typelib_CompoundTypeDescription *pDesc = reinterpret_cast <typelib_CompoundTypeDescription*>(desc.get());
if ( pDesc->pBaseTypeDescription )
{
base = getClass( pDesc->pBaseTypeDescription->aBase.pTypeName, runtime );
}
else
{
if ( isExc )
// we are currently creating the root UNO exception
base = PyRef(PyExc_Exception);
}
}
PyRef args( PyTuple_New( 3 ), SAL_NO_ACQUIRE, NOT_NULL );
PyRef pyTypeName = ustring2PyString( name /*.replace( '.', '_' )*/ );
PyRef bases;
if ( base.is() )
{
{ // for CC, keeping ref-count being 1
bases = PyRef( PyTuple_New( 1 ), SAL_NO_ACQUIRE );
}
PyTuple_SetItem( bases.get(), 0 , base.getAcquired() );
}
else
{
bases = PyRef( PyTuple_New( 0 ), SAL_NO_ACQUIRE );
}
PyTuple_SetItem( args.get(), 0, pyTypeName.getAcquired());
PyTuple_SetItem( args.get(), 1, bases.getAcquired() );
PyTuple_SetItem( args.get(), 2, PyDict_New() );
PyRef ret(
PyObject_CallObject(reinterpret_cast <PyObject *>(&PyType_Type) , args.get()),
SAL_NO_ACQUIRE );
// now overwrite ctor and attrib functions
if ( isInterface )
{
PyObject_SetAttrString(
ret.get(), "__pyunointerface__" ,
ustring2PyString(name).get() );
}
else
{
PyRef ctor = getObjectFromUnoModule( runtime,"_uno_struct__init__" );
PyRef setter = getObjectFromUnoModule( runtime,"_uno_struct__setattr__" );
PyRef getter = getObjectFromUnoModule( runtime,"_uno_struct__getattr__" );
PyRef repr = getObjectFromUnoModule( runtime,"_uno_struct__repr__" );
PyRef eq = getObjectFromUnoModule( runtime,"_uno_struct__eq__" );
PyRef ne = getObjectFromUnoModule( runtime,"_uno_struct__ne__" );
PyObject_SetAttrString(
ret.get(), "__pyunostruct__" ,
ustring2PyString(name).get() );
PyObject_SetAttrString(
ret.get(), "typeName" ,
ustring2PyString(name).get() );
PyObject_SetAttrString(
ret.get(), "__init__" , ctor.get() );
PyObject_SetAttrString(
ret.get(), "__getattr__" , getter.get() );
PyObject_SetAttrString(
ret.get(), "__setattr__" , setter.get() );
PyObject_SetAttrString(
ret.get(), "__repr__" , repr.get() );
PyObject_SetAttrString(
ret.get(), "__str__" , repr.get() );
PyObject_SetAttrString(
ret.get(), "__eq__" , eq.get() );
PyObject_SetAttrString(
ret.get(), "__ne__" , ne.get() );
}
return ret;
}
bool isInstanceOfStructOrException( PyObject *obj)
{
PyRef attr(
PyObject_GetAttrString(obj, "__class__" ),
SAL_NO_ACQUIRE );
if (attr.is())
return PyObject_HasAttrString(attr.get(), "__pyunostruct__" );
else
return false ;
}
bool isInterfaceClass( const Runtime &runtime, PyObject * obj )
{
const ClassSet & set = runtime.getImpl()->cargo->interfaceSet;
return set.find( obj ) != set.end();
}
PyRef getClass( const OUString & name , const Runtime &runtime)
{
PyRef ret;
RuntimeCargo *cargo =runtime.getImpl()->cargo;
ExceptionClassMap::iterator ii = cargo->exceptionMap.find( name );
if ( ii == cargo->exceptionMap.end() )
{
ret = createClass( name, runtime );
cargo->exceptionMap[name] = ret;
if ( PyObject_HasAttrString(
ret.get(), "__pyunointerface__" ) )
cargo->interfaceSet.insert( ret );
PyObject_SetAttrString(
ret.get(), "__pyunointerface__" ,
ustring2PyString(name).get() );
}
else
{
ret = ii->second;
}
return ret;
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
Messung V0.5 C=92 H=96 G=93
¤ Dauer der Verarbeitung: 0.11 Sekunden
(vorverarbeitet)
¤
*© Formatika GbR, Deutschland