mirror of
https://github.com/sparkle-project/Sparkle.git
synced 2025-11-01 15:34:38 +00:00
268 lines
8.8 KiB
Objective-C
268 lines
8.8 KiB
Objective-C
//
|
|
// SUTestWebServer.m
|
|
// Sparkle
|
|
//
|
|
// Created by Kevin Wojniak on 10/8/15.
|
|
// Copyright © 2015 Sparkle Project. All rights reserved.
|
|
//
|
|
|
|
#import "SUTestWebServer.h"
|
|
#import <sys/socket.h>
|
|
#import <netinet/in.h>
|
|
|
|
@class SUTestWebServerConnection;
|
|
|
|
@protocol SUTestWebServerConnectionDelegate <NSObject>
|
|
@required
|
|
- (void)connectionDidClose:(SUTestWebServerConnection*)sender;
|
|
@end
|
|
|
|
@interface SUTestWebServerConnection : NSObject <NSStreamDelegate>
|
|
@end
|
|
|
|
@implementation SUTestWebServerConnection
|
|
{
|
|
NSString* _workingDirectory;
|
|
NSInputStream *_inputStream;
|
|
NSOutputStream *_outputStream;
|
|
NSData *_dataToWrite;
|
|
NSInteger _numBytesToWrite;
|
|
|
|
__weak id<SUTestWebServerConnectionDelegate> _delegate;
|
|
}
|
|
|
|
- (instancetype)initWithNativeHandle:(CFSocketNativeHandle)handle workingDirectory:(NSString*)workingDirectory delegate:(id<SUTestWebServerConnectionDelegate>)delegate SPU_OBJC_DIRECT
|
|
{
|
|
self = [super init];
|
|
assert(self != nil);
|
|
|
|
_workingDirectory = workingDirectory;
|
|
_delegate = delegate;
|
|
|
|
CFReadStreamRef readStream = NULL;
|
|
CFWriteStreamRef writeStream = NULL;
|
|
CFStreamCreatePairWithSocket(NULL, handle, &readStream, &writeStream);
|
|
assert(readStream != NULL);
|
|
assert(writeStream != NULL);
|
|
|
|
_inputStream = (__bridge NSInputStream*)readStream;
|
|
assert(_inputStream != nil);
|
|
_inputStream.delegate = self;
|
|
|
|
_outputStream = (__bridge NSOutputStream*)writeStream;
|
|
assert(_outputStream != nil);
|
|
_outputStream.delegate = self;
|
|
|
|
[_inputStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
|
[_outputStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
|
[_inputStream open];
|
|
[_outputStream open];
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)close SPU_OBJC_DIRECT
|
|
{
|
|
NSInputStream *inputStream = _inputStream;
|
|
NSOutputStream *outputStream = _outputStream;
|
|
if (inputStream == nil) {
|
|
assert(outputStream == nil);
|
|
return;
|
|
}
|
|
[inputStream close];
|
|
[outputStream close];
|
|
[inputStream removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
|
[outputStream removeFromRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];
|
|
_inputStream = nil;
|
|
_outputStream = nil;
|
|
[_delegate connectionDidClose:self];
|
|
}
|
|
|
|
- (void)stream:(NSStream *)aStream handleEvent:(NSStreamEvent)eventCode
|
|
{
|
|
if (eventCode == NSStreamEventEndEncountered) {
|
|
[self close];
|
|
return;
|
|
}
|
|
if (aStream == _inputStream && eventCode == NSStreamEventHasBytesAvailable) {
|
|
uint8_t buffer[8096];
|
|
const NSInteger numBytes = [_inputStream read:buffer maxLength:sizeof(buffer)];
|
|
if (numBytes > 0) {
|
|
NSString *request = [[NSString alloc] initWithBytes:buffer length:(NSUInteger)numBytes encoding:NSUTF8StringEncoding];
|
|
NSArray *lines = [request componentsSeparatedByString:@"\r\n"];
|
|
NSString *requestLine = lines.count >= 3 ? [lines objectAtIndex:0] : nil;
|
|
NSArray *parts = requestLine ? [requestLine componentsSeparatedByString:@" "] : nil;
|
|
// Only process GET requests for existing files
|
|
if ([(NSString *)[parts objectAtIndex:0] isEqualToString:@"GET"]) {
|
|
// Use NSURL to strip out query parameters
|
|
NSString *path = [NSURL URLWithString:[parts objectAtIndex:1] relativeToURL:nil].path;
|
|
NSString *filePath = [_workingDirectory stringByAppendingString:path];
|
|
BOOL isDir = NO;
|
|
if (![[NSFileManager defaultManager] fileExistsAtPath:filePath isDirectory:&isDir] || isDir) {
|
|
NSLog(@"%@ - 404", requestLine);
|
|
[self write404];
|
|
} else {
|
|
NSLog(@"%@ - 200", requestLine);
|
|
[self write:[NSData dataWithContentsOfFile:filePath] status:YES];
|
|
}
|
|
} else {
|
|
NSLog(@"%@ - 404", requestLine);
|
|
[self write404];
|
|
}
|
|
}
|
|
} else if (aStream == _outputStream && eventCode == NSStreamEventHasSpaceAvailable && _dataToWrite != nil) {
|
|
[self checkIfCanWriteNow];
|
|
}
|
|
}
|
|
|
|
- (void)write404 SPU_OBJC_DIRECT
|
|
{
|
|
NSString *body = @"<html><head><title>404 Not Found</title></head><body><h1>Not Found</h1></body></html>";
|
|
[self write:[body dataUsingEncoding:NSUTF8StringEncoding] status:NO];
|
|
}
|
|
|
|
- (void)write:(NSData*)body status:(BOOL)status SPU_OBJC_DIRECT
|
|
{
|
|
NSString *state = status ? @"200 OK" : @"404 Not Found";
|
|
NSString *header = [NSString stringWithFormat:@"HTTP/1.0 %@\r\nContent-Length: %lu\r\n\r\n", state, body.length];
|
|
NSMutableData *response = [[header dataUsingEncoding:NSUTF8StringEncoding] mutableCopy];
|
|
[response appendData:body];
|
|
[self queueWrite:response];
|
|
}
|
|
|
|
- (void)queueWrite:(NSData*)data SPU_OBJC_DIRECT
|
|
{
|
|
assert(_dataToWrite == nil);
|
|
assert(data != nil);
|
|
assert(data.length > 0);
|
|
_dataToWrite = data;
|
|
_numBytesToWrite = (NSInteger)data.length;
|
|
[self checkIfCanWriteNow];
|
|
}
|
|
|
|
- (void)checkIfCanWriteNow SPU_OBJC_DIRECT
|
|
{
|
|
assert(_dataToWrite != nil);
|
|
if (_numBytesToWrite == 0) {
|
|
// nothing more to write, we're done.
|
|
_dataToWrite = nil;
|
|
_numBytesToWrite = -1;
|
|
} else if (_outputStream.hasSpaceAvailable) {
|
|
[self writeNow];
|
|
}
|
|
// otherwise wait for space available event
|
|
}
|
|
|
|
- (void)writeNow SPU_OBJC_DIRECT
|
|
{
|
|
assert(_outputStream != nil);
|
|
assert(_outputStream.hasSpaceAvailable);
|
|
assert(_dataToWrite != nil);
|
|
NSData *dataToWrite = _dataToWrite;
|
|
const uint8_t *bytesOffset = (const uint8_t*)dataToWrite.bytes + ((NSInteger)dataToWrite.length - _numBytesToWrite);
|
|
const NSInteger bytesWritten = [_outputStream write:bytesOffset maxLength:(NSUInteger)_numBytesToWrite];
|
|
if (bytesWritten > 0) {
|
|
_numBytesToWrite = _numBytesToWrite - bytesWritten;
|
|
assert(_numBytesToWrite >= 0);
|
|
// wait for next space available event to write more
|
|
} else {
|
|
NSLog(@"Error: bytes written = %ld (%@)", bytesWritten, [NSString stringWithUTF8String:strerror(errno)]);
|
|
}
|
|
}
|
|
|
|
@end
|
|
|
|
SPU_OBJC_DIRECT_MEMBERS @interface SUTestWebServer () <SUTestWebServerConnectionDelegate> {
|
|
CFSocketRef _socket;
|
|
}
|
|
|
|
@property (nonatomic) NSMutableArray *connections;
|
|
@property (nonatomic) NSString *workingDirectory;
|
|
|
|
- (void)accept:(CFSocketNativeHandle)address;
|
|
|
|
@end
|
|
|
|
static void connectCallback(CFSocketRef __unused s, CFSocketCallBackType type, CFDataRef __unused address, const void *data, void *info)
|
|
{
|
|
if (type == kCFSocketAcceptCallBack) {
|
|
assert(data != NULL);
|
|
assert(info != NULL);
|
|
SUTestWebServer *server = (__bridge SUTestWebServer*)info;
|
|
assert(server != nil);
|
|
[server accept:*(const CFSocketNativeHandle*)data];
|
|
}
|
|
}
|
|
|
|
@implementation SUTestWebServer
|
|
|
|
@synthesize connections = _connections;
|
|
@synthesize workingDirectory = _workingDirectory;
|
|
|
|
- (instancetype)initWithPort:(int)port workingDirectory:(NSString*)workingDirectory
|
|
{
|
|
self = [super init];
|
|
assert(self != nil);
|
|
|
|
CFSocketContext ctx;
|
|
memset(&ctx, 0, sizeof(ctx));
|
|
ctx.info = (__bridge void*)self;
|
|
_socket = CFSocketCreate(NULL, 0, 0, 0, kCFSocketAcceptCallBack, connectCallback, &ctx);
|
|
assert(_socket != NULL);
|
|
|
|
struct sockaddr_in address;
|
|
memset(&address, 0, sizeof(address));
|
|
address.sin_len = sizeof(address);
|
|
address.sin_family = AF_INET;
|
|
address.sin_port = htons(port);
|
|
address.sin_addr.s_addr = INADDR_ANY;
|
|
|
|
// will fail if port is in use.
|
|
CFSocketError socketErr = CFSocketSetAddress(_socket, (CFDataRef)[NSData dataWithBytes:&address length:sizeof(address)]);
|
|
if (socketErr != kCFSocketSuccess) {
|
|
NSLog(@"Socket error: %@", [NSString stringWithUTF8String:strerror(errno)]);
|
|
return nil;
|
|
}
|
|
|
|
_connections = [[NSMutableArray alloc] init];
|
|
_workingDirectory = workingDirectory;
|
|
|
|
CFRunLoopSourceRef source = CFSocketCreateRunLoopSource(NULL, _socket, 0);
|
|
assert(source != NULL);
|
|
CFRunLoopAddSource(CFRunLoopGetCurrent(), source, kCFRunLoopDefaultMode);
|
|
CFRelease(source);
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)connectionDidClose:(SUTestWebServerConnection *)sender
|
|
{
|
|
assert(_connections != nil);
|
|
assert([_connections containsObject:sender]);
|
|
[_connections removeObject:sender];
|
|
}
|
|
|
|
- (void)accept:(CFSocketNativeHandle)address
|
|
{
|
|
SUTestWebServerConnection *conn = [[SUTestWebServerConnection alloc] initWithNativeHandle:address workingDirectory:_workingDirectory delegate:self];
|
|
assert(conn != nil);
|
|
if (conn) {
|
|
assert(_connections != nil);
|
|
[_connections addObject:conn];
|
|
}
|
|
}
|
|
|
|
- (void)close
|
|
{
|
|
for (SUTestWebServerConnection *conn in _connections) {
|
|
[conn close];
|
|
}
|
|
if (_socket) {
|
|
CFSocketInvalidate(_socket);
|
|
CFRelease(_socket);
|
|
_socket = NULL;
|
|
}
|
|
}
|
|
|
|
@end
|