A Different Frame of Mind
Simple blog to talk about software development and anything else that piques my interest.
Syntax Hilighter
Wednesday, May 4, 2011
Happy Birthday
Happy Birthday Exavier. I Love you.
Tuesday, March 15, 2011
Jenkins-CI Accurev Plugin New Version 0.6.13
I successfully released v0.6.13 of the Jenkins-CI Accurev plugin. Yay!
Change log from the Wiki.
Version 0.6.13 (3/15/2011)
- Added option to purge overlaps when using a workspace.
- Added option to make purge workspace optional when the previous build fails.
- merged changes for snapshotting builds back into the current trunk (HUDSON-5196)
Next I will start working on bumping the parent plugin version to something more modern.
Many thanks to Alan Harder, and Scott Tatum for their help and work on this plugin.
Many thanks to Alan Harder, and Scott Tatum for their help and work on this plugin.
Friday, March 4, 2011
Jenkins-CI Accurev plugin maintenance.
I started working on the Jenkins-CI Accurev SCM Plugin to integrate some additional features that I need for doing automated builds.
from the github pull request for the new features:
I have added two options to the Accurev SCM plugin that I think others may benefit from when using an accurev workspace:
I am now working on integrating these new features with some unpublished updates from other folks and trying to bump the SCM API parent version up to something more current. hopefully I'll be able to merge all this together and release an updated plugin in the near future.
from the github pull request for the new features:
I have added two options to the Accurev SCM plugin that I think others may benefit from when using an accurev workspace:
- there is an option to purge/revert overlaps found in the workspace.
In some accurev scenarios the act of performing a build in a workspace will result in overlaps for subsequent builds. when this happens, the update/populate of the workspace will result in a failed build since accurev will refuse to update a workspace with overlaps. by finding overlaps and purging them prior to attempting to do an accurev update, we no longer fail builds just because of overlaps; and we do not cause the accurev plugin to purge/populate on subsequent builds.
- The behavior of purging the workspace on subsequent builds after a build failure occurs is now exposed as an option.
It may be important to not purge a workspace when a build fails. in my case this is important due to the way our config mgmt teams promote code between streams. even though a build fails, I don't want to re-checkout the entire workspace; 99.9% of the time the revert overlaps in conjunction with update in a workspace options will resolve build problems.
Friday, December 3, 2010
JAX-WS typeSafeEnum
I recently answered a question on stackoverflow.com concerning how to get WSIMPORT to generate an enum when generating java code from a wsdl that contains enum restricted types/elements.
The original question:
http://stackoverflow.com/questions/4235074/no-enums-generated-by-jax-ws-ri-2-2-1-for-amazon-ecs-wsdl-file/4348209#4348209
The question is:
and my answer:
JAXB client customization of the binding is exactly/pretty close to what you need -- you need to use a JAX-WS client customization for your example. JAXB and JAX-WS customizations essentially allows you to augment the definition of schema elements for WSDL/schema's that you do not control. there are MANY different things you can accomplish such as mapping xml element names to custom java elements, altering the generated API, and to answer your question, generating type-safe enum classes for elements that are restricted with an enum.
There are two ways/parts to doing a client customization for JAX-WS.
1) if the WSDL imports an external schema file 2) if the WSDL contains the entire schema definition without any imports
if the wsdl imports an external schema file then,
basically you need to create a new file (typically with a jxb extension, but it really doesn't matter) that you will maintain along side the wsdl you are generating the client stub/api for. typically I name these files schema-file-name_clientcustomization.jxb
every time you get an updated wsdl, you should validate that your JXB file is still valid for that wsdl. The biggest things I've found to look for, especially with enum restrictions, is restricted value changes, namespace changes, type name changes, etc..
the content of this new file will look something similar to this:
to use this JXB file, you need to reference it in your ant task, like so:
if the WSDL defines the entire schema internally, then you need to use a JAX-WS customization file. This case is what matches your question.
http://jax-ws.java.net/nonav/2.1.7/docs/customizations.html
JAX-WS client customization is very similar to the JAXB customization. The idea is identical, for the most part the JAX-WS portion of the customization file will alter generated artifacts that are specifically related to WSDL, whereas the embedded JAXB customization performs the same function as an external customization file: it alters the generated objects based on the schema.
The big difference is that rather than tell the JAXB parser where the schema file is, you provide a binding section that selects the schema definition (using XPATH) that you want to apply customization to.
This example I actually tested and verified to generate an Enum class for the element you are asking questions about, so you can copy this JAX-WS customization example verbatim.
you would then reference this JAX-WS customization file the same way you would reference the JXB file.
I did not validate the standalone JAXB customization example, since I really only included it as an example and as a precursor explanation for the JAX-WS customization example.
The JAX-WS customization example I did actually test/validate against the WSDL you have linked, so you should be able to use it as a starting point. I noticed that there are numerous enumerated restrictions in the defined WSDL, so I would assume you'll want to generate enums for most/all of them.
I hope this helps.
The original question:
http://stackoverflow.com/questions/4235074/no-enums-generated-by-jax-ws-ri-2-2-1-for-amazon-ecs-wsdl-file/4348209#4348209
The question is:
I'm attempting to generate artifacts for the following amazon wsdl:
http://webservices.amazon.com/AWSECommerceService/AWSECommerceService.wsdl
using the following ant task:
<taskdef name="wsimport" classname="com.sun.tools.ws.ant.WsImport"> <classpath> <pathelement location="${BUILD_LIBS}/jaxws-ri/lib/jaxws-tools.jar"/> </classpath> </taskdef> <target name="wsimport" depends="init"> <delete dir="${generated.src}" /> <mkdir dir="${generated.src}"/> <wsimport debug="true" keep="true" verbose="true" destdir="${generated.src}" package="com.amazon.webservices.ecs" wsdl="wsdl/AWSECommerceService.wsdl"/> </target>
but no java artifacts are generated for the following element which look like this:
<xs:element name="Condition"> <xs:simpletype> <xs:restriction base="xs:string"> <xs:enumeration value="All"> <xs:enumeration value="New"> <xs:enumeration value="Used"> <xs:enumeration value="Collectible"> <xs:enumeration value="Refurbished"> </xs:enumeration></xs:enumeration></xs:enumeration></xs:enumeration></xs:enumeration></xs:restriction> </xs:simpletype> </xs:element>
Basically no enums are generated even though all other elements are generated. Has anyone seen this problem before? I'm using jax-ws ri 2.2.1 http://jax-ws.java.net/2.2.1/
Thanks
and my answer:
JAXB client customization of the binding is exactly/pretty close to what you need -- you need to use a JAX-WS client customization for your example. JAXB and JAX-WS customizations essentially allows you to augment the definition of schema elements for WSDL/schema's that you do not control. there are MANY different things you can accomplish such as mapping xml element names to custom java elements, altering the generated API, and to answer your question, generating type-safe enum classes for elements that are restricted with an enum.
There are two ways/parts to doing a client customization for JAX-WS.
1) if the WSDL imports an external schema file 2) if the WSDL contains the entire schema definition without any imports
if the wsdl imports an external schema file then,
basically you need to create a new file (typically with a jxb extension, but it really doesn't matter) that you will maintain along side the wsdl you are generating the client stub/api for. typically I name these files schema-file-name_clientcustomization.jxb
every time you get an updated wsdl, you should validate that your JXB file is still valid for that wsdl. The biggest things I've found to look for, especially with enum restrictions, is restricted value changes, namespace changes, type name changes, etc..
the content of this new file will look something similar to this:
<jxb:bindings targetnamespace="http://java.sun.com/xml/ns/jaxb" version="1.0" xmlns:jxb="http://java.sun.com/xml/ns/jaxb" xmlns:xsd="http://www.w3.org/2001/XMLSchema"> <jxb:bindings node="/xsd:schema[@targetNamespace='SCHEMANAMESPACE']" schemalocation="NameOfYourSchemaFile.xsd"> <jxb:schemabindings> <jxb:package name="com.amazon.webservices.ecs"> </jxb:package></jxb:schemabindings> <jxb:bindings node="xsd:element[@name='Condition']/xsd:simpleType"> <jxb:typesafeenumclass name="ConditionEnum"> <jxb:typesafeenummember name="ALL" value="All"> <jxb:typesafeenummember name="NEW" value="New"> <jxb:typesafeenummember name="USED" value="Used"> <jxb:typesafeenummember name="COLLECTIBLE" value="Collectible"> <jxb:typesafeenummember name="REFURBISHED" value="Refurbished"> </jxb:typesafeenummember></jxb:typesafeenummember></jxb:typesafeenummember></jxb:typesafeenummember></jxb:typesafeenummember></jxb:typesafeenumclass> </jxb:bindings> </jxb:bindings> </jxb:bindings>Essentially this file defines augmentation that should be done to the referenced xsd file. all bindings elements in this file have a node attribute which is an XPATH expression that selects the schema item that you want to augment. in the example, I don't have any namespace or other information so I specified the XPATH to select just the element's simple type declaration. within that binding, we define the typesafeenumclass, this causes the jaxb/wsimport to generate an enum class to represent the referenced simple type. since it's an anonymous simple type, this effectively defines a class just for the referenced element. The generated class will be an ENUM who's memebers are defined by the "name" attribute of the typesafeEnumMember element.
to use this JXB file, you need to reference it in your ant task, like so:
<wsimport debug="true" destdir="${generated.src}" keep="true" package="com.amazon.webservices.ecs" verbose="true" wsdl="wsdl/AWSECommerceService.wsdl"> <binding dir="wsdl" includes="*.jxb"> </binding></wsimport>
if the WSDL defines the entire schema internally, then you need to use a JAX-WS customization file. This case is what matches your question.
http://jax-ws.java.net/nonav/2.1.7/docs/customizations.html
JAX-WS client customization is very similar to the JAXB customization. The idea is identical, for the most part the JAX-WS portion of the customization file will alter generated artifacts that are specifically related to WSDL, whereas the embedded JAXB customization performs the same function as an external customization file: it alters the generated objects based on the schema.
The big difference is that rather than tell the JAXB parser where the schema file is, you provide a binding section that selects the schema definition (using XPATH) that you want to apply customization to.
This example I actually tested and verified to generate an Enum class for the element you are asking questions about, so you can copy this JAX-WS customization example verbatim.
<jaxws:bindings wsdllocation="AWSECommerceService.wsdl" xmlns:jaxb="http://java.sun.com/xml/ns/jaxb" xmlns:jaxws="http://java.sun.com/xml/ns/jaxws" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" xmlns:xsd="http://www.w3.org/2001/XMLSchema"> <jaxws:bindings node="wsdl:definitions/wsdl:types/xsd:schema[@targetNamespace='http://webservices.amazon.com/AWSECommerceService/2010-11-01']"> <jaxb:schemabindings> <jaxb:package name="com.amazon.webservices.ecs"> </jaxb:package></jaxb:schemabindings> <jaxb:bindings node="xsd:element[@name='Condition']/xsd:simpleType"> <jaxb:typesafeenumclass name="ConditionEnum"> <jaxb:typesafeenummember name="ALL" value="All"> <jaxb:typesafeenummember name="NEW" value="New"> <jaxb:typesafeenummember name="USED" value="Used"> <jaxb:typesafeenummember name="COLLECTIBLE" value="Collectible"> <jaxb:typesafeenummember name="REFURBISHED" value="Refurbished"> </jaxb:typesafeenummember></jaxb:typesafeenummember></jaxb:typesafeenummember></jaxb:typesafeenummember></jaxb:typesafeenummember></jaxb:typesafeenumclass> </jaxb:bindings> </jaxws:bindings> </jaxws:bindings>
you would then reference this JAX-WS customization file the same way you would reference the JXB file.
I did not validate the standalone JAXB customization example, since I really only included it as an example and as a precursor explanation for the JAX-WS customization example.
The JAX-WS customization example I did actually test/validate against the WSDL you have linked, so you should be able to use it as a starting point. I noticed that there are numerous enumerated restrictions in the defined WSDL, so I would assume you'll want to generate enums for most/all of them.
I hope this helps.
Tuesday, July 13, 2010
Oracle Weblogic 10.3.3 JAX-WS Client
This post is primarily here because I am having trouble with a client-side JAX-WS implementation using Weblogic Server 10.3.3's JAX-WS implementation, and I can't seem to find any-one who is having a similar issue. I have talked with Oracle about this issue, and will update this post as information comes about.
I will freely admit that it is very possible this entire post and all of the issues I'm trying to solve, really only exists in my mind. I may be totally wrong here. I really hope I'm not, but then again......
Also, please forgive the lack of color/source/pretty references in this blog.. I am fairly new to blogging, and unfortunately do not have the time to go through and make things pretty. Heck, I barely have enough time to just get the content out here. maybe I'll come back and update this post with the "pretties" in the future.
I am writing an integration webservice which will consume various webservices from a couple different backend systems. I want to be able to parallelize non-dependent service calls and be able to cancel requests that take too long. The need to be able to cancel requests that exceed a specified SLA is paramount, since the requirement is to meet a very hard-sla with either a valid response or an appropriate error message if one or more of our backends are failing to respond or are taking too long. we should definitely NOT hang indefinitely waiting for backend calls.
to accomplish these requirements I am trying to use the Async Polling mode client apis that are defined in the JAX-WS specification (v2.x). I could use the Callback apis, however the orchestration of my services would be overly complex; besides the Polling mode API's seem a better fit for my requirements.
I will freely admit that it is very possible this entire post and all of the issues I'm trying to solve, really only exists in my mind. I may be totally wrong here. I really hope I'm not, but then again......
Also, please forgive the lack of color/source/pretty references in this blog.. I am fairly new to blogging, and unfortunately do not have the time to go through and make things pretty. Heck, I barely have enough time to just get the content out here. maybe I'll come back and update this post with the "pretties" in the future.
I am writing an integration webservice which will consume various webservices from a couple different backend systems. I want to be able to parallelize non-dependent service calls and be able to cancel requests that take too long. The need to be able to cancel requests that exceed a specified SLA is paramount, since the requirement is to meet a very hard-sla with either a valid response or an appropriate error message if one or more of our backends are failing to respond or are taking too long. we should definitely NOT hang indefinitely waiting for backend calls.
to accomplish these requirements I am trying to use the Async Polling mode client apis that are defined in the JAX-WS specification (v2.x). I could use the Callback apis, however the orchestration of my services would be overly complex; besides the Polling mode API's seem a better fit for my requirements.
On the surface the client side API's and runtime appear to work as the specification outlines. Generation of client stubs based on a given WSDL which includes the async as well as the standard sync methods is straight forward. Implementing a sample client which utilizes these sync and async API's is also simple, and if you watch the execution of the test client, the output and behavior seems to work as expected: Sync calls work, async calls work, and in the case of async calls, the inherited functionality provided by the Future<> interface that the Async calls seems to behave exactly like my requirements need.
So, what is the problem I am facing?
The problem is related to the requirements:
1) That I be able to cancel an in progress async request
2) That I be able to cancel a request that has yet to actually execute
I do realize that there are semantic problems with canceling a request that has already been sent to the backend system. Typically you can't necessarily tell the back end system "Sorry, stop what you're doing, I'm tired of waiting now". All you can do is move on, and stop waiting. The backend system would either figure out that you've stopped waiting at some point, or it would remain blissfully ignorant and finish processing your request. This is pretty standard behavior, and is exactly what happens when you configure a response timeout on the underlying Socket.
In the case of an in-progress request, I would expect a cancelation to cause the thread that is stuck waiting for the backend to respond to stop waiting (via an interruption) and the client socket to be closed; as described above the backend system would necessarily not be notified that we've moved on.
Now, if for some reason the request hasn't actually made it to the point where it's been sent to the backend system and we attempt to cancel it, then I would expect that whatever else happens, that request would never be sent to the backend. The request would be treated as a no-op.
The problem I am facing is that none of these expectations turn out to be what happens in reality. in reality the behavior appears to be correct from the client API; but if you watch very carefully, in the background, you will notice that requests you think have been canceled are actually executed/sent to the backend system.
to demonstrate this, as I mentioned earlier, I have written a simple hello-world type JAX-WS based service and client.
You can find the entire project on github.com here:
git://github.com/helterscelter/jaxws_async_client_sample.git
or you can browse the source via your browser here:
http://github.com/helterscelter/jaxws_async_client_sample
All of the code in this example should be built using the included ANT build.xml. the only dependency an installation of Oracle Weblogic v10.3.3
to build the sample, edit the buid.properties file to update the wls.home variable with the location you have installed weblogic server under. Then run ant against the "all" target. this will compile all client side artifacts, and the sample service implementation to test the client against.
The sample Service implementation is a very basic service which takes an arbitrary string as input, and returns that string as part of the output. the service attempts to simulate in a very basic way, a service that may return very quickly or which may return after a relatively long time. (it does this by sleeping a random time between 0 and 10 seconds) other than this very simple logic the service is very much unchanged from the automatically generated service stub that the "wsdlc" ant task generates.
The sample client is also fairly simple.
The general flow is:
- we initiate a sync call, and show the output,
- initiate 10 async calls,
- immediately cancel all 10 of the async calls
- show that the Result object for each async request thinks that it has been canceled
- then we attempt to get a result for each async call
The key here, is that in all cases the perceived output from the above is exactly what the spec says should happen.
see ./sample/output_v1.0.log in the sample project
http://github.com/helterscelter/jaxws_async_client_sample/blob/v1.4.0/sample/output_v1.0.log
I know, I know. So far, I've spent a whole lot of time explaining to you just exactly how the JAX-WS implementation provided by Weblogic does exactly what the spec says it should do.
So you may be asking, What is the problem? Why are you even bothering with this?
The problem starts showing it's self when you look at the error log output from the Weblogic instance that hosts the sample service. If you look there, you'll notice a good number of exceptions which look like this:
POST /helloWorld HTTP/1.1
Content-type: text/xml;charset="utf-8"
Soapaction: "http://www.example.org/HelloWorld/HelloWorld"
Accept: text/xml, multipart/related, text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2
User-Agent: Oracle JAX-WS 2.1.5
Connection: keep-alive
Content-Length: 277
]] Root cause of ServletException.
javax.xml.ws.WebServiceException: com.ctc.wstx.exc.WstxIOException: Connection reset by peer: socket write error
at com.sun.xml.ws.encoding.StreamSOAPCodec.encode(StreamSOAPCodec.java:118)
at com.sun.xml.ws.encoding.SOAPBindingCodec.encode(SOAPBindingCodec.java:258)
at com.sun.xml.ws.transport.http.HttpAdapter.encodePacket(HttpAdapter.java:368)
at com.sun.xml.ws.transport.http.HttpAdapter.access$100(HttpAdapter.java:102)
at com.sun.xml.ws.transport.http.HttpAdapter$HttpToolkit.handle(HttpAdapter.java:534)
Truncated. see log file for complete stacktrace
If the client application is behaving the way we expect it to, then Why do we see these errors in the log? They had to be caused by a request being in progress and then something causing the socket to be closed before the request could be completed. but how can this be when our example client seemingly succeeded in canceling all of our requests, and exited without any unexpected results?
to help figure this out, we enable a java system property in our client application, which causes the JAX-WS client runtime to dump all of it's request and response operations from the wire to stdout:
-Dcom.sun.xml.ws.transport.http.client.HttpTransportPipe.dump="true"
see ./sample/output_v1.1.log
http://github.com/helterscelter/jaxws_async_client_sample/blob/v1.4.0/sample/output_v1.1.log
now when you run the client, you see that the JAX-WS runtime sends all 10 of our Async requests to the backend/sample service. however our client exits before any of these requests can be completed, hence the server reports a bunch of errors when trying to write the reply to the socket.
This still seems like it's expected behavior, doesn't it?
What happens, if instead of exiting immediately after we've validated that all our async requests are canceled, we simulate a longer running process? After all, when this goes into production the backend service calls we're making will be part of a process which is expected to run 24x7. So, lets introduce a 30 second delay at the end of the sample client, and see what happens.
see ./sample/output_v1.2.log
http://github.com/helterscelter/jaxws_async_client_sample/blob/v1.4.0/sample/output_v1.2.log
Interesting, isn't it? Every one of the Async client requests that we thought we canceled actually completes, and a valid response is returned to the client.
it was at this point that I started asking Oracle support questions. initially they came back with the explanation that the behavior I was seeing is caused essentially by a Race condition.
because the first thing my sample client does is to make a sync jax-ws call, the jax-ws runtime takes the opportunity to fully initialize it's infrastructure and open a pool of connections to the backend service. this is intended to make subsequent calls to the service faster because everything it needs to actually make the call would be setup already. Then when I attempt to make my async calls, the async requests are initiated immediately on the already existing connections. once the request actually goes out over the socket, they said that the request couldn't really be canceled.
If I wanted my sample client to behave the way I expect it to behave, I essentially needed to remove the sync call at the beginning of the client, so that when My async calls were made, the Socket connections wouldn't have been setup yet, and the client code would be able to cancel the async requests before the jax-ws runtime could initialize the connections and send the requests to the server.
This seemed like a reasonable thing to try, so I did.
please refer to ./samples/output_v1.3.log
http://github.com/helterscelter/jaxws_async_client_sample/blob/v1.4.0/sample/output_v1.3.log
as you can see, the behavior didn't change much. all of the requests are still sent, and all of the replies are still received.
The final thing I have tried is to attempt and see what is going on under the scenes with the async methods (without digging through tons of source)
to help in figuring out whats going on, I implemented a simple Executor service, which I initialize the HelloWorld_Service() service with. the custom executor basically logs the duration, start and stop time of each Runnable that is passed to it. The executor also utilizes a custom work queue that logs some information about each Runnable that is added or removed from the work queue.
see ./samples/output_v1.4.log
http://github.com/helterscelter/jaxws_async_client_sample/blob/v1.4.0/sample/output_v1.4.log
At this point I have to say something smells foul. I expected to see the com.sun.xml.ws.client.AsyncResponseImpl instances being pushed onto the Executor service's queue. what I did not expect to see was another Async task being pushed into the run queue as a result of the initial AsyncResponseImpl running:
Offer: engine-Oracle JAX-WS 2.1.5: Stub for http://www.example.org/helloWorldfiber--1
java.lang.Thread.getStackTrace(Unknown Source)
test.client.TimingThreadPool$LoggingArrayBlockingQueue.offer(TimingThreadPool.java:92)
java.util.concurrent.ThreadPoolExecutor.execute(Unknown Source)
com.sun.xml.ws.api.pipe.Engine.addRunnable(Engine.java:76)
com.sun.xml.ws.api.pipe.Fiber.start(Fiber.java:262)
com.sun.xml.ws.client.Stub.processAsync(Stub.java:349)
com.sun.xml.ws.client.sei.SEIStub.doProcessAsync(SEIStub.java:156)
com.sun.xml.ws.client.sei.AsyncMethodHandler$SEIAsyncInvoker.do_run(AsyncMethodHandler.java:220)
com.sun.xml.ws.client.AsyncInvoker.run(AsyncInvoker.java:72)
com.sun.xml.ws.client.AsyncResponseImpl.run(AsyncResponseImpl.java:84)
java.util.concurrent.ThreadPoolExecutor$Worker.runTask(Unknown Source)
java.util.concurrent.ThreadPoolExecutor$Worker.run(Unknown Source)
java.lang.Thread.run(Unknown Source)
apparently some part of the JAX-WS runtime actually submits a com.sun.xml.ws.api.pipe.Fiber to the run queue which is what actually does the SOAP request to the backend service!
When we cancel the Result<> object that is returned from the Async invocation, that cancel does not prevent these PIPEs from running on the queue and making the request.
Have I finally succeeded demonstrating the bug I've suspected all along? Does my argument make sense?
Even as I write this, I have to admit that I'm not 100% sure. I suppose my next step would be to dive into the JAX-WS code, and see if I can figure out what it's doing?
I'll keep you posted.
Update: 03/04/2011
I should have updated this entry a good while back. it turns out that I'm not completely crazy. after much conversation with Oracle they have issued patch RHEL against Weblogic 10.3.3 to address this issue..
I should have updated this entry a good while back. it turns out that I'm not completely crazy. after much conversation with Oracle they have issued patch RHEL against Weblogic 10.3.3 to address this issue..
Hello World
A bit underwhelming, but: Hello World! I swore that I would never ever do one of these, and who knows perhaps I'll end like so many others and just revert to a consumerist view of the 'web'.
in any case, I primarily plan to use this place to discuss things I find interesting in software development; I'll probably take a fairly liberal view of what I consider to be "Software Development" so consider yourself warned.
in any case, I primarily plan to use this place to discuss things I find interesting in software development; I'll probably take a fairly liberal view of what I consider to be "Software Development" so consider yourself warned.
Subscribe to:
Posts (Atom)