cog/Plugins/HTTPSource/HTTPSource.m

234 lines
5.5 KiB
Matlab
Raw Normal View History

2009-03-05 08:01:36 +00:00
//
// HTTPSource.m
// HTTPSource
//
// Created by Vincent Spader on 3/1/07.
// Replaced by Christopher Snowhill on 3/7/20.
// Copyright 2020 __LoSnoCo__. All rights reserved.
2009-03-05 08:01:36 +00:00
//
#import "HTTPSource.h"
#import "Logging.h"
#define BUFFER_SIZE 131072
2009-03-05 08:01:36 +00:00
@implementation HTTPSource
- (NSURLSession *)createSession
2009-03-05 08:01:36 +00:00
{
queue = [[NSOperationQueue alloc] init];
[queue setMaxConcurrentOperationCount:1];
NSURLSession *session = nil;
session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
delegate:self
delegateQueue:queue];
return session;
}
- (void)URLSession:(NSURLSession *)session
dataTask:(NSURLSessionDataTask *)dataTask
didReceiveData:(NSData *)data{
long bytesBuffered = 0;
if (!task) return;
@synchronized(bufferedData) {
[bufferedData addObject:data];
_bytesBuffered += [data length];
bytesBuffered = _bytesBuffered;
}
if (bytesBuffered > BUFFER_SIZE) {
[task suspend];
}
}
2009-03-05 08:01:36 +00:00
- (void)URLSession:(NSURLSession *)session
task:(NSURLSessionTask *)task
willPerformHTTPRedirection:(NSHTTPURLResponse *)response
newRequest:(NSURLRequest *)request
completionHandler:(void (^)(NSURLRequest *))completionHandler {
NSURL * url = [request URL];
if ([redirectURLs containsObject:url]) {
@synchronized(self->task) {
self->task = nil;
}
}
else {
[redirectURLs addObject:url];
redirected = YES;
@synchronized(bufferedData) {
[bufferedData removeAllObjects];
_bytesBuffered = 0;
}
@synchronized(self->task) {
self->task = [session dataTaskWithRequest:request];
[self->task resume];
}
}
}
- (void)URLSession:(NSURLSession *)session
didBecomeInvalidWithError:(NSError *)error {
@synchronized(task) {
task = nil;
}
}
- (void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask
willCacheResponse:(NSCachedURLResponse *)proposedResponse
completionHandler:(void (^)(NSCachedURLResponse *cachedResponse))completionHandler{
}
- (void)URLSession:(NSURLSession *)session
task:(NSURLSessionTask *)task
didCompleteWithError:(NSError *)error{
@synchronized(self->task) {
self->task = nil;
}
}
- (BOOL)open:(NSURL *)url
{
redirected = NO;
redirectURLs = [[NSMutableArray alloc] init];
bufferedData = [[NSMutableArray alloc] init];
URL = url;
[redirectURLs addObject:URL];
NSURLRequest * request = [NSURLRequest requestWithURL:url];
session = [self createSession];
task = [session dataTaskWithRequest:request];
[task resume];
NSURLResponse * response = nil;
while (!response) {
@synchronized(task) {
if (!task) return NO;
redirected = NO;
response = [task response];
}
if (response && !redirected) break;
if (redirected) continue;
usleep(100);
}
if (response) {
if ([response isKindOfClass:[NSHTTPURLResponse class]]) {
NSInteger statusCode = [(NSHTTPURLResponse *)response statusCode];
if (statusCode != 200)
return NO;
}
_mimeType = [response MIMEType];
}
2009-03-05 08:01:36 +00:00
return YES;
}
- (NSString *)mimeType
{
DLog(@"Returning mimetype! %@", _mimeType);
2009-03-05 08:01:36 +00:00
return _mimeType;
}
- (BOOL)seekable
{
return NO;
}
- (BOOL)seek:(long)position whence:(int)whence
{
return NO;
}
- (long)tell
{
return _byteCount;
}
- (long)read:(void *)buffer amount:(long)amount
2009-03-05 08:01:36 +00:00
{
long totalRead = 0;
long bytesBuffered = 0;
2009-03-05 08:01:36 +00:00
while (totalRead < amount) {
NSData * dataBlock = nil;
@synchronized(bufferedData) {
if ([bufferedData count])
dataBlock = [bufferedData objectAtIndex:0];
}
if (!dataBlock) {
@synchronized(task) {
if (!task) return totalRead;
}
usleep(1000);
continue;
}
NSInteger amountReceived = [dataBlock length];
if (amountReceived <= 0) {
break;
}
NSInteger amountUsed = amountReceived;
if (amountUsed > (amount - totalRead) )
amountUsed = amount - totalRead;
const void * dataBytes = [dataBlock bytes];
memcpy(((uint8_t *)buffer) + totalRead, dataBytes, amountUsed);
if (amountUsed < amountReceived) {
NSData * dataOut = [NSData dataWithBytes:(((uint8_t *)dataBytes) + amountUsed) length:(amountReceived - amountUsed)];
@synchronized(bufferedData) {
[bufferedData removeObjectAtIndex:0];
[bufferedData insertObject:dataOut atIndex:0];
_bytesBuffered -= amountUsed;
bytesBuffered = _bytesBuffered;
}
}
else {
@synchronized(bufferedData) {
[bufferedData removeObjectAtIndex:0];
_bytesBuffered -= amountUsed;
bytesBuffered = _bytesBuffered;
}
}
if (bytesBuffered <= (BUFFER_SIZE * 3 / 4)) {
[task resume];
}
totalRead += amountUsed;
2009-03-05 08:01:36 +00:00
}
2009-03-05 08:01:36 +00:00
_byteCount += totalRead;
return totalRead;
}
- (void)close
{
if (task) [task cancel];
task = nil;
2009-03-05 08:01:36 +00:00
_mimeType = nil;
}
- (void)dealloc
{
[self close];
}
- (NSURL *)url
{
return URL;
2009-03-05 08:01:36 +00:00
}
+ (NSArray *)schemes
{
return [NSArray arrayWithObjects:@"http", @"https", nil];
2009-03-05 08:01:36 +00:00
}
@end