mirror of
https://github.com/morgan9e/macos-stats
synced 2026-04-14 00:04:15 +09:00
feat: removed WiFi network name from the network popup (starting from macOS Sonoma it's not available without CoreLocation access)
300 lines
9.3 KiB
Swift
300 lines
9.3 KiB
Swift
//
|
|
// main.swift
|
|
// Net
|
|
//
|
|
// Created by Serhiy Mytrovtsiy on 24/05/2020.
|
|
// Using Swift 5.0.
|
|
// Running on macOS 10.15.
|
|
//
|
|
// Copyright © 2020 Serhiy Mytrovtsiy. All rights reserved.
|
|
//
|
|
|
|
import Cocoa
|
|
import Kit
|
|
import SystemConfiguration
|
|
|
|
public enum Network_t: String, Codable {
|
|
case wifi
|
|
case ethernet
|
|
case bluetooth
|
|
case other
|
|
}
|
|
|
|
public struct Network_interface: Codable {
|
|
var displayName: String = ""
|
|
var BSDName: String = ""
|
|
var address: String = ""
|
|
}
|
|
|
|
public struct Network_addr: Codable {
|
|
var v4: String? = nil
|
|
var v6: String? = nil
|
|
}
|
|
|
|
public struct Network_wifi: Codable {
|
|
var countryCode: String? = nil
|
|
var ssid: String? = nil
|
|
var bssid: String? = nil
|
|
var RSSI: Int? = nil
|
|
var noise: Int? = nil
|
|
var transmitRate: Double? = nil
|
|
|
|
var standard: String? = nil
|
|
var mode: String? = nil
|
|
var security: String? = nil
|
|
var channel: String? = nil
|
|
|
|
var channelBand: String? = nil
|
|
var channelWidth: String? = nil
|
|
var channelNumber: String? = nil
|
|
|
|
mutating func reset() {
|
|
self.countryCode = nil
|
|
self.ssid = nil
|
|
self.RSSI = nil
|
|
self.noise = nil
|
|
self.transmitRate = nil
|
|
self.standard = nil
|
|
self.mode = nil
|
|
self.security = nil
|
|
self.channel = nil
|
|
}
|
|
}
|
|
|
|
public struct Bandwidth: Codable {
|
|
var upload: Int64 = 0
|
|
var download: Int64 = 0
|
|
}
|
|
|
|
public struct Network_Usage: value_t, Codable {
|
|
var bandwidth: Bandwidth = Bandwidth()
|
|
var total: Bandwidth = Bandwidth()
|
|
|
|
var laddr: String? = nil // local ip
|
|
var raddr: Network_addr = Network_addr() // remote ip
|
|
|
|
var interface: Network_interface? = nil
|
|
var connectionType: Network_t? = nil
|
|
var status: Bool = false
|
|
|
|
var wifiDetails: Network_wifi = Network_wifi()
|
|
|
|
mutating func reset() {
|
|
self.bandwidth = Bandwidth()
|
|
|
|
self.laddr = nil
|
|
self.raddr = Network_addr()
|
|
|
|
self.interface = nil
|
|
self.connectionType = nil
|
|
|
|
self.wifiDetails.reset()
|
|
}
|
|
|
|
public var widgetValue: Double = 0
|
|
}
|
|
|
|
public struct Network_Connectivity: Codable {
|
|
var status: Bool = false
|
|
var latency: Double = 0
|
|
}
|
|
|
|
public struct Network_Process: Codable {
|
|
var time: Date = Date()
|
|
var name: String = ""
|
|
var pid: String = ""
|
|
var download: Int = 0
|
|
var upload: Int = 0
|
|
var icon: NSImage {
|
|
get {
|
|
if let pid = pid_t(self.pid), let app = NSRunningApplication(processIdentifier: pid) {
|
|
return app.icon ?? Constants.defaultProcessIcon
|
|
}
|
|
return Constants.defaultProcessIcon
|
|
}
|
|
}
|
|
}
|
|
|
|
public class Network: Module {
|
|
private let popupView: Popup
|
|
private let settingsView: Settings
|
|
private let portalView: Portal
|
|
|
|
private var usageReader: UsageReader? = nil
|
|
private var processReader: ProcessReader? = nil
|
|
private var connectivityReader: ConnectivityReader? = nil
|
|
|
|
private let ipUpdater = NSBackgroundActivityScheduler(identifier: "eu.exelban.Stats.Network.IP")
|
|
private let usageReseter = NSBackgroundActivityScheduler(identifier: "eu.exelban.Stats.Network.Usage")
|
|
|
|
private var widgetActivationThreshold: Int {
|
|
Store.shared.int(key: "\(self.config.name)_widgetActivationThreshold", defaultValue: 0) * 1_024
|
|
}
|
|
private var publicIPRefreshInterval: String {
|
|
Store.shared.string(key: "\(self.name)_publicIPRefreshInterval", defaultValue: "never")
|
|
}
|
|
|
|
public init() {
|
|
self.settingsView = Settings("Network")
|
|
self.popupView = Popup("Network")
|
|
self.portalView = Portal("Network")
|
|
|
|
super.init(
|
|
popup: self.popupView,
|
|
settings: self.settingsView,
|
|
portal: self.portalView
|
|
)
|
|
guard self.available else { return }
|
|
|
|
self.usageReader = UsageReader(.network)
|
|
self.processReader = ProcessReader(.network)
|
|
self.connectivityReader = ConnectivityReader(.network)
|
|
|
|
self.settingsView.callbackWhenUpdateNumberOfProcesses = {
|
|
self.popupView.numberOfProcessesUpdated()
|
|
DispatchQueue.global(qos: .background).async {
|
|
self.processReader?.read()
|
|
}
|
|
}
|
|
|
|
self.usageReader?.callbackHandler = { [weak self] value in
|
|
self?.usageCallback(value)
|
|
}
|
|
self.usageReader?.readyCallback = { [weak self] in
|
|
self?.readyHandler()
|
|
}
|
|
|
|
self.processReader?.callbackHandler = { [weak self] value in
|
|
if let list = value {
|
|
self?.popupView.processCallback(list)
|
|
}
|
|
}
|
|
|
|
self.connectivityReader?.callbackHandler = { [weak self] value in
|
|
self?.connectivityCallback(value)
|
|
}
|
|
|
|
self.settingsView.callback = { [weak self] in
|
|
self?.usageReader?.getDetails()
|
|
self?.usageReader?.read()
|
|
}
|
|
self.settingsView.usageResetCallback = { [weak self] in
|
|
self?.setUsageReset()
|
|
}
|
|
self.settingsView.ICMPHostCallback = { [weak self] isDisabled in
|
|
if isDisabled {
|
|
self?.popupView.resetConnectivityView()
|
|
self?.connectivityCallback(Network_Connectivity(status: false))
|
|
}
|
|
}
|
|
self.settingsView.publicIPRefreshIntervalCallback = { [weak self] in
|
|
self?.setIPUpdater()
|
|
}
|
|
|
|
if let reader = self.usageReader {
|
|
self.addReader(reader)
|
|
}
|
|
if let reader = self.processReader {
|
|
self.addReader(reader)
|
|
}
|
|
if let reader = self.connectivityReader {
|
|
self.addReader(reader)
|
|
}
|
|
|
|
self.setIPUpdater()
|
|
self.setUsageReset()
|
|
}
|
|
|
|
public override func isAvailable() -> Bool {
|
|
var list: [String] = []
|
|
for interface in SCNetworkInterfaceCopyAll() as NSArray {
|
|
if let displayName = SCNetworkInterfaceGetLocalizedDisplayName(interface as! SCNetworkInterface) {
|
|
list.append(displayName as String)
|
|
}
|
|
}
|
|
return !list.isEmpty
|
|
}
|
|
|
|
private func usageCallback(_ raw: Network_Usage?) {
|
|
guard let value = raw, self.enabled else { return }
|
|
|
|
self.popupView.usageCallback(value)
|
|
self.portalView.usageCallback(value)
|
|
|
|
var upload: Int64 = 0
|
|
var download: Int64 = 0
|
|
if value.bandwidth.upload >= self.widgetActivationThreshold || value.bandwidth.download >= self.widgetActivationThreshold {
|
|
upload = value.bandwidth.upload
|
|
download = value.bandwidth.download
|
|
}
|
|
|
|
self.menuBar.widgets.filter{ $0.isActive }.forEach { (w: Widget) in
|
|
switch w.item {
|
|
case let widget as SpeedWidget: widget.setValue(upload: upload, download: download)
|
|
case let widget as NetworkChart: widget.setValue(upload: Double(upload), download: Double(download))
|
|
default: break
|
|
}
|
|
}
|
|
}
|
|
|
|
private func connectivityCallback(_ raw: Network_Connectivity?) {
|
|
guard let value = raw, self.enabled else { return }
|
|
|
|
self.popupView.connectivityCallback(value)
|
|
|
|
self.menuBar.widgets.filter{ $0.isActive }.forEach { (w: Widget) in
|
|
switch w.item {
|
|
case let widget as StateWidget: widget.setValue(value.status)
|
|
default: break
|
|
}
|
|
}
|
|
}
|
|
|
|
private func setIPUpdater() {
|
|
self.ipUpdater.invalidate()
|
|
|
|
switch self.publicIPRefreshInterval {
|
|
case "hour":
|
|
self.ipUpdater.interval = 60 * 60
|
|
case "12":
|
|
self.ipUpdater.interval = 60 * 60 * 12
|
|
case "24":
|
|
self.ipUpdater.interval = 60 * 60 * 24
|
|
default: return
|
|
}
|
|
|
|
self.ipUpdater.repeats = true
|
|
self.ipUpdater.schedule { (completion: @escaping NSBackgroundActivityScheduler.CompletionHandler) in
|
|
guard self.enabled && self.isAvailable() else {
|
|
return
|
|
}
|
|
debug("going to automatically refresh IP address...")
|
|
NotificationCenter.default.post(name: .refreshPublicIP, object: nil, userInfo: nil)
|
|
completion(NSBackgroundActivityScheduler.Result.finished)
|
|
}
|
|
}
|
|
|
|
private func setUsageReset() {
|
|
self.usageReseter.invalidate()
|
|
|
|
switch AppUpdateInterval(rawValue: Store.shared.string(key: "\(self.config.name)_usageReset", defaultValue: AppUpdateInterval.atStart.rawValue)) {
|
|
case .oncePerDay: self.usageReseter.interval = 60 * 60 * 24
|
|
case .oncePerWeek: self.usageReseter.interval = 60 * 60 * 24 * 7
|
|
case .oncePerMonth: self.usageReseter.interval = 60 * 60 * 24 * 30
|
|
case .never, .atStart: return
|
|
default: return
|
|
}
|
|
|
|
self.usageReseter.repeats = true
|
|
self.usageReseter.schedule { (completion: @escaping NSBackgroundActivityScheduler.CompletionHandler) in
|
|
guard self.enabled && self.isAvailable() else {
|
|
return
|
|
}
|
|
|
|
debug("going to reset the usage...")
|
|
NotificationCenter.default.post(name: .resetTotalNetworkUsage, object: nil, userInfo: nil)
|
|
completion(NSBackgroundActivityScheduler.Result.finished)
|
|
}
|
|
}
|
|
}
|