Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

MockServletContext should treat InvalidPathException like an IOException #23717

Closed
Spunc opened this issue Sep 25, 2019 · 6 comments
Closed

MockServletContext should treat InvalidPathException like an IOException #23717

Spunc opened this issue Sep 25, 2019 · 6 comments
Assignees
Labels
in: test Issues in the test module status: backported An issue that has been backported to maintenance branches type: bug A general bug
Milestone

Comments

@Spunc
Copy link

Spunc commented Sep 25, 2019

I have a simple ConfigurationProperties class that binds a String property to a Path object:

@Component
@ConfigurationProperties("app")
@Getter
@Setter
public class AppConfigurationProperties {
	
	private Path somePath;
}

Then I define app.somePath=C:\\temp within application.properties. This works when running the application, but a simple Test that loads the context throws:

com.example.demo.DemoApplicationTests > contextLoads() FAILED
    java.lang.IllegalStateException
        Caused by: org.springframework.boot.context.properties.ConfigurationPropertiesBindException
            Caused by: org.springframework.boot.context.properties.bind.BindException
                Caused by: org.springframework.core.convert.ConverterNotFoundException

The test class is most simple:

@SpringBootTest
public class DemoApplicationTests {

	@Test
	public void contextLoads() {
	}
}

I created a project with the most recent Spring-Boot version (2.1.8.RELEASE) with Initializr (Gradle build with Spring Web as only dependency) to replicate the problem in isolation. It can be found here. I run the project on Windows 10.

I found out that the test passes, if there are no backslashes in the property value. Furthermore, it passes if I don't use org.springframework.boot:spring-boot-starter-web but org.springframework.boot:spring-boot-starter as dependency. (The test project does not require web, but the actual project does.)

@mbhave
Copy link
Contributor

mbhave commented Sep 25, 2019

Thanks for the sample @Spunc. I ran DemoApplicationTests without changing anything and it was green. Is there something additional that needs to be done to reproduce the failure?

@wilkinsona
Copy link
Member

It looks like running on Windows is significant. The test fails for me there. We end up with the following:

  1. PathEditor setAsText(String) is called with C:\temp
  2. ResourceEditor setAsText(String) is called with C:\temp
  3. GenericWebApplicationContext (the ResourceLoader) getResource(String) is called with C:\temp
  4. A ServletContextResource with the path /C:/temp is created
  5. The stack unwinds into PathEditor and it calls exists() on the resource
  6. SpringBootMockServletContext getResource is called with /C:/temp
  7. MockServletContext getResource(String) is called with /C:/temp
  8. SpringBootMockServletContext getResourceLocation(String) is called with /C:/temp
  9. MockServletContext getResourceLocation(String) is called with /C:/temp and it prepends the base path and returns src/main/webapp/C:/temp
  10. FileSystemResourceLoader getResource(String) is called with src/main/webapp/C:/temp
  11. FileSystemResourceLoader getResourceByPath(String) is called with src/main/webapp/C:/temp
  12. A FileSystemResource is constructed with src/main/webapp/C:/temp. It creates a File and then calls toPath() which fails with an InvalidPathException

The Windows-specific behaviour is in step 12. The InvalidPathException is not thrown on macOS where a UnixPath is successfully created and returned.

This feels like a bug in MockServletContext getResource(String) to me (which is why the problem only occurs in the tests of a web application). The contract allows the method to throw a MalformedURLException or to return null. The implementation catches IOException and returns null. I think this catch needs to be broadened to map other failures, such as InvalidPathException, to null as well.

@Spunc You can work around the problem by using file:/C:/temp as the value of your app.somePath property.

@snicoll snicoll transferred this issue from spring-projects/spring-boot Sep 26, 2019
@spring-projects-issues spring-projects-issues added the status: waiting-for-triage An issue we've not yet triaged or decided on label Sep 26, 2019
@Spunc
Copy link
Author

Spunc commented Sep 26, 2019

Thank you very much for your replies. At first, I was surprised that @mbhave could not reproduce the problem. I had colleagues verify the issue and indeed, those with Windows have the problem while those with Mac-OS are not affected.

Thanks, @wilkinsona for your suggestion of a workaround. We will apply that.

@sbrannen
Copy link
Member

