Implementation of DCOM wire protocol (MSRPC) to enable development of Bi-Directional, Pure and Non-Native Java applications which can interoperate with any COM component. The implementation is not dependent on JNI for COM interoperability. It also allows for complete Windows Registry manipulation operations (create, read, update, delete) using the WinReg interface. More information about j-Interop is available here.
- Getting Started
- Frequently Asked Questions
- Third Party Dependencies
- Installation
- Examples
- License
- Third Party Licenses
- Support
- Acknowledgements
First, please follow the installation instructions. Once done, the best way to get started is by looking at the examples. You may want to skip the FAQs with [Advance] tag for the time being. You can get back to them once you are a bit comfortable with j-Interop.
If the COM server you are trying to access has a Windows COM client (executed from a Remote machine), then it is probably already configured for DCOM access.
If it is not so, please read FAQ (A5).
If the COM server you are using is a DLL
\OCX
(In-Proc), you may want to have a look at FAQ (A6).
<
Some important things, before you start:
- Please make sure that the Server Service and Remote Registry Service are running on the Workstation where the COM server resides. Read FAQ (A12) for more details.
- To avoid getting ACCESS DENIED exceptions by COM server, it is best to create a session under the identity of currently logged in user.
For e.g. if you are logged into your local machine under username administrator
, the session should be created as:
JISession session = JISession.createSession("localhost","administrator","PASSWORD");
For a Domain like MYDOMAIN
, it can be likewise:
JISession session = JISession.createSession("MYDOMAIN","DOMAINUSER","USERPASSWORD");
Incase granting administrators
permission is a concern, then :
- You can create a local user under "Users" group.
- Then go to Control Panel > Administrative Tools > Local Security Policy > Security Settings > Local Policies > Security Options :
- Double-click "DCOM: Machine Access Restrictions" policy, click Edit Security, add the user created above, allow "Remote Access"
- Double-click "DCOM: Machine Launch Restrictions" policy, click Edit Security, add the user created above, allow "Local Launch", "Remote Launch", "Local Activation", "Remote Activation"
- Go to Control Panel > Administrative Tools > Component Services > Computers > right-click My Computer > click Properties > click COM Security tab :
- In Access Permissions section, click Edit Default > add the user created above, allow "Remote Access"
- In Launch and Activation Permissions section > click Edit Default > add the user created above, allow "Local Launch", "Remote Launch", "Local Activation", "Remote Activation" (The Component Services section, to be more accurate, you can go to a specific component, and grant permission from there, instead of from "My Computer", which is a blanket grant)
Sometimes the Windows Firewall will act up if not configured properly, so please make sure that you have either configured it for DCOM protocol or turned it off. Please note that the firewall issue will prevent all DCOM Windows applications to fail as well.
Please make sure that your Windows machine (where COM server is hosted) is up to date with all the Service packs and updates from Microsoft. Many a times issues are due to improper machine configuration.
This is a good article on how to configure DCOM using DCOMCnfg.
If you are not really familiar with DCOM, then this article provides a good overview. The architecture document can be found here.
- What is the JRE version compatibility of j-Interop ?
- How do I install it ?
- What threading model do the COM servers adhere to while in use from j-Interop ? [Advance]
- What threading model does j-Interop follow ? [Advance]
- How do I configure my COM server for DCOM access ?
- My COM Server is a
DLL
\OCX
(Inproc server), how do I make it work with j-Interop ? - Why could the library not do step 7 for me automatically?
- What all COM Interfaces are directly supported ?
- What data types are supported by j-Interop ?
- COM to Java Data Type Mappings [Advance]
- Do I have to do any reference counting,memory management etc. ?
- Any configuration to be done before using j-Interop ?
- Is there any logging done ?
- What type of License does it follow ?
Tested with JRE 1.4 and 1.6 on Windows XP(SP2), Windows Vista(SP1), Windows 2K3(SP2) and Ubuntu 8.4(desktop). Compatible with JRE version 1.4 and above.
Please see the third party dependencies and installation section.
From MSDN:
On Windows side, Local servers (EXEs) are in full control of the kind of apartment(threading model) that COM is using for their objects.
The local server calls CoInitializeEx
on one or more threads and registers the class factory (or multiple class factories) from the appropriate thread with the appropriate threading model.
The In-process servers (DLLs), however run in the context of their client i.e they run in the apartment the client gives them.
By client, I don't mean j-Interop here, but a Windows COM Client.
In-process components indicate the threading model they are ready to satisfy by placing a named value (ThreadingModel
) under their InprocServer32
key in the registry:
[HKEY_CLASSES_ROOT\CLSID\{clsid}\InprocServer32]
"ThreadingModel"="Both" or "ThreadingModel"="Apartment" or "ThreadingModel"="Free".
[HKEY_CLASSES_ROOT\CLSID\{clsid}\InprocServer32]
"ThreadingModel"="Both" or "ThreadingModel"="Apartment" or "ThreadingModel"="Free"
If ThreadingModel is not specified, the component is assumed to follow the assumptions for STA-Main
and can only be loaded into the main STA in a process.
A value of Both
indicates that the component can be loaded in both MTAs and STAs.
A value of Apartment
indicates that the component can be loaded into any STA.
A value of Free
indicates that the component can be loaded into an MTA, but not into an STA.
j-Interop follows the Apartment
model i.e regardless of threading model or the component type (Local or Inproc) used by the COM Server, j-Interop synchronizes all calls to the COM Server per org.jinterop.dcom.core.JIComServer
instance (which is the starting point for each COM Server).
For e.g. in the following piece of code :
JISession session = JISession.createSession("DOMAIN", "USERNAME", "PASSWORD");
JIComServer comServer = new JIComServer(JIProgId.valueOf("Excel.Application"), "127.0.0.1", session);
IJIComObject comObject = comServer.createInstance();
IJIDispatch dispatch = (IJIDispatch)JIObjectFactory.narrowObject(comObject.queryInterface(IJIDispatch.IID));
IJITypeInfo typeInfo = dispatch.getTypeInfo(0);
FuncDesc funcDesc = typeInfo.getFuncDesc(0);
int dispId = dispatch.getIDsOfNames("Visible");
JIVariant variant = new JIVariant(Boolean.TRUE);
dispatch.put(dispId, variant);<p>
Calls from all interfaces (dispatch
and typeInfo
), even if they are running on different threads, are synchronized at the JIComServer
(comServer
) level.
That said, within an application, there can be more than one JIComServer
s running at the same time and they run independent of each other.
Ideally if your COM server is actively being used for remote access, then it is perhaps already configured for DCOM. If not you can configure it by following steps mentioned here or here.
For Windows XP (SP2), this is a good link.
Alternatively, this is also a good article.
Ideally if your COM server is actively being used for remote access, then it is perhaps already configured for DCOM. If not, you have 2 ways to do this. Both ways are recommended by Microsoft. I personally prefer the Easiest way.
Let the j-Interop library do this for you. You can set the autoRegistration
flag in the JISystem
or the JIClsid
, JIProgId
classes.
When the library encounters a Class not registered
exception, it will perform all the registry changes if the autoRegistration flag is set.
And then re-attempt loading the COM Server.
Please have a look at MSSysInfo
, MSWMI
examples.
From MSDN, here (skip to the section titled below):
Modify registry to force remoting of the object
- Use the OLE/COM Object viewer (
Oleview.exe
) that is shipped with Microsoft Visual C++ and locate the ProgID in the form ofOLEComponent.Object
under All Objects. - Select the COM object, and then from the Object menu, select CoCreateInstance Flags. Make sure that only
CLSCTX_LOCAL_SERVER
is selected. - Next, under the Implementation and Inproc Server tabs select Use Surrogate Process and leave the "Path to Custom Surrogate" blank, which allows the
Dllhost.exe
file to be loaded and the COM DLL brought within it's process space.
If you do not have Microsoft Visual C++, the OLE/COM Object Viewer utility is also available for download from the following Microsoft Web site: http://www.microsoft.com/downloads/details.aspx?familyid=5233b70d-d9b2-4cb5-aeb6-45664be858b6&displaylang=en.
- For each CLSID encountered in the DLL component, create an AppID value under
HKCR\CLSID\{clsid}
which contains the same value as the CLSID.HKCR\CLSID\{clsid}
AppID = {appid value}
- Using the same AppID value, then create an AppID key under
HKCR\AppID
.HKCR\AppID\{appid}
- Under
HKCR\AppID\{appid}
, then create the following values:(Default) =
DllSurrogate =
For example, let's say YourDll.dll
contains one class called MyClass
with CLSID {6A048AAA-7DDD-4CCC-BE59-9BBB746E5C6E}
.
To host YourDll.dll
using dllhost
, you will need to create the following entries in the registry:
HKCR\CLSID\{6A048AAA-7DDD-4CCC-BE59-9BBB746E5C6E}
- AppID =
{6A048AAA-7DDD-4CCC-BE59-9BBB746E5C6E}
- AppID =
HKCR\AppID\{6A048AAA-7DDD-4CCC-BE59-9BBB746E5C6E}
- (Default) =
Your DLL Surrogate
- DllSurrogate =
- (Default) =
You would then be able to configure this surrogate by running DCOMCNFG
and looking for the Application entry called Your DLL Surrogate
.
Please note that the use of surrogates for accessing DLLs is not a j-Interop specification, but a COM specification.
In any DCOM case for accessing a DLL
\OCX
you would need the DLLHOST.
It is a Microsoft DCOM DLL Host Process.
If the COM server being accessed is an Exe, like MSWord or MSExcel or IE then this is not required. But for DLLs , it is required.
This is how the DCOM clients talk to In-Process Servers.
You can obtain more info from here (please open in IE for proper viewing).
Also, it would be best to view j-Interop as a DCOM client when accessing COM from Java. It would be much easier to work with it then. Whatever configurations are required for a DCOM client, will be required for j-Interop also.
j-Interop behaves as a COM client to the COM Server, changes in step 6 have to be done at the server machine's registry. It is best that the user initiate those actions instead of the library doing these silently.
All automation interfaces like IDispatch
, ITypeInfo
, ITypeLib
, IEnumVariant
are directly supported.
You can start using them right away.
All DCOM datatypes including VARIANT
s are supported by j-Interop.
The only limitation in the present version is that Arrays up to Maximum 2 dimensions are accepted currently.
After going through some of the examples, it should be fairly simple (or so I hope) to gauge what COM data type maps to which Java type, but here are some hints anyways:
-
All primitive data types of COM map to their corresponding Java counterparts, with the exception of
long
, which maps toint
at the Java side. -
IIDs
are j-Interop uuids (represented asjava.lang.String
). -
OLECHAR
isJIString(LPWSTR)
. -
Top level pointers are pointers that are NOT elements of arrays, NOT members of structures or unions. All Top level "Interface" pointers (
IUnknown*
,IDispatch*
,IDispatch**
,ITypeLib**
etc.) are mapped to there referents themselves. For e.g :ITypeLib** ppTLib
orITypeInfo* ppTInfo
maps directly toJIInterfacePointer
and NOT toJIPointer(JIInterfacePointer)
. -
All Other Top level pointers are mapped as following:
- First level indirection is mapped directly to there referents.
For e.g.
int*
isint
,double*
isdouble
,BSTR*
isJIString(BSTR_FLAG)
,OLECHAR FAR *
ptr isJIString(LPWSTR)
. - Second and subsequent level indirections are mapped to there (level - 1) indirections.
For e.g.
int**
maps toJIPointer(int)
,int***
maps toJIPointer(JIPointer(int))
,double**
maps toJIPointer(double)
,double***
maps toJIPointer(JIPointer(double))
. - All data types can be mapped like rule 4 and 5(a,b).
The exception to above rules are
BSTR**
andVARIANT**
. SinceBSTR
andVARIANT
s in COM are inherently pointers themselves, they follow rule v(b) only after 3rd level of indirection. i.e.BSTR*
andBSTR**
are both mapped toJIString(BSTR)
.VARIANT*
andVARIANT**
are both mapped toJIVariant(,byRef=true);
3rd and subsequent level indirections ofBSTR
orVARIANT
s are mapped according to rule (level - 2). for e.g. theBSTR***
mapped toJIPointer(JIString(BSTR))
,VARIANT***
is mapped toJIPointer(JIVariant(,byRef=true));
- First level indirection is mapped directly to there referents.
For e.g.
-
When using
IJIDispatch
, you will be required to useJIVariants
. Automation in COM does not allow indirection beyond level 2. So simple mappings would suffice for non pointer types, and for pointer types as parameters, please use thebyRef
flag of JIVariant.Most of the times the MSDN documentation itself will tell you what the data type stands for, just use the corresponding type in j-Interop.
For e.g., from MSDN:
IDispatch::GetIDsOfNames HRESULT GetIDsOfNames( REFIID riid, OLECHAR FAR* FAR* rgszNames, unsigned int cNames, LCID lcid, DISPID FAR* rgDispId );
riid
is Reserved for future use. Must be IID_NULL.rgszNames
is Passed-in array of names to be mapped.cNames
is Count of the names to be mapped.lcid
is The locale context in which to interpret the namesrgDispId
is Caller-allocated array, each element of which contains an identifier (ID) corresponding to one of the names passed in thergszNames
array. The first element represents the member name. The subsequent elements represent each of the member's parameters.
j-Interop definition for these would be:
riid
isuuid
.rgszNames
isJIArray(JIPointer(JIString(LPWSTR)))
.cNames
isint
.lcid
isint
(for this you will have to look up MSDN, it translates to along
, which maps toint
in j-Interop.rgDispId
isJIArray(Integer)
.
-
Embedded Pointers (members of structures, unions, elements of arrays) are represented by
JIPointer(type)
, like theOLECHAR FAR* FAR*
rgszNames
in Point (6). It first got mapped toJIArray
since it is a top level pointer. Within the array, it was supposed to return pointers toOLECHAR
, therefore it got mapped toJIPointer(JIString(LPWSTR))
forming the whole definition asJIArray(JIPointer JIString(LPWSTR)))
.For more examples, please have a look at
JITypeInfoImpl.java (getFuncDesc API)
. It will show you how the mapping is done between embedded pointers and j-Interop types. Please keep MSDN handy for having a look at the actual C++ struct. Search onITypeInfo -->GetFuncDesc(...)
there. Also see theFUNCDESC
structure. -
Unless specified otherwise (like the dimensions are provided) in the documentation, all arrays are
conformant
arrays. -
You can also look up the IDL for the component if only
size_is()
is present then Array"isConformant" (true)
. If bothsize_is()
andlength_is()
are present then Array"isConformant" and "isVarying" (true,true)
. -
When you are implementing callbacks, if the COM source interface has a single level interface pointer like
IDispatch*
then mapping to local java class will beJIInterfacePointer
itself. But second level pointers to interfaces likeIDispatch**
must be declared asJIVariants
only.For e.g.:
IDispatch*
can be mapped toIJIDispatch
, butIDispatch**
should be mapped toJIVariant
only. seeMSInternetExplorer --> DWebBrowserEvents (BeforeNavigate2 and NewWindow3)
for more details.
No, the library does all this on it's own (including pinging the COM server for keeping it alive).
Yes, please make sure that the Server service and Remote Registry service is running on the target workstation (where the COM Server is hosted).
This is required for reading the registry to map the ProgIds
to their clsids
.
If you can't have this, then please use clsid
instead of progId
.
The progIdVsClsidDB.properties
maintains a mapping of progId
Vs there clsids
, if this file is present in the classpath.
This file is consulted before the registry for the progId
.
Also, if you are working with GUI components and would like to make them visible / interactive, then make sure that you read up (A5) and setup the COM Server for "Interactive User". By default, it is the "Launching User". If this option is set, then the COM Server will not present it's GUI. It is best to use this for all silient operations like working with DBs, or using Excel formulas etc.
Yes, j-Interop uses java logging by default (and to the console), but you can configure a handler for this logger to redirect output to logger mechanisms of your own choice.
There is also a method in JISystem.setInBuiltLogHandler
which creates a handler to store the logs to a file in the user's temp directory as j-Interop.log
. (e.g. for Windows systems it should be C:\Documents and Settings\your_username\Local Settings\Temp
)
Please see the license section.
This library is distributed under the LGPL v3.0 (http://www.gnu.org/licenses/lgpl-3.0.html).
Please refer to the accompanying lgpl.txt
for more details.
Those not speaking 'legal'ise can have a look here as to what it means. Incase you would like to get a LGPL v2.1 exception for your library, please write to us.
j-Interop depends on :
- jarapac - DCE/RPC protocol
- jcifs - SMB authentication
- iwombat - Open source library for UUID generation
- Gnu Crypto - Cryptographic library from GNU
Please extract all files from the j-Interop.zip
. The lib
folder has all the jars required to use j-Interop library.
Installation of j-interop.jar
is similar to any standard jar file, just add it to the classpath.
As a performance improvement, the j-Interop comes with a progIdVsClsidDB.properties
file, which gets updated with the clsid of each ProgId.
This prevents the library from accessing the Windows Registry for the clsid, when the same COM server is required again.
Please make sure that progIdVsClsidDB.properties
is also included in the classpath.
Commented examples are located in the examples
folder.
jcifs
is licensed under LGPL.jarapac
has forked out to jcifs implementation and follows the same licensing scheme.iwombat
donated the pieces of code required by the library, Many Thanks to Bob Combs and iwombat.com.
If you have any issues, please post it at the support forum or file a bug in the bugs database maintained at project website.
This product includes software developed by the iWombat.com and all other members of the Third Party Dependencies.