[icedtea-web] RFC: find jars based on Class-Path entry in manifest

Dr Andrew John Hughes ahughes at redhat.com
Thu Apr 14 17:50:24 PDT 2011


On 19:09 Thu 14 Apr     , Omair Majid wrote:

snip...

> Each String in the Map is a classpath entry. The '+' is used to join the 
> jar's directory with the actual jar name. As an example, consider the 
> applet linked in the original email. The applet's jars are located at 
> /applet/jar/*. The 'Class-Path' entries contain just the names of the 
> jar - like foo.jar, bar.jar. The page (and hence the codebase) is 
> /applet/. If we just joined the name of the jar with the codebase, the 
> result is /applet/foo.jar, which is not the right path. This bit of code 
> above joins the original jar's path to the Class-Path jars to create 
> paths like /applet/jar/foo.jar and /applet/jar/bar.jar.
> 
> On second thought, perhaps I should have documented this a little more. 
> I thought my comment:
> // we want to search for jars in the same subdir on the server
> // as this jar, so find out that subdirectory
> was sufficient, but it may not be.
> 

My main issue was that I wasn't familiar with this jar referencing syntax, but
if you can make the documentation any clearer, that's always good.

> >> >                                if (index != null)
> >> >                                    jarIndexes.add(index);
> >> >                            }
> >> >  @@ -1007,8 +1033,32 @@
> >> >                try {
> >> >                    result = loadClassExt(name);
> >> >                } catch (ClassNotFoundException cnfe) {
> >> >  +                // Not found in external loader either
> >> >
> >> >  -                // Not found in external loader either. As a last resort, look in any available indexes
> >> >  +                // Look in 'Class-Path' as specified in the manifest file
> >> >  +                try {
> >> >  +                    for (String classpath: classpathsInManfest) {
> >> >  +                        JARDesc desc;
> >> >  +                        try {
> >> >  +                            URL jarUrl = new URL(file.getCodeBase(), classpath);
> >> >  +                            desc = new JARDesc(jarUrl, null, null, false, true, false, true);
> >> >  +                        } catch (MalformedURLException mfe) {
> >> >  +                            throw new ClassNotFoundException(name);
> >> >  +                        }
> >> >  +                        try {
> >> >  +                            addNewJar(desc);
> >> >  +                        } catch (Exception e) {
> >> >  +                            throw new ClassNotFoundException(name);
> >> >  +                        }
> >> >  +                    }
> >> >  +
> >> >  +                    result = loadClassExt(name);
> >> >  +                    return result;
> >> >  +                } catch (ClassNotFoundException cnfe1) {
> >> >  +                    // continue below
> >> >  +                }
> >> >  +
> >> >  +                // As a last resort, look in any available indexes
> > Exceptions are getting eaten.  The CNFE should use the thrown exceptions as the initCause.  I'm especially
> > worried about just catching Exception; can we not be more specific?
> >
> 
> I believe we want exceptions to be eaten here. This method, loadClass, 
> can only throw ClassNotFoundExceptions. It has to search in a few 
> locations and attempt a number of things before it realizes that a class 
> can not be found. So the cause that it may throw (and that user code may 
> see) is very likely to be meaningless anyway.
> 

I have no problem with them throwing CNFEs but the cause should be there.
If it's meaningless, it can be ignored, but if it's not there, we may lose
essential debugging information.

> As for catching Exception, you do have a point. I had left it the way it 
> was originally (see the code below). I know that it can throw 
> IllegalArgumentException if the Resource is not being tracked (which 
> should not happen here). I suppose I can remove the catch block completely.
> 

Try it.

> What I want to do, essentially, is to try loading a class from the newly 
> added jars, but map any Exceptions thrown to ClassNotFoundException, 
> which I ignore. This ensures that nothing here stops the 
> JNLPClassLoader. The JNLPClassLoader can then try other methods for 
> loading the class.
> 
> >> >
> >> >                    // Currently this loads jars directly from the site. We cannot cache it because this
> >> >                    // call is initiated from within the applet, which does not have disk read/write permissions
> >> >  @@ -1026,33 +1076,11 @@
> >> >                                } catch (MalformedURLException mfe) {
> >> >                                    throw new ClassNotFoundException(name);
> >> >                                }
> >> >  -
> >> >  -                            available.add(desc);
> >> >  -
> >> >  -                            tracker.addResource(desc.getLocation(),
> >> >  -                                    desc.getVersion(),
> >> >  -                                    null,
> >> >  -                                    JNLPRuntime.getDefaultUpdatePolicy()
> >> >  -                                    );
> >> >  -
> >> >  -                            URL remoteURL;
> >> >                                try {
> >> >  -                                remoteURL = new URL(file.getCodeBase() + jarName);
> >> >  -                            } catch (MalformedURLException mfe) {
> >> >  -                                throw new ClassNotFoundException(name);
> >> >  -                            }
> >> >  -
> >> >  -                            URL u;
> >> >  -
> >> >  -                            try {
> >> >  -                                u = tracker.getCacheURL(remoteURL);
> >> >  +                                addNewJar(desc);
> >> >                                } catch (Exception e) {
> >> >                                    throw new ClassNotFoundException(name);
> >> >                                }
> >> >  -
> >> >  -                            if (u != null)
> >> >  -                                addURL(u);
> >> >  -
> > What's happening here?
> 
> No functional changes. This bit of code here would have been duplicated 
> in the part that downloads jars from the Class-Path. To eliminate the 
> duplication, I moved it to the new method addNewJar and made both places 
> call that.
> 
> Thanks,
> Omair

-- 
Andrew :)

Free Java Software Engineer
Red Hat, Inc. (http://www.redhat.com)

Support Free Java!
Contribute to GNU Classpath and IcedTea
http://www.gnu.org/software/classpath
http://icedtea.classpath.org
PGP Key: F5862A37 (https://keys.indymedia.org/)
Fingerprint = EA30 D855 D50F 90CD F54D  0698 0713 C3ED F586 2A37



More information about the distro-pkg-dev mailing list