001/* 002 * Copyright (C) 2022 - 2025, the original author or authors. 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016package io.github.ascopes.jct.assertions; 017 018import static io.github.ascopes.jct.utils.IoExceptionUtils.uncheckedIo; 019import static java.util.Objects.requireNonNull; 020import static org.assertj.core.api.Assertions.assertThat; 021 022import java.io.ByteArrayOutputStream; 023import java.io.UncheckedIOException; 024import java.nio.ByteBuffer; 025import java.nio.charset.Charset; 026import java.nio.charset.CharsetDecoder; 027import java.nio.charset.StandardCharsets; 028import java.time.Instant; 029import javax.tools.JavaFileObject; 030import org.assertj.core.api.AbstractAssert; 031import org.assertj.core.api.AbstractByteArrayAssert; 032import org.assertj.core.api.AbstractInstantAssert; 033import org.assertj.core.api.AbstractStringAssert; 034import org.assertj.core.api.AbstractUriAssert; 035import org.jspecify.annotations.Nullable; 036 037/** 038 * Abstract assertions for {@link JavaFileObject Java file objects}. 039 * 040 * @param <I> the implementation class that is extending this class. 041 * @param <A> the file object implementation type. 042 * @author Ashley Scopes 043 * @since 0.0.1 044 */ 045public abstract class AbstractJavaFileObjectAssert<I extends AbstractJavaFileObjectAssert<I, A>, A extends JavaFileObject> 046 extends AbstractAssert<I, A> { 047 048 /** 049 * Initialize this assertion. 050 * 051 * @param actual the actual value to assert on. 052 * @param selfType the type of the assertion implementation. 053 */ 054 @SuppressWarnings("DataFlowIssue") 055 protected AbstractJavaFileObjectAssert(@Nullable A actual, Class<?> selfType) { 056 super(actual, selfType); 057 } 058 059 /** 060 * Get an assertion object on the URI of the file. 061 * 062 * @return the URI assertion. 063 * @throws AssertionError if the actual value is null. 064 */ 065 public AbstractUriAssert<?> uri() { 066 isNotNull(); 067 return assertThat(actual.toUri()); 068 } 069 070 /** 071 * Get an assertion object on the name of the file. 072 * 073 * @return the string assertion. 074 * @throws AssertionError if the actual value is null. 075 */ 076 public AbstractStringAssert<?> name() { 077 isNotNull(); 078 return assertThat(actual.getName()); 079 } 080 081 /** 082 * Get an assertion object on the binary content of the file. 083 * 084 * @return the byte array assertion. 085 * @throws AssertionError if the actual value is null. 086 */ 087 public AbstractByteArrayAssert<?> binaryContent() { 088 isNotNull(); 089 return assertThat(rawContent()); 090 } 091 092 /** 093 * Get an assertion object on the content of the file, using {@link StandardCharsets#UTF_8 UTF-8} 094 * encoding. 095 * 096 * @return the string assertion. 097 * @throws AssertionError if the actual value is null. 098 * @throws UncheckedIOException if an IO error occurs reading the file content. 099 */ 100 public AbstractStringAssert<?> content() { 101 return content(StandardCharsets.UTF_8); 102 } 103 104 /** 105 * Get an assertion object on the content of the file. 106 * 107 * @param charset the charset to decode the file with. 108 * @return the string assertion. 109 * @throws AssertionError if the actual value is null. 110 * @throws NullPointerException if the charset parameter is null. 111 * @throws UncheckedIOException if an IO error occurs reading the file content. 112 */ 113 public AbstractStringAssert<?> content(Charset charset) { 114 requireNonNull(charset, "charset must not be null"); 115 return content(charset.newDecoder()); 116 } 117 118 /** 119 * Get an assertion object on the content of the file. 120 * 121 * @param charsetDecoder the charset decoder to use to decode the file to a string. 122 * @return the string assertion. 123 * @throws AssertionError if the actual value is null. 124 * @throws NullPointerException if the charset decoder parameter is null. 125 * @throws UncheckedIOException if an IO error occurs reading the file content. 126 */ 127 public AbstractStringAssert<?> content(CharsetDecoder charsetDecoder) { 128 requireNonNull(charsetDecoder, "charsetDecoder must not be null"); 129 isNotNull(); 130 131 var content = uncheckedIo(() -> charsetDecoder 132 .decode(ByteBuffer.wrap(rawContent())) 133 .toString()); 134 135 return assertThat(content); 136 } 137 138 /** 139 * Get an assertion object on the last modified timestamp. 140 * 141 * <p>This will be set to the UNIX epoch ({@code 1970-01-01T00:00:00.000Z}) if an 142 * error occurs reading the file modification time, or if the information is not available. 143 * 144 * @return the instant assertion. 145 * @throws AssertionError if the actual value is null. 146 */ 147 public AbstractInstantAssert<?> lastModified() { 148 isNotNull(); 149 150 var instant = Instant.ofEpochMilli(actual.getLastModified()); 151 return assertThat(instant); 152 } 153 154 /** 155 * Perform an assertion on the file object kind. 156 * 157 * @return the assertions for the kind. 158 * @throws AssertionError if the actual value is null. 159 */ 160 public JavaFileObjectKindAssert kind() { 161 isNotNull(); 162 163 return new JavaFileObjectKindAssert(actual.getKind()); 164 } 165 166 private byte[] rawContent() { 167 return uncheckedIo(() -> { 168 var outputStream = new ByteArrayOutputStream(); 169 try (var inputStream = actual.openInputStream()) { 170 inputStream.transferTo(outputStream); 171 return outputStream.toByteArray(); 172 } 173 }); 174 } 175}