mirror of
https://github.com/samsonjs/media.git
synced 2026-04-27 15:07:40 +00:00
Tighten DataSource contract test assertions
Assert that opening the DataSource at the end of the resource results in only RESULT_END_OF_INPUT being read. open() and read() are still permitted to throw, although this permissiveness will be removed in subsequent commits. PiperOrigin-RevId: 362905314
This commit is contained in:
parent
6f8a8fbc1c
commit
10de7b2ab0
9 changed files with 146 additions and 95 deletions
|
|
@ -72,48 +72,61 @@ public final class ContentDataSource extends BaseDataSource {
|
||||||
if (assetFileDescriptor == null) {
|
if (assetFileDescriptor == null) {
|
||||||
throw new FileNotFoundException("Could not open file descriptor for: " + uri);
|
throw new FileNotFoundException("Could not open file descriptor for: " + uri);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
long assetFileDescriptorLength = assetFileDescriptor.getLength();
|
||||||
FileInputStream inputStream = new FileInputStream(assetFileDescriptor.getFileDescriptor());
|
FileInputStream inputStream = new FileInputStream(assetFileDescriptor.getFileDescriptor());
|
||||||
this.inputStream = inputStream;
|
this.inputStream = inputStream;
|
||||||
|
|
||||||
long assetStartOffset = assetFileDescriptor.getStartOffset();
|
// We can't rely only on the "skipped < dataSpec.position" check below to detect whether the
|
||||||
long skipped = inputStream.skip(assetStartOffset + dataSpec.position) - assetStartOffset;
|
// position is beyond the end of the asset being read. This is because the file may contain
|
||||||
if (skipped != dataSpec.position) {
|
// multiple assets, and there's nothing to prevent InputStream.skip() from succeeding by
|
||||||
// We expect the skip to be satisfied in full. If it isn't then we're probably trying to
|
// skipping into the data of the next asset. Hence we also need to check against the asset
|
||||||
// skip beyond the end of the data.
|
// length explicitly, which is guaranteed to be set unless the asset extends to the end of the
|
||||||
|
// file.
|
||||||
|
if (assetFileDescriptorLength != AssetFileDescriptor.UNKNOWN_LENGTH
|
||||||
|
&& dataSpec.position > assetFileDescriptorLength) {
|
||||||
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
}
|
}
|
||||||
if (dataSpec.length != C.LENGTH_UNSET) {
|
long assetFileDescriptorOffset = assetFileDescriptor.getStartOffset();
|
||||||
bytesRemaining = dataSpec.length;
|
long skipped =
|
||||||
} else {
|
inputStream.skip(assetFileDescriptorOffset + dataSpec.position)
|
||||||
long assetFileDescriptorLength = assetFileDescriptor.getLength();
|
- assetFileDescriptorOffset;
|
||||||
if (assetFileDescriptorLength == AssetFileDescriptor.UNKNOWN_LENGTH) {
|
if (skipped != dataSpec.position) {
|
||||||
// The asset must extend to the end of the file. If FileInputStream.getChannel().size()
|
// We expect the skip to be satisfied in full. If it isn't then we're probably trying to
|
||||||
// returns 0 then the remaining length cannot be determined.
|
// read beyond the end of the last resource in the file.
|
||||||
FileChannel channel = inputStream.getChannel();
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
long channelSize = channel.size();
|
}
|
||||||
if (channelSize == 0) {
|
if (assetFileDescriptorLength == AssetFileDescriptor.UNKNOWN_LENGTH) {
|
||||||
bytesRemaining = C.LENGTH_UNSET;
|
// The asset must extend to the end of the file. We can try and resolve the length with
|
||||||
} else {
|
// FileInputStream.getChannel().size().
|
||||||
bytesRemaining = channelSize - channel.position();
|
FileChannel channel = inputStream.getChannel();
|
||||||
if (bytesRemaining < 0) {
|
long channelSize = channel.size();
|
||||||
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
if (channelSize == 0) {
|
||||||
}
|
bytesRemaining = C.LENGTH_UNSET;
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
bytesRemaining = assetFileDescriptorLength - skipped;
|
bytesRemaining = channelSize - channel.position();
|
||||||
if (bytesRemaining < 0) {
|
if (bytesRemaining < 0) {
|
||||||
|
// The skip above was satisfied in full, but skipped beyond the end of the file.
|
||||||
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
bytesRemaining = assetFileDescriptorLength - skipped;
|
||||||
|
if (bytesRemaining < 0) {
|
||||||
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new ContentDataSourceException(e);
|
throw new ContentDataSourceException(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (dataSpec.length != C.LENGTH_UNSET) {
|
||||||
|
bytesRemaining =
|
||||||
|
bytesRemaining == C.LENGTH_UNSET ? dataSpec.length : min(bytesRemaining, dataSpec.length);
|
||||||
|
}
|
||||||
opened = true;
|
opened = true;
|
||||||
transferStarted(dataSpec);
|
transferStarted(dataSpec);
|
||||||
|
return dataSpec.length != C.LENGTH_UNSET ? dataSpec.length : bytesRemaining;
|
||||||
return bytesRemaining;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
||||||
|
|
@ -31,6 +31,7 @@ import java.io.EOFException;
|
||||||
import java.io.FileInputStream;
|
import java.io.FileInputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InputStream;
|
import java.io.InputStream;
|
||||||
|
import java.nio.channels.FileChannel;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A {@link DataSource} for reading a raw resource inside the APK.
|
* A {@link DataSource} for reading a raw resource inside the APK.
|
||||||
|
|
@ -149,6 +150,7 @@ public final class RawResourceDataSource extends BaseDataSource {
|
||||||
long assetFileDescriptorLength = assetFileDescriptor.getLength();
|
long assetFileDescriptorLength = assetFileDescriptor.getLength();
|
||||||
FileInputStream inputStream = new FileInputStream(assetFileDescriptor.getFileDescriptor());
|
FileInputStream inputStream = new FileInputStream(assetFileDescriptor.getFileDescriptor());
|
||||||
this.inputStream = inputStream;
|
this.inputStream = inputStream;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// We can't rely only on the "skipped < dataSpec.position" check below to detect whether the
|
// We can't rely only on the "skipped < dataSpec.position" check below to detect whether the
|
||||||
// position is beyond the end of the resource being read. This is because the file will
|
// position is beyond the end of the resource being read. This is because the file will
|
||||||
|
|
@ -160,31 +162,45 @@ public final class RawResourceDataSource extends BaseDataSource {
|
||||||
&& dataSpec.position > assetFileDescriptorLength) {
|
&& dataSpec.position > assetFileDescriptorLength) {
|
||||||
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
}
|
}
|
||||||
inputStream.skip(assetFileDescriptor.getStartOffset());
|
long assetFileDescriptorOffset = assetFileDescriptor.getStartOffset();
|
||||||
long skipped = inputStream.skip(dataSpec.position);
|
long skipped =
|
||||||
if (skipped < dataSpec.position) {
|
inputStream.skip(assetFileDescriptorOffset + dataSpec.position)
|
||||||
|
- assetFileDescriptorOffset;
|
||||||
|
if (skipped != dataSpec.position) {
|
||||||
// We expect the skip to be satisfied in full. If it isn't then we're probably trying to
|
// We expect the skip to be satisfied in full. If it isn't then we're probably trying to
|
||||||
// read beyond the end of the last resource in the file.
|
// read beyond the end of the last resource in the file.
|
||||||
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
}
|
}
|
||||||
|
if (assetFileDescriptorLength == AssetFileDescriptor.UNKNOWN_LENGTH) {
|
||||||
|
// The asset must extend to the end of the file. We can try and resolve the length with
|
||||||
|
// FileInputStream.getChannel().size().
|
||||||
|
FileChannel channel = inputStream.getChannel();
|
||||||
|
if (channel.size() == 0) {
|
||||||
|
bytesRemaining = C.LENGTH_UNSET;
|
||||||
|
} else {
|
||||||
|
bytesRemaining = channel.size() - channel.position();
|
||||||
|
if (bytesRemaining < 0) {
|
||||||
|
// The skip above was satisfied in full, but skipped beyond the end of the file.
|
||||||
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
bytesRemaining = assetFileDescriptorLength - skipped;
|
||||||
|
if (bytesRemaining < 0) {
|
||||||
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
|
}
|
||||||
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new RawResourceDataSourceException(e);
|
throw new RawResourceDataSourceException(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dataSpec.length != C.LENGTH_UNSET) {
|
if (dataSpec.length != C.LENGTH_UNSET) {
|
||||||
bytesRemaining = dataSpec.length;
|
|
||||||
} else {
|
|
||||||
// If the length is UNKNOWN_LENGTH then the asset extends to the end of the file.
|
|
||||||
bytesRemaining =
|
bytesRemaining =
|
||||||
assetFileDescriptorLength == AssetFileDescriptor.UNKNOWN_LENGTH
|
bytesRemaining == C.LENGTH_UNSET ? dataSpec.length : min(bytesRemaining, dataSpec.length);
|
||||||
? C.LENGTH_UNSET
|
|
||||||
: (assetFileDescriptorLength - dataSpec.position);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
opened = true;
|
opened = true;
|
||||||
transferStarted(dataSpec);
|
transferStarted(dataSpec);
|
||||||
|
return dataSpec.length != C.LENGTH_UNSET ? dataSpec.length : bytesRemaining;
|
||||||
return bytesRemaining;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
||||||
|
|
@ -388,8 +388,9 @@ public final class CacheDataSource implements DataSource {
|
||||||
|
|
||||||
@Nullable private Uri actualUri;
|
@Nullable private Uri actualUri;
|
||||||
@Nullable private DataSpec requestDataSpec;
|
@Nullable private DataSpec requestDataSpec;
|
||||||
|
@Nullable private DataSpec currentDataSpec;
|
||||||
@Nullable private DataSource currentDataSource;
|
@Nullable private DataSource currentDataSource;
|
||||||
private boolean currentDataSpecLengthUnset;
|
private long currentDataSourceBytesRead;
|
||||||
private long readPosition;
|
private long readPosition;
|
||||||
private long bytesRemaining;
|
private long bytesRemaining;
|
||||||
@Nullable private CacheSpan currentHoleSpan;
|
@Nullable private CacheSpan currentHoleSpan;
|
||||||
|
|
@ -565,19 +566,27 @@ public final class CacheDataSource implements DataSource {
|
||||||
notifyCacheIgnored(reason);
|
notifyCacheIgnored(reason);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dataSpec.length != C.LENGTH_UNSET || currentRequestIgnoresCache) {
|
if (currentRequestIgnoresCache) {
|
||||||
bytesRemaining = dataSpec.length;
|
bytesRemaining = C.LENGTH_UNSET;
|
||||||
} else {
|
} else {
|
||||||
bytesRemaining = ContentMetadata.getContentLength(cache.getContentMetadata(key));
|
bytesRemaining = ContentMetadata.getContentLength(cache.getContentMetadata(key));
|
||||||
if (bytesRemaining != C.LENGTH_UNSET) {
|
if (bytesRemaining != C.LENGTH_UNSET) {
|
||||||
bytesRemaining -= dataSpec.position;
|
bytesRemaining -= dataSpec.position;
|
||||||
if (bytesRemaining <= 0) {
|
if (bytesRemaining < 0) {
|
||||||
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
openNextSource(requestDataSpec, false);
|
if (dataSpec.length != C.LENGTH_UNSET) {
|
||||||
return bytesRemaining;
|
bytesRemaining =
|
||||||
|
bytesRemaining == C.LENGTH_UNSET
|
||||||
|
? dataSpec.length
|
||||||
|
: min(bytesRemaining, dataSpec.length);
|
||||||
|
}
|
||||||
|
if (bytesRemaining > 0 || bytesRemaining == C.LENGTH_UNSET) {
|
||||||
|
openNextSource(requestDataSpec, false);
|
||||||
|
}
|
||||||
|
return dataSpec.length != C.LENGTH_UNSET ? dataSpec.length : bytesRemaining;
|
||||||
} catch (Throwable e) {
|
} catch (Throwable e) {
|
||||||
handleBeforeThrow(e);
|
handleBeforeThrow(e);
|
||||||
throw e;
|
throw e;
|
||||||
|
|
@ -587,6 +596,7 @@ public final class CacheDataSource implements DataSource {
|
||||||
@Override
|
@Override
|
||||||
public int read(byte[] buffer, int offset, int readLength) throws IOException {
|
public int read(byte[] buffer, int offset, int readLength) throws IOException {
|
||||||
DataSpec requestDataSpec = checkNotNull(this.requestDataSpec);
|
DataSpec requestDataSpec = checkNotNull(this.requestDataSpec);
|
||||||
|
DataSpec currentDataSpec = checkNotNull(this.currentDataSpec);
|
||||||
if (readLength == 0) {
|
if (readLength == 0) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
@ -603,10 +613,16 @@ public final class CacheDataSource implements DataSource {
|
||||||
totalCachedBytesRead += bytesRead;
|
totalCachedBytesRead += bytesRead;
|
||||||
}
|
}
|
||||||
readPosition += bytesRead;
|
readPosition += bytesRead;
|
||||||
|
currentDataSourceBytesRead += bytesRead;
|
||||||
if (bytesRemaining != C.LENGTH_UNSET) {
|
if (bytesRemaining != C.LENGTH_UNSET) {
|
||||||
bytesRemaining -= bytesRead;
|
bytesRemaining -= bytesRead;
|
||||||
}
|
}
|
||||||
} else if (currentDataSpecLengthUnset) {
|
} else if (isReadingFromUpstream()
|
||||||
|
&& (currentDataSpec.length == C.LENGTH_UNSET
|
||||||
|
|| currentDataSourceBytesRead < currentDataSpec.length)) {
|
||||||
|
// We've encountered RESULT_END_OF_INPUT from the upstream DataSource at a position not
|
||||||
|
// imposed by the current DataSpec. This must mean that we've reached the end of the
|
||||||
|
// resource.
|
||||||
setNoBytesRemainingAndMaybeStoreLength(castNonNull(requestDataSpec.key));
|
setNoBytesRemainingAndMaybeStoreLength(castNonNull(requestDataSpec.key));
|
||||||
} else if (bytesRemaining > 0 || bytesRemaining == C.LENGTH_UNSET) {
|
} else if (bytesRemaining > 0 || bytesRemaining == C.LENGTH_UNSET) {
|
||||||
closeCurrentSource();
|
closeCurrentSource();
|
||||||
|
|
@ -615,7 +631,16 @@ public final class CacheDataSource implements DataSource {
|
||||||
}
|
}
|
||||||
return bytesRead;
|
return bytesRead;
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
if (currentDataSpecLengthUnset && DataSourceException.isCausedByPositionOutOfRange(e)) {
|
// TODO: This is not correct, because position-out-of-range exceptions should only be thrown
|
||||||
|
// if the requested position is more than one byte beyond the end of the resource. Conversely,
|
||||||
|
// this code is assuming that a position-out-of-range exception indicates the requested
|
||||||
|
// position is exactly one byte beyond the end of the resource, which is not a case for which
|
||||||
|
// this type of exception should be thrown. This exception handling may be required for
|
||||||
|
// interop with current HttpDataSource implementations that do (incorrectly) throw a
|
||||||
|
// position-out-of-range exception at this position. It should be removed when the
|
||||||
|
// HttpDataSource implementations are fixed.
|
||||||
|
if (currentDataSpec.length == C.LENGTH_UNSET
|
||||||
|
&& DataSourceException.isCausedByPositionOutOfRange(e)) {
|
||||||
setNoBytesRemainingAndMaybeStoreLength(castNonNull(requestDataSpec.key));
|
setNoBytesRemainingAndMaybeStoreLength(castNonNull(requestDataSpec.key));
|
||||||
return C.RESULT_END_OF_INPUT;
|
return C.RESULT_END_OF_INPUT;
|
||||||
}
|
}
|
||||||
|
|
@ -760,12 +785,13 @@ public final class CacheDataSource implements DataSource {
|
||||||
currentHoleSpan = nextSpan;
|
currentHoleSpan = nextSpan;
|
||||||
}
|
}
|
||||||
currentDataSource = nextDataSource;
|
currentDataSource = nextDataSource;
|
||||||
currentDataSpecLengthUnset = nextDataSpec.length == C.LENGTH_UNSET;
|
currentDataSpec = nextDataSpec;
|
||||||
|
currentDataSourceBytesRead = 0;
|
||||||
long resolvedLength = nextDataSource.open(nextDataSpec);
|
long resolvedLength = nextDataSource.open(nextDataSpec);
|
||||||
|
|
||||||
// Update bytesRemaining, actualUri and (if writing to cache) the cache metadata.
|
// Update bytesRemaining, actualUri and (if writing to cache) the cache metadata.
|
||||||
ContentMetadataMutations mutations = new ContentMetadataMutations();
|
ContentMetadataMutations mutations = new ContentMetadataMutations();
|
||||||
if (currentDataSpecLengthUnset && resolvedLength != C.LENGTH_UNSET) {
|
if (nextDataSpec.length == C.LENGTH_UNSET && resolvedLength != C.LENGTH_UNSET) {
|
||||||
bytesRemaining = resolvedLength;
|
bytesRemaining = resolvedLength;
|
||||||
ContentMetadataMutations.setContentLength(mutations, readPosition + bytesRemaining);
|
ContentMetadataMutations.setContentLength(mutations, readPosition + bytesRemaining);
|
||||||
}
|
}
|
||||||
|
|
@ -816,8 +842,8 @@ public final class CacheDataSource implements DataSource {
|
||||||
try {
|
try {
|
||||||
currentDataSource.close();
|
currentDataSource.close();
|
||||||
} finally {
|
} finally {
|
||||||
|
currentDataSpec = null;
|
||||||
currentDataSource = null;
|
currentDataSource = null;
|
||||||
currentDataSpecLengthUnset = false;
|
|
||||||
if (currentHoleSpan != null) {
|
if (currentHoleSpan != null) {
|
||||||
cache.releaseHoleSpan(currentHoleSpan);
|
cache.releaseHoleSpan(currentHoleSpan);
|
||||||
currentHoleSpan = null;
|
currentHoleSpan = null;
|
||||||
|
|
|
||||||
|
|
@ -23,6 +23,7 @@ import com.google.android.exoplayer2.upstream.DataSpec;
|
||||||
import com.google.android.exoplayer2.util.PriorityTaskManager;
|
import com.google.android.exoplayer2.util.PriorityTaskManager;
|
||||||
import com.google.android.exoplayer2.util.PriorityTaskManager.PriorityTooLowException;
|
import com.google.android.exoplayer2.util.PriorityTaskManager.PriorityTooLowException;
|
||||||
import com.google.android.exoplayer2.util.Util;
|
import com.google.android.exoplayer2.util.Util;
|
||||||
|
import java.io.EOFException;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.InterruptedIOException;
|
import java.io.InterruptedIOException;
|
||||||
|
|
||||||
|
|
@ -142,6 +143,14 @@ public final class CacheWriter {
|
||||||
nextPosition += readBlockToCache(nextPosition, nextRequestLength);
|
nextPosition += readBlockToCache(nextPosition, nextRequestLength);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: Remove allowShortContent parameter, this code block, and return the number of bytes
|
||||||
|
// cached. The caller can then check whether fewer bytes were cached than were requested.
|
||||||
|
if (!allowShortContent
|
||||||
|
&& dataSpec.length != C.LENGTH_UNSET
|
||||||
|
&& nextPosition != dataSpec.position + dataSpec.length) {
|
||||||
|
throw new EOFException();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
||||||
|
|
@ -154,7 +154,7 @@ public final class CacheDataSourceTest {
|
||||||
try {
|
try {
|
||||||
cacheDataSource =
|
cacheDataSource =
|
||||||
createCacheDataSource(/* setReadException= */ false, /* unknownLength= */ false);
|
createCacheDataSource(/* setReadException= */ false, /* unknownLength= */ false);
|
||||||
cacheDataSource.open(buildDataSpec(TEST_DATA.length, /* length= */ 1, defaultCacheKey));
|
cacheDataSource.open(buildDataSpec(TEST_DATA.length + 1, /* length= */ 1, defaultCacheKey));
|
||||||
fail();
|
fail();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// Expected.
|
// Expected.
|
||||||
|
|
@ -204,7 +204,7 @@ public final class CacheDataSourceTest {
|
||||||
cacheDataSource =
|
cacheDataSource =
|
||||||
createCacheDataSource(
|
createCacheDataSource(
|
||||||
/* setReadException= */ false, /* unknownLength= */ false, cacheKeyFactory);
|
/* setReadException= */ false, /* unknownLength= */ false, cacheKeyFactory);
|
||||||
cacheDataSource.open(buildDataSpec(TEST_DATA.length, /* length= */ 1, customCacheKey));
|
cacheDataSource.open(buildDataSpec(TEST_DATA.length + 1, /* length= */ 1, customCacheKey));
|
||||||
fail();
|
fail();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// Expected.
|
// Expected.
|
||||||
|
|
@ -257,7 +257,7 @@ public final class CacheDataSourceTest {
|
||||||
cacheDataSource =
|
cacheDataSource =
|
||||||
createCacheDataSource(
|
createCacheDataSource(
|
||||||
/* setReadException= */ false, /* unknownLength= */ false, cacheKeyFactory);
|
/* setReadException= */ false, /* unknownLength= */ false, cacheKeyFactory);
|
||||||
cacheDataSource.open(buildDataSpec(TEST_DATA.length, /* length= */ 1, customCacheKey));
|
cacheDataSource.open(buildDataSpec(TEST_DATA.length + 1, /* length= */ 1, customCacheKey));
|
||||||
fail();
|
fail();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// Expected.
|
// Expected.
|
||||||
|
|
|
||||||
|
|
@ -27,16 +27,15 @@ import com.google.android.exoplayer2.testutil.FailOnCloseDataSink;
|
||||||
import com.google.android.exoplayer2.testutil.FakeDataSet;
|
import com.google.android.exoplayer2.testutil.FakeDataSet;
|
||||||
import com.google.android.exoplayer2.testutil.FakeDataSource;
|
import com.google.android.exoplayer2.testutil.FakeDataSource;
|
||||||
import com.google.android.exoplayer2.testutil.TestUtil;
|
import com.google.android.exoplayer2.testutil.TestUtil;
|
||||||
import com.google.android.exoplayer2.upstream.DataSourceException;
|
|
||||||
import com.google.android.exoplayer2.upstream.DataSpec;
|
import com.google.android.exoplayer2.upstream.DataSpec;
|
||||||
import com.google.android.exoplayer2.upstream.FileDataSource;
|
import com.google.android.exoplayer2.upstream.FileDataSource;
|
||||||
import com.google.android.exoplayer2.util.Util;
|
import com.google.android.exoplayer2.util.Util;
|
||||||
|
import java.io.EOFException;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.concurrent.atomic.AtomicBoolean;
|
import java.util.concurrent.atomic.AtomicBoolean;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Ignore;
|
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.junit.runner.RunWith;
|
import org.junit.runner.RunWith;
|
||||||
import org.mockito.MockitoAnnotations;
|
import org.mockito.MockitoAnnotations;
|
||||||
|
|
@ -175,7 +174,6 @@ public final class CacheWriterTest {
|
||||||
assertCachedData(cache, fakeDataSet);
|
assertCachedData(cache, fakeDataSet);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Ignore("Currently broken. See https://github.com/google/ExoPlayer/issues/7326.")
|
|
||||||
@Test
|
@Test
|
||||||
public void cacheLengthExceedsActualDataLength() throws Exception {
|
public void cacheLengthExceedsActualDataLength() throws Exception {
|
||||||
FakeDataSet fakeDataSet = new FakeDataSet().setRandomData("test_data", 100);
|
FakeDataSet fakeDataSet = new FakeDataSet().setRandomData("test_data", 100);
|
||||||
|
|
@ -206,18 +204,14 @@ public final class CacheWriterTest {
|
||||||
Uri testUri = Uri.parse("test_data");
|
Uri testUri = Uri.parse("test_data");
|
||||||
DataSpec dataSpec = new DataSpec(testUri, /* position= */ 0, /* length= */ 1000);
|
DataSpec dataSpec = new DataSpec(testUri, /* position= */ 0, /* length= */ 1000);
|
||||||
|
|
||||||
IOException exception =
|
CacheWriter cacheWriter =
|
||||||
assertThrows(
|
new CacheWriter(
|
||||||
IOException.class,
|
new CacheDataSource(cache, dataSource),
|
||||||
() ->
|
dataSpec,
|
||||||
new CacheWriter(
|
/* allowShortContent= */ false,
|
||||||
new CacheDataSource(cache, dataSource),
|
/* temporaryBuffer= */ null,
|
||||||
dataSpec,
|
/* progressListener= */ null);
|
||||||
/* allowShortContent= */ false,
|
assertThrows(EOFException.class, cacheWriter::cache);
|
||||||
/* temporaryBuffer= */ null,
|
|
||||||
/* progressListener= */ null)
|
|
||||||
.cache());
|
|
||||||
assertThat(DataSourceException.isCausedByPositionOutOfRange(exception)).isTrue();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
|
||||||
|
|
@ -233,16 +233,22 @@ public abstract class DataSourceContractTest {
|
||||||
new DataSpec.Builder().setUri(resource.getUri()).setPosition(resourceLength).build();
|
new DataSpec.Builder().setUri(resource.getUri()).setPosition(resourceLength).build();
|
||||||
try {
|
try {
|
||||||
long length = dataSource.open(dataSpec);
|
long length = dataSource.open(dataSpec);
|
||||||
// TODO: For any cases excluded from the requirement that a position-out-of-range exception
|
// The DataSource.open() contract requires the returned length to equal the length in the
|
||||||
// is thrown, decide what the allowed behavior should be for the first read, and assert it.
|
// DataSpec if set. This is true even though the DataSource implementation may know that
|
||||||
|
// fewer bytes will be read in this case.
|
||||||
if (length != C.LENGTH_UNSET) {
|
if (length != C.LENGTH_UNSET) {
|
||||||
assertThat(length).isEqualTo(0);
|
assertThat(length).isEqualTo(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
byte[] data =
|
||||||
|
unboundedReadsAreIndefinite() ? Util.EMPTY_BYTE_ARRAY : Util.readToEnd(dataSource);
|
||||||
|
assertThat(data).isEmpty();
|
||||||
|
} catch (IOException e) {
|
||||||
|
// TODO: Remove this catch and require that implementations do not throw.
|
||||||
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// TODO: Decide whether to assert that a position-out-of-range exception must or must not be
|
// TODO: Remove this catch and require that implementations do not throw.
|
||||||
// thrown (with exclusions if necessary), rather than just asserting it must be a
|
|
||||||
// position-out-of-range exception *if* one is thrown at all.
|
|
||||||
assertThat(DataSourceException.isCausedByPositionOutOfRange(e)).isTrue();
|
assertThat(DataSourceException.isCausedByPositionOutOfRange(e)).isTrue();
|
||||||
} finally {
|
} finally {
|
||||||
dataSource.close();
|
dataSource.close();
|
||||||
|
|
@ -270,17 +276,20 @@ public abstract class DataSourceContractTest {
|
||||||
.build();
|
.build();
|
||||||
try {
|
try {
|
||||||
long length = dataSource.open(dataSpec);
|
long length = dataSource.open(dataSpec);
|
||||||
// TODO: For any cases excluded from the requirement that a position-out-of-range exception
|
|
||||||
// is thrown, decide what the allowed behavior should be for the first read, and assert it.
|
|
||||||
|
|
||||||
// The DataSource.open() contract requires the returned length to equal the length in the
|
// The DataSource.open() contract requires the returned length to equal the length in the
|
||||||
// DataSpec if set. This is true even though the DataSource implementation may know that
|
// DataSpec if set. This is true even though the DataSource implementation may know that
|
||||||
// fewer bytes will be read in this case.
|
// fewer bytes will be read in this case.
|
||||||
assertThat(length).isEqualTo(1);
|
assertThat(length).isEqualTo(1);
|
||||||
|
|
||||||
|
try {
|
||||||
|
byte[] data =
|
||||||
|
unboundedReadsAreIndefinite() ? Util.EMPTY_BYTE_ARRAY : Util.readToEnd(dataSource);
|
||||||
|
assertThat(data).isEmpty();
|
||||||
|
} catch (IOException e) {
|
||||||
|
// TODO: Remove this catch and require that implementations do not throw.
|
||||||
|
}
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// TODO: Decide whether to assert that a position-out-of-range exception must or must not be
|
// TODO: Remove this catch and require that implementations do not throw.
|
||||||
// thrown (with exclusions if necessary), rather than just asserting it must be a
|
|
||||||
// position-out-of-range exception *if* one is thrown at all.
|
|
||||||
assertThat(DataSourceException.isCausedByPositionOutOfRange(e)).isTrue();
|
assertThat(DataSourceException.isCausedByPositionOutOfRange(e)).isTrue();
|
||||||
} finally {
|
} finally {
|
||||||
dataSource.close();
|
dataSource.close();
|
||||||
|
|
|
||||||
|
|
@ -117,7 +117,7 @@ public class FakeDataSource extends BaseDataSource {
|
||||||
throw new IOException("Data is empty: " + dataSpec.uri);
|
throw new IOException("Data is empty: " + dataSpec.uri);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dataSpec.position >= totalLength) {
|
if (dataSpec.position > totalLength) {
|
||||||
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
throw new DataSourceException(DataSourceException.POSITION_OUT_OF_RANGE);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -211,22 +211,6 @@ public final class FakeDataSourceTest {
|
||||||
} finally {
|
} finally {
|
||||||
dataSource.close();
|
dataSource.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
// DataSpec out of bounds.
|
|
||||||
dataSource =
|
|
||||||
new FakeDataSource(
|
|
||||||
new FakeDataSet()
|
|
||||||
.newDefaultData()
|
|
||||||
.appendReadData(TestUtil.buildTestData(/* length= */ 10))
|
|
||||||
.endData());
|
|
||||||
try {
|
|
||||||
dataSource.open(new DataSpec(uri, /* position= */ 10, C.LENGTH_UNSET));
|
|
||||||
fail("IOException expected.");
|
|
||||||
} catch (IOException e) {
|
|
||||||
// Expected.
|
|
||||||
} finally {
|
|
||||||
dataSource.close();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void assertBuffer(byte[] expected) {
|
private static void assertBuffer(byte[] expected) {
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue