initial commit, iOS network native library
This commit is contained in:
commit
e49ee1551d
405
.gitignore
vendored
Normal file
405
.gitignore
vendored
Normal file
@ -0,0 +1,405 @@
|
|||||||
|
# globs
|
||||||
|
Makefile.in
|
||||||
|
*.userprefs
|
||||||
|
*.usertasks
|
||||||
|
config.make
|
||||||
|
config.status
|
||||||
|
aclocal.m4
|
||||||
|
install-sh
|
||||||
|
autom4te.cache/
|
||||||
|
*.tar.gz
|
||||||
|
tarballs/
|
||||||
|
test-results/
|
||||||
|
|
||||||
|
# Mac bundle stuff
|
||||||
|
*.dmg
|
||||||
|
*.app
|
||||||
|
|
||||||
|
# content below from: https://github.com/github/gitignore/blob/main/Global/macOS.gitignore
|
||||||
|
# General
|
||||||
|
.DS_Store
|
||||||
|
.AppleDouble
|
||||||
|
.LSOverride
|
||||||
|
|
||||||
|
# Icon must end with two \r
|
||||||
|
Icon
|
||||||
|
|
||||||
|
|
||||||
|
# Thumbnails
|
||||||
|
._*
|
||||||
|
|
||||||
|
# Files that might appear in the root of a volume
|
||||||
|
.DocumentRevisions-V100
|
||||||
|
.fseventsd
|
||||||
|
.Spotlight-V100
|
||||||
|
.TemporaryItems
|
||||||
|
.Trashes
|
||||||
|
.VolumeIcon.icns
|
||||||
|
.com.apple.timemachine.donotpresent
|
||||||
|
|
||||||
|
# Directories potentially created on remote AFP share
|
||||||
|
.AppleDB
|
||||||
|
.AppleDesktop
|
||||||
|
Network Trash Folder
|
||||||
|
Temporary Items
|
||||||
|
.apdisk
|
||||||
|
|
||||||
|
# content below from: https://github.com/github/gitignore/blob/main/Global/Windows.gitignore
|
||||||
|
# Windows thumbnail cache files
|
||||||
|
Thumbs.db
|
||||||
|
ehthumbs.db
|
||||||
|
ehthumbs_vista.db
|
||||||
|
|
||||||
|
# Dump file
|
||||||
|
*.stackdump
|
||||||
|
|
||||||
|
# Folder config file
|
||||||
|
[Dd]esktop.ini
|
||||||
|
|
||||||
|
# Recycle Bin used on file shares
|
||||||
|
$RECYCLE.BIN/
|
||||||
|
|
||||||
|
# Windows Installer files
|
||||||
|
*.cab
|
||||||
|
*.msi
|
||||||
|
*.msix
|
||||||
|
*.msm
|
||||||
|
*.msp
|
||||||
|
|
||||||
|
# Windows shortcuts
|
||||||
|
*.lnk
|
||||||
|
|
||||||
|
# content below from: https://github.com/github/gitignore/blob/master/VisualStudio.gitignore
|
||||||
|
## Ignore Visual Studio temporary files, build results, and
|
||||||
|
## files generated by popular Visual Studio add-ons.
|
||||||
|
##
|
||||||
|
## Get latest from https://github.com/github/gitignore/blob/master/VisualStudio.gitignore
|
||||||
|
|
||||||
|
# User-specific files
|
||||||
|
*.suo
|
||||||
|
*.user
|
||||||
|
*.userosscache
|
||||||
|
*.sln.docstates
|
||||||
|
|
||||||
|
# User-specific files (MonoDevelop/Xamarin Studio)
|
||||||
|
*.userprefs
|
||||||
|
|
||||||
|
# Build results
|
||||||
|
[Dd]ebug/
|
||||||
|
[Dd]ebugPublic/
|
||||||
|
[Rr]elease/
|
||||||
|
[Rr]eleases/
|
||||||
|
x64/
|
||||||
|
x86/
|
||||||
|
bld/
|
||||||
|
[Bb]in/
|
||||||
|
[Oo]bj/
|
||||||
|
[Ll]og/
|
||||||
|
|
||||||
|
# Visual Studio 2015/2017 cache/options directory
|
||||||
|
.vs/
|
||||||
|
# Uncomment if you have tasks that create the project's static files in wwwroot
|
||||||
|
#wwwroot/
|
||||||
|
|
||||||
|
# Visual Studio 2017 auto generated files
|
||||||
|
Generated\ Files/
|
||||||
|
|
||||||
|
# MSTest test Results
|
||||||
|
[Tt]est[Rr]esult*/
|
||||||
|
[Bb]uild[Ll]og.*
|
||||||
|
|
||||||
|
# NUNIT
|
||||||
|
*.VisualState.xml
|
||||||
|
TestResult.xml
|
||||||
|
|
||||||
|
# Build Results of an ATL Project
|
||||||
|
[Dd]ebugPS/
|
||||||
|
[Rr]eleasePS/
|
||||||
|
dlldata.c
|
||||||
|
|
||||||
|
# Benchmark Results
|
||||||
|
BenchmarkDotNet.Artifacts/
|
||||||
|
|
||||||
|
# .NET Core
|
||||||
|
project.lock.json
|
||||||
|
project.fragment.lock.json
|
||||||
|
artifacts/
|
||||||
|
|
||||||
|
# StyleCop
|
||||||
|
StyleCopReport.xml
|
||||||
|
|
||||||
|
# Files built by Visual Studio
|
||||||
|
*_i.c
|
||||||
|
*_p.c
|
||||||
|
*_h.h
|
||||||
|
*.ilk
|
||||||
|
*.meta
|
||||||
|
*.obj
|
||||||
|
*.iobj
|
||||||
|
*.pch
|
||||||
|
*.pdb
|
||||||
|
*.ipdb
|
||||||
|
*.pgc
|
||||||
|
*.pgd
|
||||||
|
*.rsp
|
||||||
|
*.sbr
|
||||||
|
*.tlb
|
||||||
|
*.tli
|
||||||
|
*.tlh
|
||||||
|
*.tmp
|
||||||
|
*.tmp_proj
|
||||||
|
*_wpftmp.csproj
|
||||||
|
*.log
|
||||||
|
*.vspscc
|
||||||
|
*.vssscc
|
||||||
|
.builds
|
||||||
|
*.pidb
|
||||||
|
*.svclog
|
||||||
|
*.scc
|
||||||
|
|
||||||
|
# Chutzpah Test files
|
||||||
|
_Chutzpah*
|
||||||
|
|
||||||
|
# Visual C++ cache files
|
||||||
|
ipch/
|
||||||
|
*.aps
|
||||||
|
*.ncb
|
||||||
|
*.opendb
|
||||||
|
*.opensdf
|
||||||
|
*.sdf
|
||||||
|
*.cachefile
|
||||||
|
*.VC.db
|
||||||
|
*.VC.VC.opendb
|
||||||
|
|
||||||
|
# Visual Studio profiler
|
||||||
|
*.psess
|
||||||
|
*.vsp
|
||||||
|
*.vspx
|
||||||
|
*.sap
|
||||||
|
|
||||||
|
# Visual Studio Trace Files
|
||||||
|
*.e2e
|
||||||
|
|
||||||
|
# TFS 2012 Local Workspace
|
||||||
|
$tf/
|
||||||
|
|
||||||
|
# Guidance Automation Toolkit
|
||||||
|
*.gpState
|
||||||
|
|
||||||
|
# ReSharper is a .NET coding add-in
|
||||||
|
_ReSharper*/
|
||||||
|
*.[Rr]e[Ss]harper
|
||||||
|
*.DotSettings.user
|
||||||
|
|
||||||
|
# JustCode is a .NET coding add-in
|
||||||
|
.JustCode
|
||||||
|
|
||||||
|
# TeamCity is a build add-in
|
||||||
|
_TeamCity*
|
||||||
|
|
||||||
|
# DotCover is a Code Coverage Tool
|
||||||
|
*.dotCover
|
||||||
|
|
||||||
|
# AxoCover is a Code Coverage Tool
|
||||||
|
.axoCover/*
|
||||||
|
!.axoCover/settings.json
|
||||||
|
|
||||||
|
# Visual Studio code coverage results
|
||||||
|
*.coverage
|
||||||
|
*.coveragexml
|
||||||
|
|
||||||
|
# NCrunch
|
||||||
|
_NCrunch_*
|
||||||
|
.*crunch*.local.xml
|
||||||
|
nCrunchTemp_*
|
||||||
|
|
||||||
|
# MightyMoose
|
||||||
|
*.mm.*
|
||||||
|
AutoTest.Net/
|
||||||
|
|
||||||
|
# Web workbench (sass)
|
||||||
|
.sass-cache/
|
||||||
|
|
||||||
|
# Installshield output folder
|
||||||
|
[Ee]xpress/
|
||||||
|
|
||||||
|
# DocProject is a documentation generator add-in
|
||||||
|
DocProject/buildhelp/
|
||||||
|
DocProject/Help/*.HxT
|
||||||
|
DocProject/Help/*.HxC
|
||||||
|
DocProject/Help/*.hhc
|
||||||
|
DocProject/Help/*.hhk
|
||||||
|
DocProject/Help/*.hhp
|
||||||
|
DocProject/Help/Html2
|
||||||
|
DocProject/Help/html
|
||||||
|
|
||||||
|
# Click-Once directory
|
||||||
|
publish/
|
||||||
|
|
||||||
|
# Publish Web Output
|
||||||
|
*.[Pp]ublish.xml
|
||||||
|
*.azurePubxml
|
||||||
|
# Note: Comment the next line if you want to checkin your web deploy settings,
|
||||||
|
# but database connection strings (with potential passwords) will be unencrypted
|
||||||
|
*.pubxml
|
||||||
|
*.publishproj
|
||||||
|
|
||||||
|
# Microsoft Azure Web App publish settings. Comment the next line if you want to
|
||||||
|
# checkin your Azure Web App publish settings, but sensitive information contained
|
||||||
|
# in these scripts will be unencrypted
|
||||||
|
PublishScripts/
|
||||||
|
|
||||||
|
# NuGet Packages
|
||||||
|
*.nupkg
|
||||||
|
# The packages folder can be ignored because of Package Restore
|
||||||
|
**/[Pp]ackages/*
|
||||||
|
# except build/, which is used as an MSBuild target.
|
||||||
|
!**/[Pp]ackages/build/
|
||||||
|
# Uncomment if necessary however generally it will be regenerated when needed
|
||||||
|
#!**/[Pp]ackages/repositories.config
|
||||||
|
# NuGet v3's project.json files produces more ignorable files
|
||||||
|
*.nuget.props
|
||||||
|
*.nuget.targets
|
||||||
|
|
||||||
|
# Microsoft Azure Build Output
|
||||||
|
csx/
|
||||||
|
*.build.csdef
|
||||||
|
|
||||||
|
# Microsoft Azure Emulator
|
||||||
|
ecf/
|
||||||
|
rcf/
|
||||||
|
|
||||||
|
# Windows Store app package directories and files
|
||||||
|
AppPackages/
|
||||||
|
BundleArtifacts/
|
||||||
|
Package.StoreAssociation.xml
|
||||||
|
_pkginfo.txt
|
||||||
|
*.appx
|
||||||
|
|
||||||
|
# Visual Studio cache files
|
||||||
|
# files ending in .cache can be ignored
|
||||||
|
*.[Cc]ache
|
||||||
|
# but keep track of directories ending in .cache
|
||||||
|
!*.[Cc]ache/
|
||||||
|
|
||||||
|
# Others
|
||||||
|
ClientBin/
|
||||||
|
~$*
|
||||||
|
*~
|
||||||
|
*.dbmdl
|
||||||
|
*.dbproj.schemaview
|
||||||
|
*.jfm
|
||||||
|
*.pfx
|
||||||
|
*.publishsettings
|
||||||
|
orleans.codegen.cs
|
||||||
|
|
||||||
|
# Including strong name files can present a security risk
|
||||||
|
# (https://github.com/github/gitignore/pull/2483#issue-259490424)
|
||||||
|
#*.snk
|
||||||
|
|
||||||
|
# Since there are multiple workflows, uncomment next line to ignore bower_components
|
||||||
|
# (https://github.com/github/gitignore/pull/1529#issuecomment-104372622)
|
||||||
|
#bower_components/
|
||||||
|
|
||||||
|
# RIA/Silverlight projects
|
||||||
|
Generated_Code/
|
||||||
|
|
||||||
|
# Backup & report files from converting an old project file
|
||||||
|
# to a newer Visual Studio version. Backup files are not needed,
|
||||||
|
# because we have git ;-)
|
||||||
|
_UpgradeReport_Files/
|
||||||
|
Backup*/
|
||||||
|
UpgradeLog*.XML
|
||||||
|
UpgradeLog*.htm
|
||||||
|
ServiceFabricBackup/
|
||||||
|
*.rptproj.bak
|
||||||
|
|
||||||
|
# SQL Server files
|
||||||
|
*.mdf
|
||||||
|
*.ldf
|
||||||
|
*.ndf
|
||||||
|
|
||||||
|
# Business Intelligence projects
|
||||||
|
*.rdl.data
|
||||||
|
*.bim.layout
|
||||||
|
*.bim_*.settings
|
||||||
|
*.rptproj.rsuser
|
||||||
|
|
||||||
|
# Microsoft Fakes
|
||||||
|
FakesAssemblies/
|
||||||
|
|
||||||
|
# GhostDoc plugin setting file
|
||||||
|
*.GhostDoc.xml
|
||||||
|
|
||||||
|
# Node.js Tools for Visual Studio
|
||||||
|
.ntvs_analysis.dat
|
||||||
|
node_modules/
|
||||||
|
|
||||||
|
# Visual Studio 6 build log
|
||||||
|
*.plg
|
||||||
|
|
||||||
|
# Visual Studio 6 workspace options file
|
||||||
|
*.opt
|
||||||
|
|
||||||
|
# Visual Studio 6 auto-generated workspace file (contains which files were open etc.)
|
||||||
|
*.vbw
|
||||||
|
|
||||||
|
# Visual Studio LightSwitch build output
|
||||||
|
**/*.HTMLClient/GeneratedArtifacts
|
||||||
|
**/*.DesktopClient/GeneratedArtifacts
|
||||||
|
**/*.DesktopClient/ModelManifest.xml
|
||||||
|
**/*.Server/GeneratedArtifacts
|
||||||
|
**/*.Server/ModelManifest.xml
|
||||||
|
_Pvt_Extensions
|
||||||
|
|
||||||
|
# Paket dependency manager
|
||||||
|
.paket/paket.exe
|
||||||
|
paket-files/
|
||||||
|
|
||||||
|
# FAKE - F# Make
|
||||||
|
.fake/
|
||||||
|
|
||||||
|
# JetBrains Rider
|
||||||
|
.idea/
|
||||||
|
*.sln.iml
|
||||||
|
|
||||||
|
# CodeRush personal settings
|
||||||
|
.cr/personal
|
||||||
|
|
||||||
|
# Python Tools for Visual Studio (PTVS)
|
||||||
|
__pycache__/
|
||||||
|
*.pyc
|
||||||
|
|
||||||
|
# Cake - Uncomment if you are using it
|
||||||
|
# tools/**
|
||||||
|
# !tools/packages.config
|
||||||
|
|
||||||
|
# Tabs Studio
|
||||||
|
*.tss
|
||||||
|
|
||||||
|
# Telerik's JustMock configuration file
|
||||||
|
*.jmconfig
|
||||||
|
|
||||||
|
# BizTalk build output
|
||||||
|
*.btp.cs
|
||||||
|
*.btm.cs
|
||||||
|
*.odx.cs
|
||||||
|
*.xsd.cs
|
||||||
|
|
||||||
|
# OpenCover UI analysis results
|
||||||
|
OpenCover/
|
||||||
|
|
||||||
|
# Azure Stream Analytics local run output
|
||||||
|
ASALocalRun/
|
||||||
|
|
||||||
|
# MSBuild Binary and Structured Log
|
||||||
|
*.binlog
|
||||||
|
|
||||||
|
# NVidia Nsight GPU debugger configuration file
|
||||||
|
*.nvuser
|
||||||
|
|
||||||
|
# MFractors (Xamarin productivity tool) working folder
|
||||||
|
.mfractor/
|
||||||
|
|
||||||
|
# Local History for Visual Studio
|
||||||
|
.localhistory/
|
25
Network.sln
Normal file
25
Network.sln
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
|
||||||
|
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||||
|
# Visual Studio Version 16
|
||||||
|
VisualStudioVersion = 25.0.1706.0
|
||||||
|
MinimumVisualStudioVersion = 10.0.40219.1
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Network", "Network\Network.csproj", "{1D93202E-4401-4C5A-BB91-A3D7C657FB7E}"
|
||||||
|
EndProject
|
||||||
|
Global
|
||||||
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
|
Debug|Any CPU = Debug|Any CPU
|
||||||
|
Release|Any CPU = Release|Any CPU
|
||||||
|
EndGlobalSection
|
||||||
|
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||||
|
{1D93202E-4401-4C5A-BB91-A3D7C657FB7E}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{1D93202E-4401-4C5A-BB91-A3D7C657FB7E}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{1D93202E-4401-4C5A-BB91-A3D7C657FB7E}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{1D93202E-4401-4C5A-BB91-A3D7C657FB7E}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
EndGlobalSection
|
||||||
|
GlobalSection(SolutionProperties) = preSolution
|
||||||
|
HideSolutionNode = FALSE
|
||||||
|
EndGlobalSection
|
||||||
|
GlobalSection(ExtensibilityGlobals) = postSolution
|
||||||
|
SolutionGuid = {C0290D8F-BF20-4987-86C6-81B38710F2DA}
|
||||||
|
EndGlobalSection
|
||||||
|
EndGlobal
|
56
Network/Connectivity.cs
Normal file
56
Network/Connectivity.cs
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
public partial class Connectivity
|
||||||
|
{
|
||||||
|
const string hostName = "www.baidu.com";
|
||||||
|
static NetworkStatus currentStatus;
|
||||||
|
static Action<NetworkStatus>? changedInternal;
|
||||||
|
|
||||||
|
public static NetworkStatus Status => PlatformGetStatus();
|
||||||
|
|
||||||
|
public static event Action<NetworkStatus> Changed
|
||||||
|
{
|
||||||
|
add
|
||||||
|
{
|
||||||
|
var running = changedInternal != null;
|
||||||
|
changedInternal += value;
|
||||||
|
if (!running && changedInternal != null)
|
||||||
|
{
|
||||||
|
currentStatus = Status;
|
||||||
|
StartListeners();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
remove
|
||||||
|
{
|
||||||
|
var running = changedInternal != null;
|
||||||
|
changedInternal -= value;
|
||||||
|
if (running && changedInternal == null)
|
||||||
|
{
|
||||||
|
StopListeners();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static partial NetworkStatus PlatformGetStatus();
|
||||||
|
|
||||||
|
private static partial void StartListeners();
|
||||||
|
|
||||||
|
private static partial void StopListeners();
|
||||||
|
|
||||||
|
static void OnConnectivityChanged()
|
||||||
|
{
|
||||||
|
var status = Status;
|
||||||
|
if (currentStatus != status)
|
||||||
|
{
|
||||||
|
currentStatus = status;
|
||||||
|
changedInternal?.Invoke(status);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public enum NetworkStatus
|
||||||
|
{
|
||||||
|
NotReachable,
|
||||||
|
ReachableViaCarrierDataNetwork,
|
||||||
|
ReachableViaWiFiNetwork
|
||||||
|
}
|
15
Network/Network.csproj
Normal file
15
Network/Network.csproj
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFrameworks>net7.0-ios</TargetFrameworks>
|
||||||
|
<RootNamespace>Blahblah.Library.Network</RootNamespace>
|
||||||
|
<UseMaui>true</UseMaui>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
<SingleProject>true</SingleProject>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
|
||||||
|
<SupportedOSPlatformVersion Condition="$([MSBuild]::GetTargetPlatformIdentifier('$(TargetFramework)')) == 'ios'">14.2</SupportedOSPlatformVersion>
|
||||||
|
<SupportedOSPlatformVersion Condition="$([MSBuild]::GetTargetPlatformIdentifier('$(TargetFramework)')) == 'maccatalyst'">14.0</SupportedOSPlatformVersion>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
</Project>
|
68
Network/NetworkHelper.cs
Normal file
68
Network/NetworkHelper.cs
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
public partial class NetworkHelper
|
||||||
|
{
|
||||||
|
public const string AcceptAll = "*/*";
|
||||||
|
public const string AcceptHttp = "text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8";
|
||||||
|
public const string AcceptImage = "image/webp,image/*,*/*;q=0.8";
|
||||||
|
public const string AcceptJpegImage = "image/jpeg,image/*,*/*;q=0.8";
|
||||||
|
public const string AcceptJson = "application/json";
|
||||||
|
|
||||||
|
const string AcceptHeader = "Accept";
|
||||||
|
const string ReferrerHeader = "Referer";
|
||||||
|
|
||||||
|
const int Timeout = 30;
|
||||||
|
const int ImageTimeout = 60;
|
||||||
|
|
||||||
|
public static CancellationTokenSource TimeoutTokenSource => new(Timeout * 1000);
|
||||||
|
public static CancellationTokenSource ImageTimeoutTokenSource => new(ImageTimeout * 1000);
|
||||||
|
|
||||||
|
static string? proxyHost;
|
||||||
|
static int? proxyPort;
|
||||||
|
|
||||||
|
public static void SetProxy(string? host = null, int? port = null)
|
||||||
|
{
|
||||||
|
//bool changed = proxyHost != host || proxyPort != port;
|
||||||
|
|
||||||
|
proxyHost = host;
|
||||||
|
proxyPort = port;
|
||||||
|
|
||||||
|
/*
|
||||||
|
if (changed)
|
||||||
|
{
|
||||||
|
// TODO: background download
|
||||||
|
}
|
||||||
|
//*/
|
||||||
|
}
|
||||||
|
|
||||||
|
public static NetworkHelper CreateSession(int timeout = Timeout, bool useCookie = true, bool waitsConnectivity = true, Dictionary<string, string>? additionalHeaders = null)
|
||||||
|
{
|
||||||
|
var helper = new NetworkHelper(timeout, useCookie, waitsConnectivity, additionalHeaders);
|
||||||
|
return helper;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static NetworkHelper CreateImageSession(int timeout = ImageTimeout, bool useCookie = false, bool waitsConnectivity = false, Dictionary<string, string>? additionalHeaders = null)
|
||||||
|
{
|
||||||
|
return CreateSession(timeout, false, false, additionalHeaders);
|
||||||
|
}
|
||||||
|
|
||||||
|
public string? Accept { get; set; }
|
||||||
|
public string? Referrer { get; set; }
|
||||||
|
|
||||||
|
readonly object sync = new();
|
||||||
|
|
||||||
|
public partial Task<NetworkResult<string>> GetContentAsync(string url, StringHandler? process = null, CancellationToken token = default);
|
||||||
|
}
|
||||||
|
|
||||||
|
public class HttpResponseException : Exception
|
||||||
|
{
|
||||||
|
public int StatusCode { get; }
|
||||||
|
|
||||||
|
public string Url { get; }
|
||||||
|
|
||||||
|
public HttpResponseException(int code, string url) : base($"HTTP response failed with status code {code}, url: {url}")
|
||||||
|
{
|
||||||
|
StatusCode = code;
|
||||||
|
Url = url;
|
||||||
|
}
|
||||||
|
}
|
37
Network/NetworkResult.cs
Normal file
37
Network/NetworkResult.cs
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
public class NetworkResult<T>
|
||||||
|
{
|
||||||
|
public T? Result { get; }
|
||||||
|
|
||||||
|
public Exception? Exception { get; }
|
||||||
|
|
||||||
|
public NetworkResult(T? obj, Exception? exception = null)
|
||||||
|
{
|
||||||
|
Result = obj;
|
||||||
|
Exception = exception;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ThrowIfException(bool throwIfDefault = false)
|
||||||
|
{
|
||||||
|
if (Exception != null)
|
||||||
|
{
|
||||||
|
throw Exception;
|
||||||
|
}
|
||||||
|
if (throwIfDefault && Equals(Result, default))
|
||||||
|
{
|
||||||
|
throw new Exception($"Network result is {{{default}}}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static implicit operator NetworkResult<T>(T? obj)
|
||||||
|
{
|
||||||
|
return new(obj);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static implicit operator NetworkResult<T>(Exception ex)
|
||||||
|
{
|
||||||
|
return new(default, ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
46
Network/NetworkTask.cs
Normal file
46
Network/NetworkTask.cs
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
public abstract partial class NetworkTask : IDisposable
|
||||||
|
{
|
||||||
|
public string Url { get; }
|
||||||
|
|
||||||
|
public CancellationToken Token { get; }
|
||||||
|
|
||||||
|
public bool IsCancelled => disposed || Token.IsCancellationRequested == true;
|
||||||
|
|
||||||
|
bool disposed;
|
||||||
|
|
||||||
|
public void SetCancelled()
|
||||||
|
{
|
||||||
|
OnCancelled();
|
||||||
|
Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetException(Exception ex)
|
||||||
|
{
|
||||||
|
OnException(ex);
|
||||||
|
Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected abstract void OnCancelled();
|
||||||
|
|
||||||
|
protected abstract void OnException(Exception ex);
|
||||||
|
|
||||||
|
private partial void DisposingInternal();
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
if (!disposed)
|
||||||
|
{
|
||||||
|
disposed = true;
|
||||||
|
|
||||||
|
DisposingInternal();
|
||||||
|
}
|
||||||
|
|
||||||
|
GC.SuppressFinalize(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public delegate void StepHandler<T>(float progress, T? update);
|
||||||
|
|
||||||
|
public delegate string StringHandler(string original);
|
195
Network/Platforms/iOS/PlatformConnectivity.cs
Normal file
195
Network/Platforms/iOS/PlatformConnectivity.cs
Normal file
@ -0,0 +1,195 @@
|
|||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
|
using System.Net;
|
||||||
|
using CoreFoundation;
|
||||||
|
using CoreTelephony;
|
||||||
|
using SystemConfiguration;
|
||||||
|
|
||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
[SuppressMessage("Interoperability", "CA1416:Validate platform compatibility", Justification = "<Pending>")]
|
||||||
|
partial class Connectivity
|
||||||
|
{
|
||||||
|
static readonly Lazy<CTCellularData> cellularData = new(() => new CTCellularData());
|
||||||
|
|
||||||
|
static ReachabilityListener? listener;
|
||||||
|
|
||||||
|
private static partial NetworkStatus PlatformGetStatus()
|
||||||
|
{
|
||||||
|
var restricted = cellularData.Value.RestrictedState == CTCellularDataRestrictedState.Restricted;
|
||||||
|
|
||||||
|
var internetStatus = InternetConnectionStatus();
|
||||||
|
if (internetStatus == NetworkStatus.ReachableViaCarrierDataNetwork && !restricted)
|
||||||
|
{
|
||||||
|
return NetworkStatus.ReachableViaCarrierDataNetwork;
|
||||||
|
}
|
||||||
|
if (internetStatus == NetworkStatus.ReachableViaWiFiNetwork)
|
||||||
|
{
|
||||||
|
return NetworkStatus.ReachableViaWiFiNetwork;
|
||||||
|
}
|
||||||
|
|
||||||
|
var remoteStatus = RemoteHostStatus();
|
||||||
|
if (remoteStatus == NetworkStatus.ReachableViaCarrierDataNetwork && !restricted)
|
||||||
|
{
|
||||||
|
return NetworkStatus.ReachableViaCarrierDataNetwork;
|
||||||
|
}
|
||||||
|
if (remoteStatus == NetworkStatus.ReachableViaWiFiNetwork)
|
||||||
|
{
|
||||||
|
return NetworkStatus.ReachableViaWiFiNetwork;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NetworkStatus.NotReachable;
|
||||||
|
}
|
||||||
|
|
||||||
|
static NetworkStatus RemoteHostStatus()
|
||||||
|
{
|
||||||
|
using var remote = new NetworkReachability(hostName);
|
||||||
|
|
||||||
|
if (!remote.TryGetFlags(out var flags))
|
||||||
|
{
|
||||||
|
return NetworkStatus.NotReachable;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!IsReachableWithoutRequiringConnection(flags))
|
||||||
|
{
|
||||||
|
return NetworkStatus.NotReachable;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((flags & NetworkReachabilityFlags.IsWWAN) != 0)
|
||||||
|
{
|
||||||
|
return NetworkStatus.ReachableViaCarrierDataNetwork;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NetworkStatus.ReachableViaWiFiNetwork;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool IsReachableWithoutRequiringConnection(NetworkReachabilityFlags flags)
|
||||||
|
{
|
||||||
|
// Is it reachable with the current network configuration?
|
||||||
|
var isReachable = (flags & NetworkReachabilityFlags.Reachable) != 0;
|
||||||
|
|
||||||
|
// Do we need a connection to reach it?
|
||||||
|
var noConnectionRequired = (flags & NetworkReachabilityFlags.ConnectionRequired) == 0;
|
||||||
|
|
||||||
|
// Since the network stack will automatically try to get the WAN up,
|
||||||
|
// probe that
|
||||||
|
if ((flags & NetworkReachabilityFlags.IsWWAN) != 0)
|
||||||
|
{
|
||||||
|
noConnectionRequired = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return isReachable && noConnectionRequired;
|
||||||
|
}
|
||||||
|
|
||||||
|
static NetworkStatus InternetConnectionStatus()
|
||||||
|
{
|
||||||
|
var status = NetworkStatus.NotReachable;
|
||||||
|
|
||||||
|
var defaultNetworkAvailable = IsNetworkAvailable(out var flags);
|
||||||
|
|
||||||
|
// If the connection is reachable and no connection is required, then assume it's WiFi
|
||||||
|
if (defaultNetworkAvailable)
|
||||||
|
{
|
||||||
|
status = NetworkStatus.ReachableViaWiFiNetwork;
|
||||||
|
}
|
||||||
|
else if ((flags & NetworkReachabilityFlags.IsWWAN) != 0)
|
||||||
|
{
|
||||||
|
// If it's a WWAN connection..
|
||||||
|
status = NetworkStatus.ReachableViaCarrierDataNetwork;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the connection is on-demand or on-traffic and no user intervention
|
||||||
|
// is required, then assume WiFi.
|
||||||
|
if (((flags & NetworkReachabilityFlags.ConnectionOnDemand) != 0 || (flags & NetworkReachabilityFlags.ConnectionOnTraffic) != 0) &&
|
||||||
|
(flags & NetworkReachabilityFlags.InterventionRequired) == 0)
|
||||||
|
{
|
||||||
|
status = NetworkStatus.ReachableViaWiFiNetwork;
|
||||||
|
}
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static bool IsNetworkAvailable(out NetworkReachabilityFlags flags)
|
||||||
|
{
|
||||||
|
var ip = new IPAddress(0);
|
||||||
|
using var route = new NetworkReachability(ip);
|
||||||
|
|
||||||
|
if (!route.TryGetFlags(out flags))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return IsReachableWithoutRequiringConnection(flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static partial void StartListeners()
|
||||||
|
{
|
||||||
|
StopListeners();
|
||||||
|
|
||||||
|
listener = new ReachabilityListener();
|
||||||
|
listener.ReachabilityChanged += OnConnectivityChanged;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static partial void StopListeners()
|
||||||
|
{
|
||||||
|
if (listener == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
listener.ReachabilityChanged -= OnConnectivityChanged;
|
||||||
|
listener.Dispose();
|
||||||
|
listener = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
class ReachabilityListener : IDisposable
|
||||||
|
{
|
||||||
|
NetworkReachability? routeReachability;
|
||||||
|
NetworkReachability? remoteReachability;
|
||||||
|
|
||||||
|
public event Action? ReachabilityChanged;
|
||||||
|
|
||||||
|
public ReachabilityListener()
|
||||||
|
{
|
||||||
|
var ip = new IPAddress(0);
|
||||||
|
routeReachability = new NetworkReachability(ip);
|
||||||
|
routeReachability.SetNotification(OnChange);
|
||||||
|
routeReachability.Schedule(CFRunLoop.Main, CFRunLoop.ModeDefault);
|
||||||
|
|
||||||
|
remoteReachability = new NetworkReachability(hostName);
|
||||||
|
|
||||||
|
// Need to probe before we queue, or we wont get any meaningful values
|
||||||
|
// this only happens when you create NetworkReachability from a hostname
|
||||||
|
remoteReachability.TryGetFlags(out var flags);
|
||||||
|
|
||||||
|
remoteReachability.SetNotification(OnChange);
|
||||||
|
remoteReachability.Schedule(CFRunLoop.Main, CFRunLoop.ModeDefault);
|
||||||
|
|
||||||
|
cellularData.Value.RestrictionDidUpdateNotifier = new(OnRestrictedStateChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
routeReachability?.Dispose();
|
||||||
|
routeReachability = null;
|
||||||
|
remoteReachability?.Dispose();
|
||||||
|
remoteReachability = null;
|
||||||
|
|
||||||
|
cellularData.Value.RestrictionDidUpdateNotifier = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
void OnRestrictedStateChanged(CTCellularDataRestrictedState state)
|
||||||
|
{
|
||||||
|
ReachabilityChanged?.Invoke();
|
||||||
|
}
|
||||||
|
|
||||||
|
async void OnChange(NetworkReachabilityFlags flags)
|
||||||
|
{
|
||||||
|
// Add in artifical delay so the connection status has time to change
|
||||||
|
// else it will return true no matter what.
|
||||||
|
await Task.Delay(100);
|
||||||
|
|
||||||
|
ReachabilityChanged?.Invoke();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
237
Network/Platforms/iOS/PlatformNetworkHelper.cs
Normal file
237
Network/Platforms/iOS/PlatformNetworkHelper.cs
Normal file
@ -0,0 +1,237 @@
|
|||||||
|
using CoreGraphics;
|
||||||
|
using Foundation;
|
||||||
|
|
||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
partial class NetworkHelper : NSObject, INSUrlSessionDataDelegate
|
||||||
|
{
|
||||||
|
static bool IsResponseSuccess(NSHttpUrlResponse response)
|
||||||
|
{
|
||||||
|
return (int)response.StatusCode is >= 200 and < 300;
|
||||||
|
}
|
||||||
|
|
||||||
|
NSUrlSession? urlSession;
|
||||||
|
|
||||||
|
readonly Dictionary<NSUrlSessionTask, NetworkTask> tasks = new();
|
||||||
|
|
||||||
|
private NetworkHelper(int timeout, bool useCookie, bool waitsConnectivity = true, Dictionary<string, string>? additionalHeaders = null)
|
||||||
|
{
|
||||||
|
var configuration = NSUrlSessionConfiguration.EphemeralSessionConfiguration;
|
||||||
|
configuration.WaitsForConnectivity = waitsConnectivity;
|
||||||
|
configuration.RequestCachePolicy = NSUrlRequestCachePolicy.UseProtocolCachePolicy;
|
||||||
|
configuration.TimeoutIntervalForRequest = timeout;
|
||||||
|
if (additionalHeaders != null)
|
||||||
|
{
|
||||||
|
configuration.HttpAdditionalHeaders = NSDictionary.FromObjectsAndKeys(
|
||||||
|
additionalHeaders.Values.Select(v => FromObject(v)).ToArray(),
|
||||||
|
additionalHeaders.Keys.Select(k => FromObject(k)).ToArray());
|
||||||
|
}
|
||||||
|
if (useCookie)
|
||||||
|
{
|
||||||
|
configuration.HttpCookieStorage = NSHttpCookieStorage.SharedStorage;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
configuration.HttpShouldSetCookies = false;
|
||||||
|
}
|
||||||
|
if (proxyHost != null && proxyPort != null)
|
||||||
|
{
|
||||||
|
configuration.StrongConnectionProxyDictionary = new ProxyConfigurationDictionary
|
||||||
|
{
|
||||||
|
HttpEnable = true,
|
||||||
|
HttpsProxyHost = proxyHost,
|
||||||
|
HttpsProxyPort = proxyPort,
|
||||||
|
HttpProxyHost = proxyHost,
|
||||||
|
HttpProxyPort = proxyPort
|
||||||
|
};
|
||||||
|
}
|
||||||
|
urlSession = NSUrlSession.FromConfiguration(configuration, this, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Dispose(bool disposing)
|
||||||
|
{
|
||||||
|
if (disposing)
|
||||||
|
{
|
||||||
|
lock (sync)
|
||||||
|
{
|
||||||
|
foreach (var task in tasks.Values)
|
||||||
|
{
|
||||||
|
task.Dispose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (urlSession != null)
|
||||||
|
{
|
||||||
|
urlSession.FinishTasksAndInvalidate();
|
||||||
|
urlSession.Dispose();
|
||||||
|
urlSession = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
base.Dispose(disposing);
|
||||||
|
}
|
||||||
|
|
||||||
|
public partial Task<NetworkResult<string>> GetContentAsync(string url, StringHandler? process, CancellationToken token)
|
||||||
|
{
|
||||||
|
var uri = NSUrl.FromString(url) ?? throw new ArgumentNullException(nameof(url));
|
||||||
|
var request = new NSMutableUrlRequest(uri)
|
||||||
|
{
|
||||||
|
[AcceptHeader] = Accept ?? AcceptHttp,
|
||||||
|
[ReferrerHeader] = Referrer
|
||||||
|
};
|
||||||
|
|
||||||
|
var task = (urlSession?.CreateDataTask(request)) ?? throw new NullReferenceException("Url request is null");
|
||||||
|
var taskSource = new TaskCompletionSource<NetworkResult<string>>();
|
||||||
|
var stringTask = new StringTask(url, taskSource, token)
|
||||||
|
{
|
||||||
|
Process = process
|
||||||
|
};
|
||||||
|
lock (sync)
|
||||||
|
{
|
||||||
|
tasks.Add(task, stringTask);
|
||||||
|
}
|
||||||
|
task.Resume();
|
||||||
|
return taskSource.Task;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Task<NetworkResult<bool>> GetImageAsync(string url, string filePath, StepHandler<CGImage>? step = null, CancellationToken token = default)
|
||||||
|
{
|
||||||
|
var uri = NSUrl.FromString(url) ?? throw new ArgumentNullException(nameof(url));
|
||||||
|
var request = new NSMutableUrlRequest(uri)
|
||||||
|
{
|
||||||
|
[AcceptHeader] = Accept ?? AcceptImage,
|
||||||
|
[ReferrerHeader] = Referrer
|
||||||
|
};
|
||||||
|
|
||||||
|
var task = (urlSession?.CreateDataTask(request)) ?? throw new NullReferenceException("Url request is null");
|
||||||
|
var taskSource = new TaskCompletionSource<NetworkResult<bool>>();
|
||||||
|
var imageTask = new ImageTask(url, filePath, step != null, taskSource, token)
|
||||||
|
{
|
||||||
|
Step = step
|
||||||
|
};
|
||||||
|
lock (sync)
|
||||||
|
{
|
||||||
|
tasks.Add(task, imageTask);
|
||||||
|
}
|
||||||
|
task.Resume();
|
||||||
|
return taskSource.Task;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Task<NetworkResult<string>> PostAsync(string url, Action<NSMutableUrlRequest>? prepare = null, CancellationToken token = default)
|
||||||
|
{
|
||||||
|
var uri = NSUrl.FromString(url) ?? throw new ArgumentNullException(nameof(url));
|
||||||
|
var request = new NSMutableUrlRequest(uri)
|
||||||
|
{
|
||||||
|
HttpMethod = "POST",
|
||||||
|
[AcceptHeader] = Accept ?? AcceptImage,
|
||||||
|
[ReferrerHeader] = Referrer
|
||||||
|
};
|
||||||
|
prepare?.Invoke(request);
|
||||||
|
|
||||||
|
var task = (urlSession?.CreateDataTask(request)) ?? throw new NullReferenceException("Url request is null");
|
||||||
|
var taskSource = new TaskCompletionSource<NetworkResult<string>>();
|
||||||
|
var stringTask = new StringTask(url, taskSource, token);
|
||||||
|
lock (sync)
|
||||||
|
{
|
||||||
|
tasks.Add(task, stringTask);
|
||||||
|
}
|
||||||
|
task.Resume();
|
||||||
|
return taskSource.Task;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool TryGetTask(NSUrlSessionTask task, out NetworkTask? networkTask)
|
||||||
|
{
|
||||||
|
bool got;
|
||||||
|
lock (sync)
|
||||||
|
{
|
||||||
|
got = tasks.TryGetValue(task, out networkTask);
|
||||||
|
}
|
||||||
|
return got;
|
||||||
|
}
|
||||||
|
|
||||||
|
[Export("URLSession:dataTask:didReceiveResponse:completionHandler:")]
|
||||||
|
public void DidReceiveResponse(NSUrlSession session, NSUrlSessionDataTask dataTask, NSUrlResponse response, Action<NSUrlSessionResponseDisposition> completionHandler)
|
||||||
|
{
|
||||||
|
if (!TryGetTask(dataTask, out NetworkTask? networkTask) || networkTask == null)
|
||||||
|
{
|
||||||
|
completionHandler(NSUrlSessionResponseDisposition.Cancel);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (networkTask.IsCancelled)
|
||||||
|
{
|
||||||
|
completionHandler(NSUrlSessionResponseDisposition.Cancel);
|
||||||
|
networkTask.SetException(new TaskCanceledException($"Cancelled on response received: {dataTask.CurrentRequest?.Url}", null, networkTask.Token));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (response is NSHttpUrlResponse httpResponse)
|
||||||
|
{
|
||||||
|
if (IsResponseSuccess(httpResponse))
|
||||||
|
{
|
||||||
|
if (networkTask.OnResponsed(httpResponse))
|
||||||
|
{
|
||||||
|
completionHandler(NSUrlSessionResponseDisposition.Allow);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
completionHandler(NSUrlSessionResponseDisposition.Cancel);
|
||||||
|
networkTask.SetException(new TaskCanceledException($"Cancelled on task responsed: {dataTask.CurrentRequest?.Url}", null, networkTask.Token));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
completionHandler(NSUrlSessionResponseDisposition.Cancel);
|
||||||
|
networkTask.SetException(new HttpResponseException((int)httpResponse.StatusCode, httpResponse.Url.AbsoluteString ?? networkTask.Url));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
completionHandler(NSUrlSessionResponseDisposition.Cancel);
|
||||||
|
networkTask.SetException(new InvalidDataException($"Response is not instance of <NSHttpUrlResponse> but <{response?.GetType()}>"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[Export("URLSession:dataTask:didReceiveData:")]
|
||||||
|
public void DidReceiveData(NSUrlSession session, NSUrlSessionDataTask dataTask, NSData data)
|
||||||
|
{
|
||||||
|
if (!TryGetTask(dataTask, out NetworkTask? networkTask) || networkTask == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (networkTask.IsCancelled)
|
||||||
|
{
|
||||||
|
networkTask.SetException(new TaskCanceledException($"Cancelled on data received: {dataTask.CurrentRequest?.Url}", null, networkTask.Token));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (networkTask.Data == null)
|
||||||
|
{
|
||||||
|
networkTask.SetException(new InvalidDataException("<NSMutableData> is null"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
networkTask.Data.AppendData(data);
|
||||||
|
networkTask.OnReceived((int)data.Length);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Export("URLSession:task:didCompleteWithError:")]
|
||||||
|
public void DidCompleteWithError(NSUrlSession session, NSUrlSessionTask task, NSError error)
|
||||||
|
{
|
||||||
|
if (!TryGetTask(task, out NetworkTask? networkTask) || networkTask == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (networkTask.IsCancelled)
|
||||||
|
{
|
||||||
|
networkTask.SetException(new TaskCanceledException($"Cancelled on completed: {task.CurrentRequest?.Url}", null, networkTask.Token));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (networkTask.Data == null)
|
||||||
|
{
|
||||||
|
networkTask.SetException(new InvalidDataException("<NSMutableData> is null"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (error != null)
|
||||||
|
{
|
||||||
|
networkTask.SetException(new Exception(error.ToString()));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
networkTask.SetCompleted(task.Response as NSHttpUrlResponse);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
49
Network/Platforms/iOS/PlatformNetworkTask.cs
Normal file
49
Network/Platforms/iOS/PlatformNetworkTask.cs
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
using Foundation;
|
||||||
|
|
||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
partial class NetworkTask
|
||||||
|
{
|
||||||
|
public NSMutableData? Data => data;
|
||||||
|
|
||||||
|
NSMutableData? data;
|
||||||
|
|
||||||
|
public NetworkTask(string url, CancellationToken token)
|
||||||
|
{
|
||||||
|
Url = url;
|
||||||
|
Token = token;
|
||||||
|
|
||||||
|
data = new NSMutableData();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetCompleted(NSHttpUrlResponse? response)
|
||||||
|
{
|
||||||
|
OnCompleted(response);
|
||||||
|
Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual bool OnResponsed(NSHttpUrlResponse response)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public virtual void OnReceived(int length)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
protected abstract void OnCompleted(NSHttpUrlResponse? response);
|
||||||
|
|
||||||
|
private partial void DisposingInternal()
|
||||||
|
{
|
||||||
|
Disposing();
|
||||||
|
if (data != null)
|
||||||
|
{
|
||||||
|
data.Dispose();
|
||||||
|
data = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected virtual void Disposing()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
88
Network/Platforms/iOS/Tasks/ContentTask.cs
Normal file
88
Network/Platforms/iOS/Tasks/ContentTask.cs
Normal file
@ -0,0 +1,88 @@
|
|||||||
|
using Foundation;
|
||||||
|
|
||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
public abstract class ContentTask<T, TResult> : NetworkTask
|
||||||
|
{
|
||||||
|
public StepHandler<T>? Step { get; set; }
|
||||||
|
|
||||||
|
protected virtual bool AllowDefaultResult => false;
|
||||||
|
|
||||||
|
protected TaskCompletionSource<NetworkResult<TResult>>? taskSource;
|
||||||
|
|
||||||
|
protected long expectedContentLength;
|
||||||
|
|
||||||
|
protected ContentTask(string url, TaskCompletionSource<NetworkResult<TResult>> source, CancellationToken token) : base(url, token)
|
||||||
|
{
|
||||||
|
taskSource = source ?? throw new ArgumentNullException(nameof(source));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnCancelled()
|
||||||
|
{
|
||||||
|
taskSource?.TrySetCanceled();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnException(Exception ex)
|
||||||
|
{
|
||||||
|
taskSource?.TrySetResult(ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
public override bool OnResponsed(NSHttpUrlResponse response)
|
||||||
|
{
|
||||||
|
expectedContentLength = response.ExpectedContentLength;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected virtual T? Update(float progress)
|
||||||
|
{
|
||||||
|
return default;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void OnReceived(int length)
|
||||||
|
{
|
||||||
|
if (Step != null)
|
||||||
|
{
|
||||||
|
if (Data == null)
|
||||||
|
{
|
||||||
|
Step(0f, default);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
float progress = (float)Data.Length / expectedContentLength;
|
||||||
|
var update = Update(progress);
|
||||||
|
Step(progress, update);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected virtual TResult? Completed(NSHttpUrlResponse? response)
|
||||||
|
{
|
||||||
|
throw new NotImplementedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnCompleted(NSHttpUrlResponse? response)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var result = Completed(response);
|
||||||
|
if (result == null && !AllowDefaultResult)
|
||||||
|
{
|
||||||
|
throw new NullReferenceException("Result is null");
|
||||||
|
}
|
||||||
|
taskSource?.TrySetResult(result);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
OnException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Disposing()
|
||||||
|
{
|
||||||
|
if (taskSource?.Task.IsCanceled == false)
|
||||||
|
{
|
||||||
|
taskSource.TrySetCanceled();
|
||||||
|
taskSource = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
30
Network/Platforms/iOS/Tasks/DownloadTask.cs
Normal file
30
Network/Platforms/iOS/Tasks/DownloadTask.cs
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
using Foundation;
|
||||||
|
|
||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
public class DownloadTask<T> : ContentTask<T, bool>
|
||||||
|
{
|
||||||
|
public string FilePath { get; }
|
||||||
|
|
||||||
|
protected override bool AllowDefaultResult => true;
|
||||||
|
|
||||||
|
public DownloadTask(string url, string filePath, TaskCompletionSource<NetworkResult<bool>> source, CancellationToken token) : base(url, source, token)
|
||||||
|
{
|
||||||
|
FilePath = filePath;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override bool Completed(NSHttpUrlResponse? response)
|
||||||
|
{
|
||||||
|
if (Data == null)
|
||||||
|
{
|
||||||
|
throw new NullReferenceException("Data is null");
|
||||||
|
}
|
||||||
|
var result = Data.Save(FilePath, NSDataWritingOptions.Atomic, out NSError? error);
|
||||||
|
if (error != null)
|
||||||
|
{
|
||||||
|
throw new Exception(error?.ToString() ?? "UUnknown error while saving file");
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
14
Network/Platforms/iOS/Tasks/FileTask.cs
Normal file
14
Network/Platforms/iOS/Tasks/FileTask.cs
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
public class FileTask : DownloadTask<float>
|
||||||
|
{
|
||||||
|
public FileTask(string url, string filePath, TaskCompletionSource<NetworkResult<bool>> source, CancellationToken token) : base(url, filePath, source, token)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override float Update(float progress)
|
||||||
|
{
|
||||||
|
return progress;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
38
Network/Platforms/iOS/Tasks/ImageTask.cs
Normal file
38
Network/Platforms/iOS/Tasks/ImageTask.cs
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
using CoreGraphics;
|
||||||
|
using ImageIO;
|
||||||
|
|
||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
public class ImageTask : DownloadTask<CGImage>
|
||||||
|
{
|
||||||
|
CGImageSource? imageSource;
|
||||||
|
|
||||||
|
public ImageTask(string url, string filePath, bool createImage, TaskCompletionSource<NetworkResult<bool>> source, CancellationToken token = default) : base(url, filePath, source, token)
|
||||||
|
{
|
||||||
|
if (createImage)
|
||||||
|
{
|
||||||
|
imageSource = CGImageSource.CreateIncremental(null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override CGImage? Update(float progress)
|
||||||
|
{
|
||||||
|
if (imageSource == null || Data == null)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
imageSource.UpdateData(Data, Data.Length >= (uint)expectedContentLength);
|
||||||
|
return imageSource.CreateImage(0, null!);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void Disposing()
|
||||||
|
{
|
||||||
|
if (imageSource != null)
|
||||||
|
{
|
||||||
|
imageSource.Dispose();
|
||||||
|
imageSource = null;
|
||||||
|
}
|
||||||
|
base.Disposing();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
26
Network/Platforms/iOS/Tasks/StringTask.cs
Normal file
26
Network/Platforms/iOS/Tasks/StringTask.cs
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
using Foundation;
|
||||||
|
|
||||||
|
namespace Blahblah.Library.Network;
|
||||||
|
|
||||||
|
public class StringTask : ContentTask<string, string>
|
||||||
|
{
|
||||||
|
public StringHandler? Process { get; set; }
|
||||||
|
|
||||||
|
public StringTask(string url, TaskCompletionSource<NetworkResult<string>> source, CancellationToken token) : base(url, source, token)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override string? Completed(NSHttpUrlResponse? response)
|
||||||
|
{
|
||||||
|
if (Data == null)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
string s = NSString.FromData(Data, NSStringEncoding.UTF8);
|
||||||
|
if (Process == null)
|
||||||
|
{
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
return Process(s);
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user