Thanks for the great detective work, @wilkinsona!

We'll look into it.

@sbrannen sbrannen added in: test Issues in the test module type: bug A general bug and removed status: waiting-for-triage An issue we've not yet triaged or decided on labels Sep 26, 2019
@sbrannen sbrannen added this to the 5.2.1 milestone Sep 26, 2019
@sbrannen sbrannen self-assigned this Oct 29, 2019
@spring-projects-issues spring-projects-issues added status: backported An issue that has been backported to maintenance branches and removed for: backport-to-5.1.x labels Oct 29, 2019
@sbrannen sbrannen changed the title ConfigurationProperties: ConverterNotFoundException in Test only MockServletContext should treat InvalidPathException like an IOException Oct 30, 2019
sbrannen added a commit that referenced this issue Oct 30, 2019
Prior to this commit, if MockServletContext was configured with a
FileSystemResourceLoader, invocations of the following methods on a
Microsoft Windows operating system resulted in an InvalidPathException
if the supplied path contained a colon (such as "C:\\temp"). This is
inconsistent with the behavior on non-Windows operating systems. In
addition, for comparable errors resulting in an IOException, those
methods (except getRealPath()) return null instead of throwing the
exception.

- getResourcePaths()
- getResource()
- getResourceAsStream()
- getRealPath()

This commit makes handling of InvalidPathException and IOException
consistent for these methods: both exceptions now result in null be
returned by these methods.

Closes gh-23717
pull bot pushed a commit to scope-demo/spring-framework that referenced this issue Oct 30, 2019
Prior to this commit, if MockServletContext was configured with a
FileSystemResourceLoader, invocations of the following methods on a
Microsoft Windows operating system resulted in an InvalidPathException
if the supplied path contained a colon (such as "C:\\temp"). This is
inconsistent with the behavior on non-Windows operating systems. In
addition, for comparable errors resulting in an IOException, those
methods (except getRealPath()) return null instead of throwing the
exception.

- getResourcePaths()
- getResource()
- getResourceAsStream()
- getRealPath()

This commit makes handling of InvalidPathException and IOException
consistent for these methods: both exceptions now result in null be
returned by these methods.

Closes spring-projectsgh-23717
@xak2000
Copy link
Contributor

xak2000 commented Nov 13, 2019

I also run into this problem with spring 5.1.5. Now I updated to 5.1.11 and the problem "solved", but now it prints warning to the log.

@sbrannen @wilkinsona
I don't understand why org.springframework.mock.web.MockServletContext.getResourceLocation(String path) should prepend resourceBasePath to an absolute Windows path like C:\temp? So the result path is src/main/webapp/C:/temp. This would never work anyway.

If we leave /C:/temp here, then this.file.toPath() would not throw. It creates WindowsPath("C:\temp") instance.

Maybe this is incorrect to leave it like this because this is servlet context anyway (I'm not sure here what's the intention). But it is also have no sense to just do a concatenation like this. Maybe it works on unix systems, where any absolute path can be converted to relative just by appending another path at the beginning, but it doesn't work on Windows, where path can contain a hard drive letter.

If the path must really be prepended with servlet context path here, maybe it is better to remove the letter from path before doing this? So /C:/temp will become /temp and then src/main/webapp/temp. I can't reason about this because to be honest I don't fully understand this code and why we need to manipulate paths from application.properties this way in tests. If I created a property like my-location=C:\temp and run test with it, I don't expect it to be manipulated in some ways to be changed to point to some another path in a servlet context. Maybe for relative paths, but not for absolute.

P.S. In my case a property was just like this:

my-prop=${java.io.tmpdir}/some/path

This was intentionally done in test properties to make sure the test runs on Windows and Unix machines without changes. And it worked while I has custom @ConfigurationPropertiesBinding Converter bean in my context that convert String to Path by just doing Paths.get(string). Then I removed this custom converter in favor of standard SimpleTypeConverter with PathEditor. And then run into this issue.

@sbrannen
Copy link
Member

@xak2000, since this issue has already been closed and the change has already been released, would you please open a new ticket to address your concerns?

Thanks in advance!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
in: test Issues in the test module status: backported An issue that has been backported to maintenance branches type: bug A general bug
Projects
None yet
Development

No branches or pull requests

6 participants