From 7615b9236bda77f3f0701a7d1a7bb69fc2cda794 Mon Sep 17 00:00:00 2001 From: meeh <meeh@mail.i2p> Date: Tue, 18 Sep 2018 15:36:38 +0000 Subject: [PATCH] Adding all new code, removed a lot obsolete code and fixed import paths etc. Mac OS X launcher: * UI built on Swift * Why? * Apple seems to on purpose make it harder to get into Objective-C these days * Swift is compiled to native code, but has easiness of Javascript in programming * Perfect for the OS X UI, many guides & tutorials as well * "Backend" in Objective-C++ / C++14 * Why? * Originally written in Objective-C / C++14 with C++17 backports * Only for backend because of the time the development takes * Short summary of features: * Java * It can detect java from: * JAVA_HOME environment variable * "Internet Plug-Ins" Apple stuff * By the /usr/libexec/java_home binary helper * It can unpack a new version of I2P * Unpacks to ~/Library/I2P * Can check currently unpacked version in ~/Library/I2P via i2p.jar's "net.i2p.CoreVersion" * User Interface (a popover, see https://youtu.be/k8L3lQ5rUq0 for example of this concept) * Router control tab view * It can start the router * It can stop the router * It can detect already running router, then avoid fireing up one * It can show basic information about the router state & version * Log view tab (not yet done) * While left-click triggers popover, right-click draws a minimal context menu --- .../16x16-Itoopie Transparent.png | Bin .../macosx/{I2PLauncher => }/AppDelegate.h | 0 .../I2PLauncher/Base.lproj/UserInterfaces.xib | 40 +++++ .../I2PLauncher/I2PLauncher-Bridging-Header.h | 5 + launchers/macosx/I2PLauncher/Info.plist | 66 +++---- launchers/macosx/I2PLauncher/README.md | 11 -- .../macosx/I2PLauncher/Storyboard.storyboard | 162 ++++++++++++++++++ .../I2PLauncher/SwiftMainDelegate.swift | 104 +++++++++++ .../I2PLauncher/Utils/ArrayExtensions.swift | 29 ++++ .../I2PLauncher/Utils/DateTimeUtils.swift | 87 ++++++++++ .../I2PLauncher/Utils/EventMonitor.swift | 37 ++++ .../Utils/ReflectionFunctions.swift | 57 ++++++ .../I2PLauncher/Utils/StringExtensions.swift | 35 ++++ .../I2PLauncher/routermgmt/DetectJava.swift | 136 +++++++++++++++ .../routermgmt/I2PSubprocess.swift | 32 ++++ .../routermgmt/RouterDeployer.swift | 28 +++ .../RouterProcessStatus+ObjectiveC.swift | 18 ++ .../routermgmt/RouterProcessStatus.swift | 89 ++++++++++ .../I2PLauncher/routermgmt/RouterRunner.swift | 69 ++++++++ .../subprocesses/AppleStuffExceptionHandler.h | 12 ++ .../subprocesses/AppleStuffExceptionHandler.m | 22 +++ .../I2PLauncher/subprocesses/Error.swift | 22 +++ .../subprocesses/ExecutionResult.swift | 63 +++++++ .../subprocesses/Subprocess+CompactAPI.swift | 118 +++++++++++++ .../I2PLauncher/subprocesses/Subprocess.swift | 159 +++++++++++++++++ .../subprocesses/TaskPipeline.swift | 113 ++++++++++++ .../userinterface/LogViewController.swift | 24 +++ .../userinterface/PopoverViewController.swift | 118 +++++++++++++ .../userinterface/StatusBarController.swift | 120 +++++++++++++ .../{I2PLauncher => }/ItoopieTransparent.png | Bin .../macosx/{I2PLauncher => }/JavaHelper.h | 0 .../macosx/{I2PLauncher => }/RouterTask.h | 0 .../macosx/{I2PLauncher => }/RouterTask.mm | 0 .../macosx/{I2PLauncher => }/fullBuild.sh | 0 .../images/128x128-Itoopie Transparent.png | Bin .../images/128x128-Itoopie Transparent@2x.png | Bin .../images/16x16-Itoopie Transparent@2x.png | Bin .../images/256x256-Itoopie Transparent.png | Bin .../images/256x256-Itoopie Transparent@2x.png | Bin .../images/32x32-Itoopie Transparent.png | Bin .../images/32x32-Itoopie Transparent@2x.png | Bin .../images/512x512-Itoopie Transparent.png | Bin .../images/512x512-Itoopie Transparent@2x.png | Bin .../{I2PLauncher => }/images/AppIcon.icns | Bin .../{I2PLauncher => include}/PidWatcher.h | 0 .../{I2PLauncher => include}/portcheck.h | 0 launchers/macosx/{I2PLauncher => }/main.mm | 0 47 files changed, 1735 insertions(+), 41 deletions(-) rename launchers/macosx/{I2PLauncher => }/16x16-Itoopie Transparent.png (100%) rename launchers/macosx/{I2PLauncher => }/AppDelegate.h (100%) create mode 100644 launchers/macosx/I2PLauncher/Base.lproj/UserInterfaces.xib create mode 100644 launchers/macosx/I2PLauncher/I2PLauncher-Bridging-Header.h delete mode 100644 launchers/macosx/I2PLauncher/README.md create mode 100644 launchers/macosx/I2PLauncher/Storyboard.storyboard create mode 100644 launchers/macosx/I2PLauncher/SwiftMainDelegate.swift create mode 100644 launchers/macosx/I2PLauncher/Utils/ArrayExtensions.swift create mode 100644 launchers/macosx/I2PLauncher/Utils/DateTimeUtils.swift create mode 100644 launchers/macosx/I2PLauncher/Utils/EventMonitor.swift create mode 100644 launchers/macosx/I2PLauncher/Utils/ReflectionFunctions.swift create mode 100644 launchers/macosx/I2PLauncher/Utils/StringExtensions.swift create mode 100644 launchers/macosx/I2PLauncher/routermgmt/DetectJava.swift create mode 100644 launchers/macosx/I2PLauncher/routermgmt/I2PSubprocess.swift create mode 100644 launchers/macosx/I2PLauncher/routermgmt/RouterDeployer.swift create mode 100644 launchers/macosx/I2PLauncher/routermgmt/RouterProcessStatus+ObjectiveC.swift create mode 100644 launchers/macosx/I2PLauncher/routermgmt/RouterProcessStatus.swift create mode 100644 launchers/macosx/I2PLauncher/routermgmt/RouterRunner.swift create mode 100644 launchers/macosx/I2PLauncher/subprocesses/AppleStuffExceptionHandler.h create mode 100644 launchers/macosx/I2PLauncher/subprocesses/AppleStuffExceptionHandler.m create mode 100644 launchers/macosx/I2PLauncher/subprocesses/Error.swift create mode 100644 launchers/macosx/I2PLauncher/subprocesses/ExecutionResult.swift create mode 100644 launchers/macosx/I2PLauncher/subprocesses/Subprocess+CompactAPI.swift create mode 100644 launchers/macosx/I2PLauncher/subprocesses/Subprocess.swift create mode 100644 launchers/macosx/I2PLauncher/subprocesses/TaskPipeline.swift create mode 100644 launchers/macosx/I2PLauncher/userinterface/LogViewController.swift create mode 100644 launchers/macosx/I2PLauncher/userinterface/PopoverViewController.swift create mode 100644 launchers/macosx/I2PLauncher/userinterface/StatusBarController.swift rename launchers/macosx/{I2PLauncher => }/ItoopieTransparent.png (100%) rename launchers/macosx/{I2PLauncher => }/JavaHelper.h (100%) rename launchers/macosx/{I2PLauncher => }/RouterTask.h (100%) rename launchers/macosx/{I2PLauncher => }/RouterTask.mm (100%) rename launchers/macosx/{I2PLauncher => }/fullBuild.sh (100%) rename launchers/macosx/{I2PLauncher => }/images/128x128-Itoopie Transparent.png (100%) rename launchers/macosx/{I2PLauncher => }/images/128x128-Itoopie Transparent@2x.png (100%) rename launchers/macosx/{I2PLauncher => }/images/16x16-Itoopie Transparent@2x.png (100%) rename launchers/macosx/{I2PLauncher => }/images/256x256-Itoopie Transparent.png (100%) rename launchers/macosx/{I2PLauncher => }/images/256x256-Itoopie Transparent@2x.png (100%) rename launchers/macosx/{I2PLauncher => }/images/32x32-Itoopie Transparent.png (100%) rename launchers/macosx/{I2PLauncher => }/images/32x32-Itoopie Transparent@2x.png (100%) rename launchers/macosx/{I2PLauncher => }/images/512x512-Itoopie Transparent.png (100%) rename launchers/macosx/{I2PLauncher => }/images/512x512-Itoopie Transparent@2x.png (100%) rename launchers/macosx/{I2PLauncher => }/images/AppIcon.icns (100%) rename launchers/macosx/{I2PLauncher => include}/PidWatcher.h (100%) rename launchers/macosx/{I2PLauncher => include}/portcheck.h (100%) rename launchers/macosx/{I2PLauncher => }/main.mm (100%) diff --git a/launchers/macosx/I2PLauncher/16x16-Itoopie Transparent.png b/launchers/macosx/16x16-Itoopie Transparent.png similarity index 100% rename from launchers/macosx/I2PLauncher/16x16-Itoopie Transparent.png rename to launchers/macosx/16x16-Itoopie Transparent.png diff --git a/launchers/macosx/I2PLauncher/AppDelegate.h b/launchers/macosx/AppDelegate.h similarity index 100% rename from launchers/macosx/I2PLauncher/AppDelegate.h rename to launchers/macosx/AppDelegate.h diff --git a/launchers/macosx/I2PLauncher/Base.lproj/UserInterfaces.xib b/launchers/macosx/I2PLauncher/Base.lproj/UserInterfaces.xib new file mode 100644 index 0000000000..a733a1265a --- /dev/null +++ b/launchers/macosx/I2PLauncher/Base.lproj/UserInterfaces.xib @@ -0,0 +1,40 @@ +<?xml version="1.0" encoding="UTF-8"?> +<document type="com.apple.InterfaceBuilder3.Cocoa.XIB" version="3.0" toolsVersion="14113" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES" customObjectInstantitationMethod="direct"> + <dependencies> + <plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="14113"/> + </dependencies> + <objects> + <customObject id="-2" userLabel="File's Owner" customClass="NSApplication"> + <connections> + <outlet property="delegate" destination="Voe-Tx-rLC" id="GzC-gU-4Uq"/> + </connections> + </customObject> + <customObject id="-1" userLabel="First Responder" customClass="FirstResponder"/> + <customObject id="-3" userLabel="Application" customClass="NSObject"/> + <customObject id="Voe-Tx-rLC" customClass="AppDelegate"/> + <customObject id="YLy-65-1bz" customClass="NSFontManager"/> + <menu id="TTS-U7-WkT" userLabel="statusBarContextMenu"> + <items> + <menuItem title="Open Console" id="cOe-UL-1TW"> + <modifierMask key="keyEquivalentModifierMask"/> + <connections> + <action selector="openConsoleClicked:" target="-1" id="uR3-lb-ikG"/> + </connections> + </menuItem> + <menuItem title="Start I2P Router" id="RQ8-Q2-68A"> + <modifierMask key="keyEquivalentModifierMask"/> + <connections> + <action selector="startRouterClicked:" target="-1" id="Vl3-cC-77e"/> + </connections> + </menuItem> + <menuItem title="Exit" id="hsL-CH-m3C"> + <modifierMask key="keyEquivalentModifierMask"/> + <connections> + <action selector="quickClicked:" target="-1" id="hiW-5d-nBX"/> + </connections> + </menuItem> + </items> + <point key="canvasLocation" x="17" y="167"/> + </menu> + </objects> +</document> diff --git a/launchers/macosx/I2PLauncher/I2PLauncher-Bridging-Header.h b/launchers/macosx/I2PLauncher/I2PLauncher-Bridging-Header.h new file mode 100644 index 0000000000..a83635db3c --- /dev/null +++ b/launchers/macosx/I2PLauncher/I2PLauncher-Bridging-Header.h @@ -0,0 +1,5 @@ +// +// Use this file to import your target's public headers that you would like to expose to Swift. +// +#import "SBridge.h" +#import "AppleStuffExceptionHandler.h" diff --git a/launchers/macosx/I2PLauncher/Info.plist b/launchers/macosx/I2PLauncher/Info.plist index 1e0372a288..6ba8490e6a 100644 --- a/launchers/macosx/I2PLauncher/Info.plist +++ b/launchers/macosx/I2PLauncher/Info.plist @@ -3,47 +3,53 @@ <plist version="1.0"> <dict> <key>CFBundleDevelopmentRegion</key> - <string>English</string> + <string>$(DEVELOPMENT_LANGUAGE)</string> <key>CFBundleExecutable</key> - <string>I2PLauncher</string> - <key>NSHumanReadableCopyright</key> - <string>Public Domain</string> - <key>CFBundleGetInfoString</key> - <string>0.9.35-experimental</string> + <string>$(EXECUTABLE_NAME)</string> <key>CFBundleIconFile</key> - <string>images/AppIcon.icns</string> + <string></string> <key>CFBundleIdentifier</key> - <string>net.i2p.launcher</string> + <string>$(PRODUCT_BUNDLE_IDENTIFIER)</string> <key>CFBundleInfoDictionaryVersion</key> <string>6.0</string> <key>CFBundleName</key> - <string>I2P</string> + <string>$(PRODUCT_NAME)</string> <key>CFBundlePackageType</key> <string>APPL</string> <key>CFBundleShortVersionString</key> - <string>0.0.1</string> - <key>CFBundleSignature</key> - <string>I2P</string> + <string>1.0</string> + <key>CFBundleURLTypes</key> + <array> + <dict> + <key>CFBundleTypeRole</key> + <string>Editor</string> + <key>CFBundleURLIconFile</key> + <string>ItoopieTransparent</string> + <key>CFBundleURLName</key> + <string>http+i2p</string> + <key>CFBundleURLSchemes</key> + <array> + <string>http+i2p</string> + </array> + </dict> + </array> <key>CFBundleVersion</key> - <string>0.0.1</string> - <key>NSUserNotificationAlertStyle</key> - <string>alert</string> - <key>NSAppleScriptEnabled</key> + <string>1</string> + <key>LSMinimumSystemVersion</key> + <string>$(MACOSX_DEPLOYMENT_TARGET)</string> + <key>LSUIElement</key> <true/> - <key>CGDisableCoalescedUpdates</key> + <key>NSAppleScriptEnabled</key> <true/> - <key>LSMinimumSystemVersion</key> - <string>10.5</string> - <key>CFBundleDisplayName</key> - <string>I2P</string> - <key>LSMinimumSystemVersionByArchitecture</key> - <dict> - <key>i386</key> - <string>10.5.0</string> - <key>x86_64</key> - <string>10.6.0</string> - </dict> - <key>LSUIElement</key> - <string>1</string> + <key>NSHumanReadableCopyright</key> + <string>Copyright © 2018 The I2P Project. All rights reserved.</string> + <key>NSMainNibFile</key> + <string>UserInterfaces</string> + <key>NSPrincipalClass</key> + <string>NSApplication</string> + <key>NSServices</key> + <array> + <dict/> + </array> </dict> </plist> diff --git a/launchers/macosx/I2PLauncher/README.md b/launchers/macosx/I2PLauncher/README.md deleted file mode 100644 index e82338e834..0000000000 --- a/launchers/macosx/I2PLauncher/README.md +++ /dev/null @@ -1,11 +0,0 @@ -# The Objective-C++ part of the Mac OS X launcher - -## Why? - -Code signing, OS X integration. - -## Howto build? - -Preferred tool is [ninja](https://ninja-build.org/). A makefile is also available. - -Build with: `ninja` diff --git a/launchers/macosx/I2PLauncher/Storyboard.storyboard b/launchers/macosx/I2PLauncher/Storyboard.storyboard new file mode 100644 index 0000000000..11e2e37bd8 --- /dev/null +++ b/launchers/macosx/I2PLauncher/Storyboard.storyboard @@ -0,0 +1,162 @@ +<?xml version="1.0" encoding="UTF-8"?> +<document type="com.apple.InterfaceBuilder3.Cocoa.Storyboard.XIB" version="3.0" toolsVersion="14113" targetRuntime="MacOSX.Cocoa" propertyAccessControl="none" useAutolayout="YES"> + <dependencies> + <deployment identifier="macosx"/> + <plugIn identifier="com.apple.InterfaceBuilder.CocoaPlugin" version="14113"/> + <capability name="box content view" minToolsVersion="7.0"/> + <capability name="documents saved in the Xcode 8 format" minToolsVersion="8.0"/> + </dependencies> + <scenes> + <!--PopoverController--> + <scene sceneID="4eF-i8-6L0"> + <objects> + <viewController identifier="PopoverView" storyboardIdentifier="PopoverView" id="Ii7-Rb-Ls8" userLabel="PopoverController" customClass="PopoverViewController" customModule="I2PLauncher" customModuleProvider="target" sceneMemberID="viewController"> + <tabView key="view" allowsTruncatedLabels="NO" initialItem="Fle-u6-lgB" id="dmf-Go-6qY"> + <rect key="frame" x="0.0" y="0.0" width="450" height="300"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <font key="font" metaFont="system"/> + <tabViewItems> + <tabViewItem label="Router Status & Control" identifier="" id="Fle-u6-lgB"> + <view key="view" canDrawConcurrently="YES" id="xXg-Bt-RWR" customClass="RouterStatusView" customModule="I2PLauncher" customModuleProvider="target"> + <rect key="frame" x="10" y="33" width="430" height="254"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <subviews> + <box fixedFrame="YES" title="Router information and status" translatesAutoresizingMaskIntoConstraints="NO" id="e8C-qI-SCp"> + <rect key="frame" x="6" y="16" width="279" height="231"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <view key="contentView" id="zBB-wE-VXr"> + <rect key="frame" x="2" y="2" width="275" height="214"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <subviews> + <textField identifier="RouterStatusLabel" horizontalHuggingPriority="251" verticalHuggingPriority="750" fixedFrame="YES" translatesAutoresizingMaskIntoConstraints="NO" id="0Eo-re-5WK" userLabel="RouterStatusLabel"> + <rect key="frame" x="6" y="190" width="245" height="17"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" title="Router status: Unknown" id="m7V-Se-tnf"> + <font key="font" metaFont="system"/> + <color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/> + <color key="backgroundColor" name="controlColor" catalog="System" colorSpace="catalog"/> + </textFieldCell> + </textField> + <textField identifier="RouterVersionLabel" horizontalHuggingPriority="251" verticalHuggingPriority="750" fixedFrame="YES" translatesAutoresizingMaskIntoConstraints="NO" id="sGy-NV-gmH" userLabel="RouterVersionLabel"> + <rect key="frame" x="6" y="171" width="245" height="17"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" title="Router version: Unknown" id="Mda-Os-8O9"> + <font key="font" metaFont="system"/> + <color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/> + <color key="backgroundColor" name="controlColor" catalog="System" colorSpace="catalog"/> + </textFieldCell> + </textField> + <textField identifier="RouterStartedByLabel" horizontalHuggingPriority="251" verticalHuggingPriority="750" fixedFrame="YES" translatesAutoresizingMaskIntoConstraints="NO" id="tgG-IT-Ojg" userLabel="RouterStartedByLabel"> + <rect key="frame" x="6" y="152" width="245" height="17"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" title="Router started by launcher? No" id="WBg-nH-kwu"> + <font key="font" metaFont="system"/> + <color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/> + <color key="backgroundColor" name="controlColor" catalog="System" colorSpace="catalog"/> + </textFieldCell> + </textField> + <textField identifier="RouterUptimeLabel" horizontalHuggingPriority="251" verticalHuggingPriority="750" fixedFrame="YES" translatesAutoresizingMaskIntoConstraints="NO" id="f2W-Kc-cxB" userLabel="RouterUptimeLabel"> + <rect key="frame" x="6" y="133" width="245" height="17"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <textFieldCell key="cell" scrollable="YES" lineBreakMode="clipping" sendsActionOnEndEditing="YES" title="Router uptime: Unknown" id="uQ0-cW-tYL"> + <font key="font" metaFont="system"/> + <color key="textColor" name="labelColor" catalog="System" colorSpace="catalog"/> + <color key="backgroundColor" name="controlColor" catalog="System" colorSpace="catalog"/> + </textFieldCell> + </textField> + </subviews> + </view> + </box> + <box fixedFrame="YES" title="Quick Control" translatesAutoresizingMaskIntoConstraints="NO" id="IIP-Qi-7dp"> + <rect key="frame" x="287" y="16" width="150" height="231"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <view key="contentView" identifier="QuickControlView" id="D8V-d8-0wT"> + <rect key="frame" x="2" y="2" width="146" height="214"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <subviews> + <button identifier="startstopRouterButton" verticalHuggingPriority="750" fixedFrame="YES" translatesAutoresizingMaskIntoConstraints="NO" id="1eu-Qw-TD9" userLabel="start-stop-button"> + <rect key="frame" x="0.0" y="176" width="147" height="32"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <buttonCell key="cell" type="push" title="Start/Stop Router" bezelStyle="rounded" alignment="center" borderStyle="border" imageScaling="proportionallyDown" inset="2" id="OYN-sS-eQH"> + <behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/> + <font key="font" metaFont="system"/> + </buttonCell> + </button> + <button identifier="restartRouterButton" verticalHuggingPriority="750" fixedFrame="YES" translatesAutoresizingMaskIntoConstraints="NO" id="C1m-1t-Xiq" userLabel="restart-button"> + <rect key="frame" x="10" y="150" width="128" height="32"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <buttonCell key="cell" type="push" title="Restart Router" bezelStyle="rounded" alignment="center" borderStyle="border" imageScaling="proportionallyDown" inset="2" id="sih-uF-V06"> + <behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/> + <font key="font" metaFont="system"/> + </buttonCell> + </button> + <button identifier="openConsoleButton" verticalHuggingPriority="750" fixedFrame="YES" translatesAutoresizingMaskIntoConstraints="NO" id="XZi-oz-5gy" userLabel="open-console-button"> + <rect key="frame" x="11" y="123" width="126" height="32"/> + <autoresizingMask key="autoresizingMask" flexibleMaxX="YES" flexibleMinY="YES"/> + <buttonCell key="cell" type="push" title="Open Console" bezelStyle="rounded" alignment="center" borderStyle="border" imageScaling="proportionallyDown" inset="2" id="Yh8-lj-JFi"> + <behavior key="behavior" pushIn="YES" lightByBackground="YES" lightByGray="YES"/> + <font key="font" metaFont="system"/> + </buttonCell> + </button> + </subviews> + </view> + </box> + </subviews> + <connections> + <outlet property="quickControlView" destination="D8V-d8-0wT" id="4to-rN-2eL"/> + <outlet property="routerStartStopButton" destination="1eu-Qw-TD9" id="FLt-MK-y5u"/> + <outlet property="routerStartedByLabel" destination="tgG-IT-Ojg" id="dA0-3w-cuF"/> + <outlet property="routerStatusLabel" destination="0Eo-re-5WK" id="7dm-Et-eSn"/> + <outlet property="routerUptimeLabel" destination="f2W-Kc-cxB" id="4Ya-qU-eb3"/> + <outlet property="routerVersionLabel" destination="sGy-NV-gmH" id="tM5-4M-DKy"/> + </connections> + </view> + </tabViewItem> + <tabViewItem label="Router Log Viewer" identifier="" id="IFq-CR-cjD" customClass="LogViewerViewController" customModule="I2PLauncher" customModuleProvider="target"> + <view key="view" id="7U9-d7-IVr"> + <rect key="frame" x="10" y="33" width="430" height="254"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <subviews> + <scrollView identifier="LoggerTextScrollView" fixedFrame="YES" horizontalLineScroll="10" horizontalPageScroll="10" verticalLineScroll="10" verticalPageScroll="10" hasHorizontalScroller="NO" usesPredominantAxisScrolling="NO" translatesAutoresizingMaskIntoConstraints="NO" id="jca-Kn-cO2" userLabel="LoggerTextScrollView"> + <rect key="frame" x="11" y="17" width="409" height="228"/> + <autoresizingMask key="autoresizingMask"/> + <clipView key="contentView" ambiguous="YES" id="E5l-WA-qOn"> + <rect key="frame" x="1" y="1" width="407" height="226"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <subviews> + <textView identifier="LoggerTextView" ambiguous="YES" importsGraphics="NO" verticallyResizable="YES" usesFontPanel="YES" findStyle="panel" continuousSpellChecking="YES" allowsUndo="YES" usesRuler="YES" allowsNonContiguousLayout="YES" quoteSubstitution="YES" dashSubstitution="YES" spellingCorrection="YES" smartInsertDelete="YES" id="bgQ-8i-Xgb" userLabel="LoggerTextView"> + <rect key="frame" x="0.0" y="0.0" width="407" height="226"/> + <autoresizingMask key="autoresizingMask" widthSizable="YES" heightSizable="YES"/> + <color key="backgroundColor" white="1" alpha="1" colorSpace="calibratedWhite"/> + <size key="minSize" width="407" height="226"/> + <size key="maxSize" width="463" height="10000000"/> + <color key="insertionPointColor" white="0.0" alpha="1" colorSpace="calibratedWhite"/> + </textView> + </subviews> + <color key="backgroundColor" white="1" alpha="1" colorSpace="calibratedWhite"/> + </clipView> + <scroller key="horizontalScroller" hidden="YES" verticalHuggingPriority="750" doubleValue="1" horizontal="YES" id="MRF-Wt-zdZ"> + <rect key="frame" x="-100" y="-100" width="87" height="18"/> + <autoresizingMask key="autoresizingMask"/> + </scroller> + <scroller key="verticalScroller" verticalHuggingPriority="750" doubleValue="1" horizontal="NO" id="Xq6-ur-WuT"> + <rect key="frame" x="392" y="1" width="16" height="226"/> + <autoresizingMask key="autoresizingMask"/> + </scroller> + </scrollView> + </subviews> + </view> + <connections> + <outlet property="scrollView" destination="jca-Kn-cO2" id="qAi-hi-PsC"/> + <outlet property="textFieldView" destination="bgQ-8i-Xgb" id="SbC-0r-xPR"/> + </connections> + </tabViewItem> + </tabViewItems> + </tabView> + </viewController> + <customObject id="d8g-wS-Zts" userLabel="First Responder" customClass="NSResponder" sceneMemberID="firstResponder"/> + </objects> + <point key="canvasLocation" x="-823" y="166"/> + </scene> + </scenes> +</document> diff --git a/launchers/macosx/I2PLauncher/SwiftMainDelegate.swift b/launchers/macosx/I2PLauncher/SwiftMainDelegate.swift new file mode 100644 index 0000000000..c0bcefb337 --- /dev/null +++ b/launchers/macosx/I2PLauncher/SwiftMainDelegate.swift @@ -0,0 +1,104 @@ +// +// SwiftMainDelegate.swift +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation +import Cocoa + +@objc class SwiftMainDelegate : NSObject { + + //let statusItem = NSStatusBar.system().statusItem(withLength: NSSquareStatusItemLength ) + let statusBarController = StatusBarController() + let javaDetector = DetectJava() + static let objCBridge = SBridge() + + override init() { + super.init() + + self.javaDetector.findIt() + if (!javaDetector.isJavaFound()) { + print("Could not find java....") + terminate("No java..") + } + let javaBinPath = self.javaDetector.javaHome + print("Found java home = ", javaBinPath) + + let (portIsNotTaken, descPort) = RouterProcessStatus.checkTcpPortForListen(port: 7657) + if (!portIsNotTaken) { + RouterProcessStatus.isRouterRunning = true + RouterProcessStatus.isRouterChildProcess = false + print("I2P Router seems to be running") + } else { + RouterProcessStatus.isRouterRunning = false + print("I2P Router seems to NOT be running") + + } + + + } + + func findInstalledI2PVersion(jarPath: String, javaBin: String) { + var i2pPath = NSHomeDirectory() + i2pPath += "/Library/I2P" + var jExecPath:String = javaBin + + // Sometimes, home will return the binary, sometimes the actual home dir. This fixes the diverge. + // If JRE is detected, binary follows - if it's JDK, home follows. + if (jExecPath.hasSuffix("Home")) { + jExecPath += "/jre/bin/java" + } + + let subCmd = jExecPath + " -cp " + jarPath + " net.i2p.CoreVersion" + + var cmdArgs:[String] = ["-c", subCmd] + print(cmdArgs) + let sub:Subprocess = Subprocess.init(executablePath: "/bin/sh", arguments: cmdArgs) + let results:ExecutionResult = sub.execute(captureOutput: true)! + if (results.didCaptureOutput) { + print("captured output") + let i2pVersion = results.outputLines.first?.replace(target: "I2P Core version: ", withString: "") + print("I2P version detected: ",i2pVersion!) + RouterProcessStatus.routerVersion = i2pVersion + } else { + print("did NOT captured output") + + } + } + + @objc func applicationDidFinishLaunching() { + print("Hello from swift!") + var i2pPath = NSHomeDirectory() + i2pPath += "/Library/I2P" + + let javaBinPath = self.javaDetector.javaHome.replace(target: " ", withString: "\\ ") + + let fileManager = FileManager() + var ok = ObjCBool(true) + let doesI2PDirExists = fileManager.fileExists(atPath: i2pPath, isDirectory: &ok) + + if (!doesI2PDirExists) { + // Deploy + } + + let i2pJarPath = i2pPath + "/lib/i2p.jar" + + findInstalledI2PVersion(jarPath: i2pJarPath, javaBin: javaBinPath) + } + + @objc static func openLink(url: String) { + objCBridge.openUrl(url) + } + + @objc func applicationWillTerminate() { + // Shutdown stuff + } + + @objc func terminate(_ why: Any?) { + print("Stopping cause of ", why!) + } +} + diff --git a/launchers/macosx/I2PLauncher/Utils/ArrayExtensions.swift b/launchers/macosx/I2PLauncher/Utils/ArrayExtensions.swift new file mode 100644 index 0000000000..ccfc807b0c --- /dev/null +++ b/launchers/macosx/I2PLauncher/Utils/ArrayExtensions.swift @@ -0,0 +1,29 @@ +// +// ArrayExtension.swift +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + + +extension Array where Element: NSAttributedString { + func joined2(separator: NSAttributedString) -> NSAttributedString { + var isFirst = true + return self.reduce(NSMutableAttributedString()) { + (r, e) in + if isFirst { + isFirst = false + } else { + r.append(separator) + } + r.append(e) + return r + } + } + +} + + diff --git a/launchers/macosx/I2PLauncher/Utils/DateTimeUtils.swift b/launchers/macosx/I2PLauncher/Utils/DateTimeUtils.swift new file mode 100644 index 0000000000..471dffc241 --- /dev/null +++ b/launchers/macosx/I2PLauncher/Utils/DateTimeUtils.swift @@ -0,0 +1,87 @@ +// +// DateTimeUtils.swift +// I2PLauncher +// +// Created by Mikal Villa on 18/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + +extension Date { + init(date: NSDate) { + self.init(timeIntervalSinceReferenceDate: date.timeIntervalSinceReferenceDate) + } +} + +extension NSDate { + convenience init(date: Date) { + self.init(timeIntervalSinceReferenceDate: date.timeIntervalSinceReferenceDate) + } +} + +class DateTimeUtils { + static func timeAgoSinceDate(date:NSDate, numericDates:Bool) -> String { + let calendar = NSCalendar.current + let unitFlags: Set<Calendar.Component> = [.minute, .hour, .day, .weekOfYear, .month, .year, .second] + let now = NSDate() + let earliest = now.earlierDate(date as Date) + let latest = (earliest == now as Date) ? date : now + let components = calendar.dateComponents(unitFlags, from: earliest as Date, to: latest as Date) + + if (components.year! >= 2) { + return "\(components.year!) years ago" + } else if (components.year! >= 1){ + if (numericDates){ + return "1 year ago" + } else { + return "Last year" + } + } else if (components.month! >= 2) { + return "\(components.month!) months ago" + } else if (components.month! >= 1){ + if (numericDates){ + return "1 month ago" + } else { + return "Last month" + } + } else if (components.weekOfYear! >= 2) { + return "\(components.weekOfYear!) weeks ago" + } else if (components.weekOfYear! >= 1){ + if (numericDates){ + return "1 week ago" + } else { + return "Last week" + } + } else if (components.day! >= 2) { + return "\(components.day!) days ago" + } else if (components.day! >= 1){ + if (numericDates){ + return "1 day ago" + } else { + return "Yesterday" + } + } else if (components.hour! >= 2) { + return "\(components.hour!) hours ago" + } else if (components.hour! >= 1){ + if (numericDates){ + return "1 hour ago" + } else { + return "An hour ago" + } + } else if (components.minute! >= 2) { + return "\(components.minute!) minutes ago" + } else if (components.minute! >= 1){ + if (numericDates){ + return "1 minute ago" + } else { + return "A minute ago" + } + } else if (components.second! >= 3) { + return "\(components.second!) seconds ago" + } else { + return "Just now" + } + + } +} diff --git a/launchers/macosx/I2PLauncher/Utils/EventMonitor.swift b/launchers/macosx/I2PLauncher/Utils/EventMonitor.swift new file mode 100644 index 0000000000..008e0e44e8 --- /dev/null +++ b/launchers/macosx/I2PLauncher/Utils/EventMonitor.swift @@ -0,0 +1,37 @@ +// +// EventMonitor.swift +// I2PLauncher +// +// Created by Mikal Villa on 02/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation +import Cocoa + +public class EventMonitor { + private var monitor: Any? + private let mask: NSEvent.EventTypeMask + private let handler: (NSEvent?) -> Void + + public init(mask: NSEvent.EventTypeMask, handler: @escaping (NSEvent?) -> Void) { + self.mask = mask + self.handler = handler + } + + deinit { + stop() + } + + public func start() { + monitor = NSEvent.addGlobalMonitorForEvents(matching: mask, handler: handler) + } + + public func stop() { + if monitor != nil { + NSEvent.removeMonitor(monitor!) + monitor = nil + } + } +} + diff --git a/launchers/macosx/I2PLauncher/Utils/ReflectionFunctions.swift b/launchers/macosx/I2PLauncher/Utils/ReflectionFunctions.swift new file mode 100644 index 0000000000..dffbcf84bf --- /dev/null +++ b/launchers/macosx/I2PLauncher/Utils/ReflectionFunctions.swift @@ -0,0 +1,57 @@ +// +// ReflectionFunctions.swift +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + +class ReflectionFunctions { + + /// Given pointer to first element of a C array, invoke a function for each element + func enumerateCArray<T>(array: UnsafePointer<T>, count: UInt32, f: (UInt32, T) -> ()) { + var ptr = array + for i in 0..<count { + f(i, ptr.pointee) + ptr = ptr.successor() + } + } + + /// Return name for a method + func methodName(m: Method) -> String? { + let sel = method_getName(m) + let nameCString = sel_getName(sel) + return String(cString: nameCString!) + } + + /// Print the names for each method in a class + func printMethodNamesForClass(cls: AnyClass) { + var methodCount: UInt32 = 0 + let methodList = class_copyMethodList(cls, &methodCount) + if methodList != nil && methodCount > 0 { + enumerateCArray(array: methodList!, count: methodCount) { i, m in + let name = methodName(m: m!) ?? "unknown" + print("#\(i): \(name)") + } + + free(methodList) + } + } + + /// Print the names for each method in a class with a specified name + func printMethodNamesForClassNamed(classname: String) { + // NSClassFromString() is declared to return AnyClass!, but should be AnyClass? + let maybeClass: AnyClass? = NSClassFromString(classname) + if let cls: AnyClass = maybeClass { + printMethodNamesForClass(cls: cls) + } + else { + print("\(classname): no such class") + } + } +} + + + diff --git a/launchers/macosx/I2PLauncher/Utils/StringExtensions.swift b/launchers/macosx/I2PLauncher/Utils/StringExtensions.swift new file mode 100644 index 0000000000..1bd3f1898a --- /dev/null +++ b/launchers/macosx/I2PLauncher/Utils/StringExtensions.swift @@ -0,0 +1,35 @@ +// +// StringExtensions.swift +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + +extension String { + + func replace(target: String, withString: String) -> String + { + return self.replacingOccurrences(of: target, with: withString, options: NSString.CompareOptions.literal, range: nil) + } + + /// Returns an array of string obtained splitting self at each newline ("\n"). + /// If the last character is a newline, it will be ignored (no empty string + /// will be appended at the end of the array) + public func splitByNewline() -> [String] { + return self.split { $0 == "\n" }.map(String.init) + } + + /// Returns an array of string obtained splitting self at each space, newline or TAB character + public func splitByWhitespace() -> [String] { + let whitespaces = Set<Character>([" ", "\n", "\t"]) + return self.split { whitespaces.contains($0) }.map(String.init) + } + + public func splitByColon() -> [String] { + return self.split { $0 == ":" }.map(String.init) + } +} + diff --git a/launchers/macosx/I2PLauncher/routermgmt/DetectJava.swift b/launchers/macosx/I2PLauncher/routermgmt/DetectJava.swift new file mode 100644 index 0000000000..aa5d59ad02 --- /dev/null +++ b/launchers/macosx/I2PLauncher/routermgmt/DetectJava.swift @@ -0,0 +1,136 @@ +// +// DetectJava.swift +// JavaI2PWrapper +// +// Created by Mikal Villa on 24/03/2018. +// Copyright © 2018 I2P. All rights reserved. +// + +import Foundation + +class DetectJava : NSObject { + + var hasJRE : Bool = false + var userWantJRE : Bool = false + var userAcceptOracleEULA : Bool = false + + // Java checks + var javaHome: String = ""{ + + //Called before the change + willSet(newValue){ + print("DetectJava.javaHome will change from "+self.javaHome+" to "+newValue) + } + + //Called after the change + didSet{ + hasJRE = true + print("MDetectJava.javaHome did change from "+oldValue+" to "+self.javaHome) + } + }; + private var testedEnv : Bool = false + private var testedJH : Bool = false + private var testedDfl : Bool = false + + func isJavaFound() -> Bool { + if !(self.javaHome.isEmpty) { + return true + } + return false + } + + func findIt() { + print("Start with checking environment variable") + self.checkJavaEnvironmentVariable() + if !(self.javaHome.isEmpty) { + RouterProcessStatus.knownJavaBinPath = Optional.some(self.javaHome) + hasJRE = true + return + } + print("Checking default JRE install path") + self.checkDefaultJREPath() + if !(self.javaHome.isEmpty) { + RouterProcessStatus.knownJavaBinPath = Optional.some(self.javaHome) + hasJRE = true + return + } + print("Checking with the java_home util") + self.runJavaHomeCmd() + if !(self.javaHome.isEmpty) { + RouterProcessStatus.knownJavaBinPath = Optional.some(self.javaHome) + hasJRE = true + return + } + } + + func runJavaHomeCmd() { + let task = Process() + task.launchPath = "/usr/libexec/java_home" + task.arguments = [] + let pipe = Pipe() + task.standardOutput = pipe + let outHandle = pipe.fileHandleForReading + outHandle.waitForDataInBackgroundAndNotify() + + var obs1 : NSObjectProtocol! + obs1 = NotificationCenter.default.addObserver(forName: NSNotification.Name.NSFileHandleDataAvailable, + object: outHandle, queue: nil) { notification -> Void in + let data = outHandle.availableData + if data.count > 0 { + let str = NSString(data: data, encoding: String.Encoding.utf8.rawValue) + if (str != nil) { + let stringVal = str! as String + print("got output: "+stringVal) + self.javaHome = stringVal + } + // TODO: Found something, check it out + outHandle.waitForDataInBackgroundAndNotify() + } else { + print("EOF on stdout from process") + NotificationCenter.default.removeObserver(obs1) + // No JRE so far + } + } + + var obs2 : NSObjectProtocol! + obs2 = NotificationCenter.default.addObserver(forName: Process.didTerminateNotification, + object: task, queue: nil) { notification -> Void in + print("terminated") + NotificationCenter.default.removeObserver(obs2) + } + + task.launch() + task.waitUntilExit() + self.testedJH = true + } + + func checkDefaultJREPath() { + let defaultJREPath = "/Library/Internet Plug-Ins/JavaAppletPlugin.plugin/Contents/Home/bin/java" + if FileManager.default.fileExists(atPath: defaultJREPath) { + // Found it!! + self.javaHome = defaultJREPath + } + self.testedDfl = true + // No JRE here + } + + func getEnvironmentVar(_ name: String) -> String? { + guard let rawValue = getenv(name) else { return nil } + return String(utf8String: rawValue) + } + + func checkJavaEnvironmentVariable() { + let dic = ProcessInfo.processInfo.environment + //ProcessInfo.processInfo.environment["JAVA_HOME"] + if let javaHomeEnv = dic["JAVA_HOME"] { + // Maybe we got an JRE + print("Found JAVA_HOME with value:") + print(javaHomeEnv) + self.javaHome = javaHomeEnv + } + self.testedEnv = true + print("JAVA HOME is not set") + } + + +} diff --git a/launchers/macosx/I2PLauncher/routermgmt/I2PSubprocess.swift b/launchers/macosx/I2PLauncher/routermgmt/I2PSubprocess.swift new file mode 100644 index 0000000000..386543961e --- /dev/null +++ b/launchers/macosx/I2PLauncher/routermgmt/I2PSubprocess.swift @@ -0,0 +1,32 @@ +// +// I2PSubprocess.swift +// I2PLauncher +// +// Created by Mikal Villa on 18/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + +protocol I2PSubprocess { + var subprocessPath: String? { get set } + var arguments: String? { get set } + var timeWhenStarted: Date? { get set } + + func findJava(); + +} + +extension I2PSubprocess { + func toString() -> String { + let jp = self.subprocessPath! + let args = self.arguments! + + var presentation:String = "I2PSubprocess[ cmd=" + presentation += jp + presentation += " , args=" + presentation += args + presentation += "]" + return presentation + } +} diff --git a/launchers/macosx/I2PLauncher/routermgmt/RouterDeployer.swift b/launchers/macosx/I2PLauncher/routermgmt/RouterDeployer.swift new file mode 100644 index 0000000000..31e407ae3f --- /dev/null +++ b/launchers/macosx/I2PLauncher/routermgmt/RouterDeployer.swift @@ -0,0 +1,28 @@ +// +// RouterDeployer.swift +// I2PLauncher +// +// Created by Mikal Villa on 18/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + +class RouterDeployer: NSObject, I2PSubprocess { + var subprocessPath: String? + var timeWhenStarted: Date? + + var arguments: String? + + func findJava() { + // + } + + let javaBinaryPath = RouterProcessStatus.knownJavaBinPath + + let defaultFlagsForExtractorJob:[String] = [ + "-Xmx512M", + "-Xms128m", + "-Djava.awt.headless=true" + ] +} diff --git a/launchers/macosx/I2PLauncher/routermgmt/RouterProcessStatus+ObjectiveC.swift b/launchers/macosx/I2PLauncher/routermgmt/RouterProcessStatus+ObjectiveC.swift new file mode 100644 index 0000000000..cda1db5005 --- /dev/null +++ b/launchers/macosx/I2PLauncher/routermgmt/RouterProcessStatus+ObjectiveC.swift @@ -0,0 +1,18 @@ +// +// RouterProcessStatus+ObjectiveC.swift +// I2PLauncher +// +// Created by Mikal Villa on 18/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + +extension RouterProcessStatus { + static func createNewRouterProcess(i2pPath: String, javaBinPath: String) { + let bridge = SBridge() + let timeWhenStarted = Date() + RouterProcessStatus.routerStartedAt = timeWhenStarted + bridge.startupI2PRouter(i2pPath, javaBinPath: javaBinPath) + } +} diff --git a/launchers/macosx/I2PLauncher/routermgmt/RouterProcessStatus.swift b/launchers/macosx/I2PLauncher/routermgmt/RouterProcessStatus.swift new file mode 100644 index 0000000000..eb4ed37bd1 --- /dev/null +++ b/launchers/macosx/I2PLauncher/routermgmt/RouterProcessStatus.swift @@ -0,0 +1,89 @@ +// +// RouterProcessStatus.swift +// I2PLauncher +// +// Created by Mikal Villa on 18/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation +import AppKit + +class RouterProcessStatus : NSObject { + + /** + * + * Why the functions bellow? Because the Objective-C bridge is limited, it can't do Swift "static's" over it. + * + **/ + + func setRouterStatus(_ isRunning: Bool = false) { + RouterProcessStatus.isRouterRunning = isRunning + } + + func setRouterRanByUs(_ ranByUs: Bool = false) { + RouterProcessStatus.isRouterChildProcess = ranByUs + } + + func getRouterIsRunning() -> Bool? { + return RouterProcessStatus.isRouterRunning + } +} + +extension RouterProcessStatus { + static var isRouterRunning : Bool = false + static var isRouterChildProcess : Bool = false + static var routerVersion : String? = Optional.none + static var routerUptime : String? = Optional.none + static var routerStartedAt : Date? = Optional.none + static var knownJavaBinPath : String? = Optional.none + static var i2pDirectoryPath : String = NSHomeDirectory() + "/Library/I2P" + + static var knownRouterSubTaskRef : I2PSubprocess? = Optional.none + +} + + + + +extension RouterProcessStatus { + static func checkTcpPortForListen(port: in_port_t) -> (Bool, descr: String){ + + let socketFileDescriptor = socket(AF_INET, SOCK_STREAM, 0) + if socketFileDescriptor == -1 { + return (false, "SocketCreationFailed, \(descriptionOfLastError())") + } + + var addr = sockaddr_in() + let sizeOfSockkAddr = MemoryLayout<sockaddr_in>.size + addr.sin_len = __uint8_t(sizeOfSockkAddr) + addr.sin_family = sa_family_t(AF_INET) + addr.sin_port = Int(OSHostByteOrder()) == OSLittleEndian ? _OSSwapInt16(port) : port + addr.sin_addr = in_addr(s_addr: inet_addr("0.0.0.0")) + addr.sin_zero = (0, 0, 0, 0, 0, 0, 0, 0) + var bind_addr = sockaddr() + memcpy(&bind_addr, &addr, Int(sizeOfSockkAddr)) + + if Darwin.bind(socketFileDescriptor, &bind_addr, socklen_t(sizeOfSockkAddr)) == -1 { + let details = descriptionOfLastError() + release(socket: socketFileDescriptor) + return (false, "\(port), BindFailed, \(details)") + } + if listen(socketFileDescriptor, SOMAXCONN ) == -1 { + let details = descriptionOfLastError() + release(socket: socketFileDescriptor) + return (false, "\(port), ListenFailed, \(details)") + } + release(socket: socketFileDescriptor) + return (true, "\(port) is free for use") + } + + static func release(socket: Int32) { + Darwin.shutdown(socket, SHUT_RDWR) + close(socket) + } + static func descriptionOfLastError() -> String { + return String(cString: UnsafePointer(strerror(errno))) ?? "Error: \(errno)" + } +} + diff --git a/launchers/macosx/I2PLauncher/routermgmt/RouterRunner.swift b/launchers/macosx/I2PLauncher/routermgmt/RouterRunner.swift new file mode 100644 index 0000000000..50a7fd761d --- /dev/null +++ b/launchers/macosx/I2PLauncher/routermgmt/RouterRunner.swift @@ -0,0 +1,69 @@ +// +// RouterRunner.swift +// I2PLauncher +// +// Created by Mikal Villa on 18/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + +class RouterRunner: NSObject, I2PSubprocess { + + var subprocessPath: String? + var arguments: String? + var timeWhenStarted: Date? + + var currentRunningProcess: Subprocess? + var currentProcessResults: ExecutionResult? + + func findJava() { + self.subprocessPath = RouterProcessStatus.knownJavaBinPath + } + + let defaultStartupFlags:[String] = [ + "-Xmx512M", + "-Xms128m", + "-Djava.awt.headless=true", + "-Dwrapper.logfile=/tmp/router.log", + "-Dwrapper.logfile.loglevel=DEBUG", + "-Dwrapper.java.pidfile=/tmp/routerjvm.pid", + "-Dwrapper.console.loglevel=DEBUG" + ] + + private func subInit(cmdPath: String?, cmdArgs: String?) { + // Use this as common init + self.subprocessPath = cmdPath + self.arguments = cmdArgs + if (self.arguments?.isEmpty)! { + self.arguments = Optional.some(defaultStartupFlags.joined(separator: " ")) + }; + var newArgs:[String] = ["-c ", + self.subprocessPath!, + " ", + self.arguments!, + ] + self.currentRunningProcess = Optional.some(Subprocess.init(executablePath: "/bin/sh", arguments: newArgs)) + } + + init(cmdPath: String?, _ cmdArgs: String? = Optional.none) { + super.init() + self.subInit(cmdPath: cmdPath, cmdArgs: cmdArgs) + } + + init(coder: NSCoder) { + super.init() + self.subInit(cmdPath: Optional.none, cmdArgs: Optional.none) + } + + func execute() { + if (self.currentRunningProcess != Optional.none!) { + print("Already executing! Process ", self.toString()) + } + self.timeWhenStarted = Date() + RouterProcessStatus.routerStartedAt = self.timeWhenStarted + + self.currentProcessResults = self.currentRunningProcess?.execute(captureOutput: true) + } + +} diff --git a/launchers/macosx/I2PLauncher/subprocesses/AppleStuffExceptionHandler.h b/launchers/macosx/I2PLauncher/subprocesses/AppleStuffExceptionHandler.h new file mode 100644 index 0000000000..fc9fd98770 --- /dev/null +++ b/launchers/macosx/I2PLauncher/subprocesses/AppleStuffExceptionHandler.h @@ -0,0 +1,12 @@ +// +// AppleStuffExceptionHandler.h +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +#import <Foundation/Foundation.h> + +extern NSException* __nullable AppleStuffExecuteWithPossibleExceptionInBlock(dispatch_block_t _Nonnull block); + diff --git a/launchers/macosx/I2PLauncher/subprocesses/AppleStuffExceptionHandler.m b/launchers/macosx/I2PLauncher/subprocesses/AppleStuffExceptionHandler.m new file mode 100644 index 0000000000..951a38355b --- /dev/null +++ b/launchers/macosx/I2PLauncher/subprocesses/AppleStuffExceptionHandler.m @@ -0,0 +1,22 @@ +// +// AppleStuffExceptionHandler.m +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +#import "AppleStuffExceptionHandler.h" + +NSException* __nullable AppleStuffExecuteWithPossibleExceptionInBlock(dispatch_block_t _Nonnull block) { + + @try { + if (block != nil) { + block(); + } + } + @catch (NSException *exception) { + return exception; + } + return nil; +} diff --git a/launchers/macosx/I2PLauncher/subprocesses/Error.swift b/launchers/macosx/I2PLauncher/subprocesses/Error.swift new file mode 100644 index 0000000000..cc3cfe02a0 --- /dev/null +++ b/launchers/macosx/I2PLauncher/subprocesses/Error.swift @@ -0,0 +1,22 @@ +// +// Error.swift +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + +public class Error { + + /// Prints to console the arguments and exits with status 1 + static func die(arguments: Any...) -> Never { + let output = "ERROR: " + arguments.reduce("") { $0 + "\($1) " } + let trimOutput = output.trimmingCharacters(in: CharacterSet.whitespaces) + "\n" + let stderr = FileHandle.standardError + stderr.write(trimOutput.data(using: String.Encoding.utf8)!) + exit(1) + } +} + diff --git a/launchers/macosx/I2PLauncher/subprocesses/ExecutionResult.swift b/launchers/macosx/I2PLauncher/subprocesses/ExecutionResult.swift new file mode 100644 index 0000000000..f482bfd206 --- /dev/null +++ b/launchers/macosx/I2PLauncher/subprocesses/ExecutionResult.swift @@ -0,0 +1,63 @@ +// +// ExecutionResult.swift +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + +public struct ExecutionResult { + + /// Whether the output was captured + public let didCaptureOutput : Bool + + /// The return status of the last subprocess + public var status: Int32 { + return pipelineStatuses.last! + } + + /// Return status of all subprocesses in the pipeline + public let pipelineStatuses : [Int32] + + /// The output of the subprocess. Empty string if no output was produced or not captured + public let output: String + + /// The error output of the last subprocess. Empty string if no error output was produced or not captured + public var errors : String { + return pipelineErrors?.last ?? "" + } + + /// The error output of all subprocesses in the pipeline. Empty string if no error output was produced or not captured + public let pipelineErrors : [String]? + + /// The output, split by newline + /// - SeeAlso: `output` + public var outputLines : [String] { + return self.output.splitByNewline() + } + + /// The error output, split by newline + /// - SeeAlso: `output` + public var errorsLines : [String] { + return self.errors.splitByNewline() + } + + /// An execution result where no output was captured + init(pipelineStatuses: [Int32]) { + self.pipelineStatuses = pipelineStatuses + self.didCaptureOutput = false + self.pipelineErrors = nil + self.output = "" + } + + /// An execution result where output was captured + init(pipelineStatuses: [Int32], pipelineErrors : [String], output : String) { + self.pipelineStatuses = pipelineStatuses + self.pipelineErrors = pipelineErrors + self.output = output + self.didCaptureOutput = true + } +} + diff --git a/launchers/macosx/I2PLauncher/subprocesses/Subprocess+CompactAPI.swift b/launchers/macosx/I2PLauncher/subprocesses/Subprocess+CompactAPI.swift new file mode 100644 index 0000000000..cb472e0c30 --- /dev/null +++ b/launchers/macosx/I2PLauncher/subprocesses/Subprocess+CompactAPI.swift @@ -0,0 +1,118 @@ +// +// Subprocess+CompactAPI.swift +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + + +// MARK: - Compact API +extension Subprocess { + + /** + Executes a subprocess and wait for completion, returning the output. If there is an error in creating the task, + it immediately exits the process with status 1 + - returns: the output as a String + - note: in case there is any error in executing the process or creating the task, it will halt execution. Use + the constructor and `output` instance method for a more graceful error handling + */ + public static func output( + executablePath: String, + _ arguments: String..., + workingDirectory: String = ".") -> String { + + let process = Subprocess.init(executablePath: executablePath, arguments: arguments, workingDirectory: workingDirectory) + guard let result = process.execute(captureOutput: true) else { + Error.die(arguments: "Can't execute \"\(process)\"") + } + if result.status != 0 { + let errorLines = result.errors == "" ? "" : "\n" + result.errors + Error.die(arguments: "Process \"\(process)\" returned status \(result.status)", errorLines) + } + return result.output + } + + /** + Executes a subprocess and wait for completion, returning the execution result. If there is an error in creating the task, + it immediately exits the process with status 1 + - returns: the execution result + - note: in case there is any error in executing the process or creating the task, it will halt execution. Use + the constructor and `execute` instance method for a more graceful error handling + */ + public static func execute( + executablePath: String, + _ arguments: String..., + workingDirectory: String = ".") -> ExecutionResult { + + let process = Subprocess.init(executablePath: executablePath, arguments: arguments, workingDirectory: workingDirectory) + guard let result = process.execute(captureOutput: true) else { + Error.die(arguments: "Can't execute \"\(process)\"") + } + return result + } + + /** + Executes a subprocess and wait for completion, returning the output as an array of lines. If there is an error + in creating or executing the task, it immediately exits the process with status 1 + - returns: the output as a String + - note: in case there is any error in executing the process or creating the task, it will halt execution. Use + the constructor and `output` instance method for a more graceful error handling + */ + public static func outputLines( + executablePath: String, + _ arguments: String..., + workingDirectory: String = ".") -> [String] { + + let process = Subprocess.init(executablePath: executablePath, arguments: arguments, workingDirectory: workingDirectory) + guard let result = process.execute(captureOutput: true) else { + Error.die(arguments: "Can't execute \"\(process)\"") + } + if result.status != 0 { + let errorLines = result.errors == "" ? "" : "\n" + result.errors + Error.die(arguments: "Process \"\(process)\" returned status \(result.status)", errorLines) + } + return result.outputLines + } + + /** + Executes a subprocess and wait for completion, returning the exit status. If there is an error in creating the task, + it immediately exits the process with status 1 + - returns: the output as a String + - note: in case there is any error in launching the process or creating the task, it will halt execution. Use + the constructor and the `run` instance method for a more graceful error handling + */ + public static func run( + executablePath: String, + _ arguments: String..., + workingDirectory: String = ".") -> Int32 { + + let process = Subprocess.init(executablePath: executablePath, arguments: arguments, workingDirectory: workingDirectory) + guard let result = process.run() else { + Error.die(arguments: "Can't execute \"\(process)\"") + } + return result + } + + /** + Executes a subprocess and wait for completion. If there is an error in creating the task, or if the tasks + returns an exit status other than 0, it immediately exits the process with status 1 + - note: in case there is any error in launching the process or creating the task, or if the task exists with a exit status other than 0, it will halt execution. Use + the constructor and `run` instance method for a more graceful error handling + */ + public static func runOrDie( + executablePath: String, + _ arguments: String..., + workingDirectory: String = ".") { + + let process = Subprocess.init(executablePath: executablePath, arguments: arguments, workingDirectory: workingDirectory) + guard let result = process.run() else { + Error.die(arguments: "Can't execute \"\(process)\"") + } + if result != 0 { + Error.die(arguments: "Process \"\(process)\" returned status \(result)") + } + } +} diff --git a/launchers/macosx/I2PLauncher/subprocesses/Subprocess.swift b/launchers/macosx/I2PLauncher/subprocesses/Subprocess.swift new file mode 100644 index 0000000000..2b83e747a2 --- /dev/null +++ b/launchers/macosx/I2PLauncher/subprocesses/Subprocess.swift @@ -0,0 +1,159 @@ +// +// Subprocess.swift +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + + +public class Subprocess { + /// The path to the executable + let executablePath : String + + /// Arguments to pass to the executable + let arguments : [String] + + /// Working directory for the executable + let workingDirectory : String + + /// Process to pipe to, if any + let pipeDestination : Subprocess? + + public convenience init( + executablePath: String, + arguments: [String] = [], + workingDirectory: String = "." + ) { + self.init(executablePath: executablePath, + arguments: arguments, + workingDirectory: workingDirectory, + pipeTo: nil) + } + + public init( + executablePath: String, + arguments: [String] = [], + workingDirectory: String = ".", + pipeTo: Subprocess? + ) { + self.executablePath = executablePath + self.arguments = arguments + self.workingDirectory = workingDirectory + self.pipeDestination = pipeTo + } + + /** + Returns a subprocess ready to be executed + - SeeAlso: init(executablePath:arguments:workingDirectory) + */ + public convenience init( + _ executablePath: String, + _ arguments: String..., + workingDirectory: String = ".") { + self.init(executablePath: executablePath, arguments: arguments, workingDirectory: workingDirectory, pipeTo: nil) + } +} + +// Public API +extension Subprocess { + + /** + Executes the subprocess and wait for completition, returning the exit status + - returns: the termination status, or nil if it was not possible to execute the process + */ + public func run() -> Int32? { + return self.execute(captureOutput: false)?.status + } + + /** + Executes the subprocess and wait for completion, returning the output + - returns: the output of the process, or nil if it was not possible to execute the process + - warning: the entire output will be stored in a String in memory + */ + public func output() -> String? { + return self.execute(captureOutput: true)?.output + } + + /** + Executes the subprocess and wait for completition, returning the exit status + - returns: the execution result, or nil if it was not possible to execute the process + */ + public func execute(captureOutput: Bool = false) -> ExecutionResult? { + return buildPipeline(captureOutput: captureOutput).run() + } +} + +// Piping of STDIN, STDERR and STDOUT +extension Subprocess { + + /// Pipes the output to this process to another process. + /// Will return a new subprocess, you should execute that subprocess to + /// run the entire pipe + public func pipe(to destination: Subprocess) -> Subprocess { + let downstreamProcess : Subprocess + if let existingPipe = self.pipeDestination { + downstreamProcess = existingPipe.pipe(to: destination) + } else { + downstreamProcess = destination + } + return Subprocess(executablePath: self.executablePath, arguments: self.arguments, workingDirectory: self.workingDirectory, pipeTo: downstreamProcess) + } +} + +public func | (lhs: Subprocess, rhs: Subprocess) -> Subprocess { + return lhs.pipe(to: rhs) +} + +public func | (lhs: String, rhs: String) -> String { + return "(\(lhs)\(rhs))" +} + +// MARK: - Process execution +public enum SubprocessError : LocalizedError { + case Error(status: Int, message: String) +} + +extension Subprocess { + + /// Returns the task to execute + private func task() -> Process { + let task = Process() + task.launchPath = self.executablePath + task.arguments = self.arguments + task.currentDirectoryPath = self.workingDirectory + return task + } + + /// Returns the task pipeline for all the downstream processes + public func buildPipeline(captureOutput: Bool, input: AnyObject? = nil) -> TaskPipeline { + let task = self.task() + + if let inPipe = input { + task.standardInput = inPipe + } + + if let downstreamProcess = self.pipeDestination { + let downstreamPipeline = downstreamProcess.buildPipeline(captureOutput: captureOutput, input: task.standardOutput as AnyObject) + return downstreamPipeline.addToHead(task: task) + } + return TaskPipeline(task: task, captureOutput: captureOutput) + } +} + +// Description for pretty print etc. +extension Subprocess : CustomStringConvertible { + + public var description : String { + return self.executablePath + + (self.arguments.count > 0 + ? " " + self.arguments + .map { $0.replace(target: "\\ ", withString: " ") } + .joined(separator: " ") + : "" + ) + + (self.pipeDestination != nil ? " | " + self.pipeDestination!.description : "" ) + } +} diff --git a/launchers/macosx/I2PLauncher/subprocesses/TaskPipeline.swift b/launchers/macosx/I2PLauncher/subprocesses/TaskPipeline.swift new file mode 100644 index 0000000000..a960dd3f44 --- /dev/null +++ b/launchers/macosx/I2PLauncher/subprocesses/TaskPipeline.swift @@ -0,0 +1,113 @@ +// +// TaskPipeline.swift +// I2PLauncher +// +// Created by Mikal Villa on 17/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation + + +// Extend the stdlib with the extension bellow +extension Process { + /// Launches a task, captures any objective-c exception and relaunches it as Swift error + public func launchCapturingExceptions() throws { + if let exception = AppleStuffExecuteWithPossibleExceptionInBlock({ + self.launch() + }) { + let reason = exception.reason ?? "unknown error" + throw SubprocessError.Error(status: -1, message: reason) + } + } +} + +/// A pipeline of tasks, connected in a cascade pattern with pipes +public struct TaskPipeline { + + /// List of tasks in the pipeline + let tasks: [Process] + + /// Output pipe + let outputPipe: Pipe? + + /// Whether the pipeline should capture output to stdErr and stdOut + let captureOutput : Bool + + /// Adds a task to the head of the pipeline, that is, the task will provide the input + /// for the first task currently on the head of the pipeline + func addToHead(task: Process) -> TaskPipeline { + guard let firstTask = tasks.first else { + fatalError("Expecting at least one task") + } + let inoutPipe = Pipe() + firstTask.standardInput = inoutPipe + task.standardOutput = inoutPipe + + var errorPipe : Pipe? + if self.captureOutput { + errorPipe = Pipe() + task.standardError = errorPipe + } + return TaskPipeline(tasks: [task] + self.tasks, outputPipe: self.outputPipe, captureOutput: self.captureOutput) + } + + /// Start all tasks in the pipeline, then wait for them to complete + /// - returns: the return status of the last process in the pipe, or nil if there was an error + func run() -> ExecutionResult? { + + let runTasks = launchAndReturnNotFailedTasks() + if runTasks.count != self.tasks.count { + // dropped a task? it's because it failed to start, so error + return nil + } + runTasks.forEach { $0.waitUntilExit() } + + // exit status + let exitStatuses = runTasks.map { $0.terminationStatus } + guard captureOutput else { + return ExecutionResult(pipelineStatuses: exitStatuses) + } + + // output + let errorOutput = runTasks.map { task -> String in + guard let errorPipe = task.standardError as? Pipe else { return "" } + let readData = errorPipe.fileHandleForReading.readDataToEndOfFile() + return String(data: readData, encoding: String.Encoding.utf8)! + } + let output = String(data: self.outputPipe!.fileHandleForReading.readDataToEndOfFile(), encoding: String.Encoding.utf8)! + return ExecutionResult(pipelineStatuses: exitStatuses, pipelineErrors: errorOutput, output: output) + } + + /// Run all tasks and return the tasks that did not fail to launch + private func launchAndReturnNotFailedTasks() -> [Process] { + return self.tasks.flatMap { task -> Process? in + do { + try task.launchCapturingExceptions() + return task + } catch { + return nil + } + } + } + + init(task: Process, captureOutput: Bool) { + self.tasks = [task] + self.captureOutput = captureOutput + if captureOutput { + self.outputPipe = Pipe() + task.standardOutput = self.outputPipe + let errorPipe = Pipe() + task.standardError = errorPipe + } else { + self.outputPipe = nil + } + } + + private init(tasks: [Process], outputPipe: Pipe?, captureOutput: Bool) { + self.tasks = tasks + self.outputPipe = outputPipe + self.captureOutput = captureOutput + } +} + diff --git a/launchers/macosx/I2PLauncher/userinterface/LogViewController.swift b/launchers/macosx/I2PLauncher/userinterface/LogViewController.swift new file mode 100644 index 0000000000..b8abcfc7ce --- /dev/null +++ b/launchers/macosx/I2PLauncher/userinterface/LogViewController.swift @@ -0,0 +1,24 @@ +// +// LogViewController.swift +// I2PLauncher +// +// Created by Mikal Villa on 18/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Foundation +import AppKit + +class LogViewerViewController : NSTabViewItem { + + @IBOutlet var scrollView: NSScrollView? + @IBOutlet var textFieldView: NSTextView? + + + +} + +class LogViewController { + +} + diff --git a/launchers/macosx/I2PLauncher/userinterface/PopoverViewController.swift b/launchers/macosx/I2PLauncher/userinterface/PopoverViewController.swift new file mode 100644 index 0000000000..3c1923e4f3 --- /dev/null +++ b/launchers/macosx/I2PLauncher/userinterface/PopoverViewController.swift @@ -0,0 +1,118 @@ +// +// PopoverViewController.swift +// I2PLauncher +// +// Created by Mikal Villa on 18/09/2018. +// Copyright © 2018 The I2P Project. All rights reserved. +// + +import Cocoa + +class PopoverViewController: NSViewController { + + required init?(coder: NSCoder) { + super.init(coder: coder) + //super.init(nibName: "UserInterfaces", bundle: Bundle.main)! + //let nib = NSNib(nibNamed: "UserInterfaces", bundle: Bundle.main) + + } + + + override func viewDidLoad() { + super.viewDidLoad() + // Do view setup here. + } + + + +} + +@objc class RouterStatusView : NSView { + static var instance: RouterStatusView? + + static func getInstance() -> RouterStatusView? { + if (self.instance != Optional.none) { + return RouterStatusView.instance + } + return Optional.none + } + + @IBOutlet var routerStatusLabel: NSTextField? + @IBOutlet var routerVersionLabel: NSTextField? + @IBOutlet var routerStartedByLabel: NSTextField? + @IBOutlet var routerUptimeLabel: NSTextField? + + @IBOutlet var quickControlView: NSView? + @IBOutlet var routerStartStopButton: NSButton? + + + @objc func actionBtnStartRouter(_ sender: Any?) { + NSLog("START ROUTER") + (sender as! NSButton).cell?.stringValue = "Stop Router" + let timeWhenStarted = Date() + RouterProcessStatus.routerStartedAt = timeWhenStarted + SwiftMainDelegate.objCBridge.startupI2PRouter(RouterProcessStatus.i2pDirectoryPath, javaBinPath: RouterProcessStatus.knownJavaBinPath!) + } + @objc func actionBtnStopRouter(_ sender: Any?) { + NSLog("STOP ROUTER") + } + @objc func actionBtnRestartRouter(sender: Any?) {} + + override func viewWillDraw() { + super.viewWillDraw() + if (RouterStatusView.instance == Optional.none) { + RouterStatusView.instance = self + } + self.setRouterStatusLabelText() + } + + func setRouterStatusLabelText() { + if (RouterProcessStatus.isRouterRunning) { + routerStatusLabel?.cell?.stringValue = "Router status: Running" + routerStartStopButton?.action = #selector(self.actionBtnStopRouter(_:)) + } else { + routerStatusLabel?.cell?.stringValue = "Router status: Not running" + routerStartStopButton?.action = #selector(self.actionBtnStartRouter(_:)) + } + routerStartStopButton?.needsDisplay = true + routerStartStopButton?.target = self + quickControlView?.needsDisplay = true + + if (RouterProcessStatus.routerVersion?.isEmpty)! { + routerVersionLabel?.cell?.stringValue = "Router version: Still unknown" + } else { + routerVersionLabel?.cell?.stringValue = "Router version: " + RouterProcessStatus.routerVersion! + } + if (RouterProcessStatus.routerStartedAt != Optional.none) { + routerUptimeLabel?.cell?.stringValue = "Router has runned for " + DateTimeUtils.timeAgoSinceDate(date: NSDate(date: RouterProcessStatus.routerStartedAt!), numericDates: false) + } + } + + init() { + let c = NSCoder() + super.init(coder: c)! + self.setRouterStatusLabelText() + } + + required init?(coder decoder: NSCoder) { + super.init(coder: decoder) + self.setRouterStatusLabelText() + } + +} + + +extension PopoverViewController { + static func freshController() -> PopoverViewController { + let storyboard = NSStoryboard(name: "Storyboard", bundle: Bundle.main) + //2. + let identifier = NSStoryboard.SceneIdentifier(string: "PopoverView") + //3. + guard let viewcontroller = storyboard.instantiateController(withIdentifier: identifier as String) as? PopoverViewController else { + fatalError("Why cant i find PopoverViewController? - Check PopoverViewController.storyboard") + } + //let viewcontroller = PopoverViewController() + return viewcontroller + } +} + diff --git a/launchers/macosx/I2PLauncher/userinterface/StatusBarController.swift b/launchers/macosx/I2PLauncher/userinterface/StatusBarController.swift new file mode 100644 index 0000000000..fd28d54f0c --- /dev/null +++ b/launchers/macosx/I2PLauncher/userinterface/StatusBarController.swift @@ -0,0 +1,120 @@ +// +// StatusBarController.swift +// I2PLauncher +// +// Created by Mikal Villa on 13/03/2018. +// Copyright © 2018 I2P. All rights reserved. +// + +import Foundation +import Cocoa + + +@objc class StatusBarController: NSObject, NSMenuDelegate { + + let popover = NSPopover() + let statusItem = NSStatusBar.system().statusItem(withLength: NSVariableStatusItemLength) + //let storyboard = NSStoryboard(name: "Storyboard", bundle: nil) + + @objc func handleOpenConsole(_ sender: Any?) { + SwiftMainDelegate.openLink(url: "http://localhost:7657") + } + + @objc func constructMenu() -> NSMenu { + let menu = NSMenu() + let sb = SwiftMainDelegate.objCBridge + + menu.addItem(NSMenuItem(title: "Open I2P Console", action: #selector(self.handleOpenConsole(_:)), keyEquivalent: "O")) + menu.addItem(NSMenuItem.separator()) + menu.addItem(NSMenuItem(title: "Quit I2P Launcher", action: #selector(SwiftMainDelegate.terminate(_:)), keyEquivalent: "q")) + + return menu + } + + + override init() { + super.init()//(xib: "UserInterface", bundle: nil) + popover.contentViewController = PopoverViewController.freshController() + + if let button = statusItem.button { + button.image = NSImage(named:"StatusBarButtonImage") + //button.title = "I2P" + button.toolTip = "I2P Launch Manager" + //button.isVisible = true + //button.action = #selector(self.statusBarButtonClicked) + //button.sendAction(on: [.leftMouseUp, .rightMouseUp]) + //button.doubleAction = #selector(self.systemBarIconDoubleClick) + button.target = self + button.action = #selector(self.statusBarButtonClicked(sender:)) + button.sendAction(on: [.leftMouseUp, .rightMouseUp]) + } + } + + @IBAction func openConsoleClicked(_ sender: Any) { + NSLog("openConsoleClicked got clicked") + let realSender = sender as! NSMenuItem + NSWorkspace.shared().open(URL(string: "http://127.0.0.1:7657")!) + NSLog("Sender: @%", realSender) + } + + @IBAction func quitClicked(_ sender: NSMenuItem) { + NSApplication.shared().terminate(self) + } + + // Submenu + + @IBAction func startRouterClicked(_ sender: NSMenuItem) { + + } + + @IBAction func restartRouterClicked(_ sender: NSMenuItem) { + + } + + @IBAction func stopRouterClicked(_ sender: NSMenuItem) { + + } + + func statusBarButtonClicked(sender: NSStatusBarButton) { + let event = NSApp.currentEvent! + + if event.type == NSEventType.rightMouseUp { + closePopover(sender: nil) + + let ctxMenu = constructMenu() + + statusItem.menu = ctxMenu + statusItem.popUpMenu(ctxMenu) + + // This is critical, otherwise clicks won't be processed again + statusItem.menu = nil + } else { + togglePopover(sender: nil) + } + } + + func togglePopover(sender: AnyObject?) { + if popover.isShown { + closePopover(sender: sender) + } else { + showPopover(sender: sender) + } + } + + func showPopover(sender: AnyObject?) { + if let button = statusItem.button { + let inst = RouterStatusView.getInstance() + if (inst != nil) { + if (inst != Optional.none) { RouterStatusView.getInstance()?.setRouterStatusLabelText() } + } + popover.show(relativeTo: button.bounds, of: button, preferredEdge: NSRectEdge.minY) + } + } + + func closePopover(sender: AnyObject?) { + popover.performClose(sender) + } + + +} + diff --git a/launchers/macosx/I2PLauncher/ItoopieTransparent.png b/launchers/macosx/ItoopieTransparent.png similarity index 100% rename from launchers/macosx/I2PLauncher/ItoopieTransparent.png rename to launchers/macosx/ItoopieTransparent.png diff --git a/launchers/macosx/I2PLauncher/JavaHelper.h b/launchers/macosx/JavaHelper.h similarity index 100% rename from launchers/macosx/I2PLauncher/JavaHelper.h rename to launchers/macosx/JavaHelper.h diff --git a/launchers/macosx/I2PLauncher/RouterTask.h b/launchers/macosx/RouterTask.h similarity index 100% rename from launchers/macosx/I2PLauncher/RouterTask.h rename to launchers/macosx/RouterTask.h diff --git a/launchers/macosx/I2PLauncher/RouterTask.mm b/launchers/macosx/RouterTask.mm similarity index 100% rename from launchers/macosx/I2PLauncher/RouterTask.mm rename to launchers/macosx/RouterTask.mm diff --git a/launchers/macosx/I2PLauncher/fullBuild.sh b/launchers/macosx/fullBuild.sh similarity index 100% rename from launchers/macosx/I2PLauncher/fullBuild.sh rename to launchers/macosx/fullBuild.sh diff --git a/launchers/macosx/I2PLauncher/images/128x128-Itoopie Transparent.png b/launchers/macosx/images/128x128-Itoopie Transparent.png similarity index 100% rename from launchers/macosx/I2PLauncher/images/128x128-Itoopie Transparent.png rename to launchers/macosx/images/128x128-Itoopie Transparent.png diff --git a/launchers/macosx/I2PLauncher/images/128x128-Itoopie Transparent@2x.png b/launchers/macosx/images/128x128-Itoopie Transparent@2x.png similarity index 100% rename from launchers/macosx/I2PLauncher/images/128x128-Itoopie Transparent@2x.png rename to launchers/macosx/images/128x128-Itoopie Transparent@2x.png diff --git a/launchers/macosx/I2PLauncher/images/16x16-Itoopie Transparent@2x.png b/launchers/macosx/images/16x16-Itoopie Transparent@2x.png similarity index 100% rename from launchers/macosx/I2PLauncher/images/16x16-Itoopie Transparent@2x.png rename to launchers/macosx/images/16x16-Itoopie Transparent@2x.png diff --git a/launchers/macosx/I2PLauncher/images/256x256-Itoopie Transparent.png b/launchers/macosx/images/256x256-Itoopie Transparent.png similarity index 100% rename from launchers/macosx/I2PLauncher/images/256x256-Itoopie Transparent.png rename to launchers/macosx/images/256x256-Itoopie Transparent.png diff --git a/launchers/macosx/I2PLauncher/images/256x256-Itoopie Transparent@2x.png b/launchers/macosx/images/256x256-Itoopie Transparent@2x.png similarity index 100% rename from launchers/macosx/I2PLauncher/images/256x256-Itoopie Transparent@2x.png rename to launchers/macosx/images/256x256-Itoopie Transparent@2x.png diff --git a/launchers/macosx/I2PLauncher/images/32x32-Itoopie Transparent.png b/launchers/macosx/images/32x32-Itoopie Transparent.png similarity index 100% rename from launchers/macosx/I2PLauncher/images/32x32-Itoopie Transparent.png rename to launchers/macosx/images/32x32-Itoopie Transparent.png diff --git a/launchers/macosx/I2PLauncher/images/32x32-Itoopie Transparent@2x.png b/launchers/macosx/images/32x32-Itoopie Transparent@2x.png similarity index 100% rename from launchers/macosx/I2PLauncher/images/32x32-Itoopie Transparent@2x.png rename to launchers/macosx/images/32x32-Itoopie Transparent@2x.png diff --git a/launchers/macosx/I2PLauncher/images/512x512-Itoopie Transparent.png b/launchers/macosx/images/512x512-Itoopie Transparent.png similarity index 100% rename from launchers/macosx/I2PLauncher/images/512x512-Itoopie Transparent.png rename to launchers/macosx/images/512x512-Itoopie Transparent.png diff --git a/launchers/macosx/I2PLauncher/images/512x512-Itoopie Transparent@2x.png b/launchers/macosx/images/512x512-Itoopie Transparent@2x.png similarity index 100% rename from launchers/macosx/I2PLauncher/images/512x512-Itoopie Transparent@2x.png rename to launchers/macosx/images/512x512-Itoopie Transparent@2x.png diff --git a/launchers/macosx/I2PLauncher/images/AppIcon.icns b/launchers/macosx/images/AppIcon.icns similarity index 100% rename from launchers/macosx/I2PLauncher/images/AppIcon.icns rename to launchers/macosx/images/AppIcon.icns diff --git a/launchers/macosx/I2PLauncher/PidWatcher.h b/launchers/macosx/include/PidWatcher.h similarity index 100% rename from launchers/macosx/I2PLauncher/PidWatcher.h rename to launchers/macosx/include/PidWatcher.h diff --git a/launchers/macosx/I2PLauncher/portcheck.h b/launchers/macosx/include/portcheck.h similarity index 100% rename from launchers/macosx/I2PLauncher/portcheck.h rename to launchers/macosx/include/portcheck.h diff --git a/launchers/macosx/I2PLauncher/main.mm b/launchers/macosx/main.mm similarity index 100% rename from launchers/macosx/I2PLauncher/main.mm rename to launchers/macosx/main.mm -- GitLab