Page Menu
Home
Phorge
Search
Configure Global Search
Log In
Files
F833825
mod.rs
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Flag For Later
Award Token
Size
9 KB
Referenced Files
None
Subscribers
None
mod.rs
View Options
#[macro_use]
mod
log
;
mod
api
;
mod
fork
;
use
api
::{
DAEMON_CPU_PERCENTAGE
,
DAEMON_MEM_USAGE
,
DAEMON_START_TIME
};
use
chrono
::{
DateTime
,
Utc
};
use
colored
::
Colorize
;
use
fork
::{
daemon
,
Fork
};
use
global_placeholders
::
global
;
use
macros_rs
::{
crashln
,
str
,
string
,
ternary
,
then
};
use
psutil
::
process
::{
MemoryInfo
,
Process
};
use
serde
::
Serialize
;
use
serde_json
::
json
;
use
std
::
sync
::
atomic
::{
AtomicBool
,
Ordering
};
use
std
::{
process
,
thread
::
sleep
,
time
::
Duration
};
use
pmc
::{
config
,
file
,
helpers
::{
self
,
ColoredString
},
process
::{
hash
,
id
::
Id
,
Runner
,
Status
},
};
use
tabled
::{
settings
::{
object
::
Columns
,
style
::{
BorderColor
,
Style
},
themes
::
Colorization
,
Color
,
Rotate
,
},
Table
,
Tabled
,
};
static
ENABLE_API
:
AtomicBool
=
AtomicBool
::
new
(
false
);
static
ENABLE_WEBUI
:
AtomicBool
=
AtomicBool
::
new
(
false
);
extern
"C"
fn
handle_termination_signal
(
_
:
libc
::
c_int
)
{
pid
::
remove
();
log
!
(
"[daemon] killed"
,
"pid"
=>
process
::
id
());
unsafe
{
libc
::
_exit
(
0
)
}
}
fn
restart_process
()
{
for
(
id
,
item
)
in
Runner
::
new
().
items_mut
()
{
let
mut
runner
=
Runner
::
new
();
let
children
=
pmc
::
service
::
find_chidren
(
item
.
pid
);
if
!
children
.
is_empty
()
&&
children
!=
item
.
children
{
log
!
(
"[daemon] added"
,
"children"
=>
format!
(
"{children:?}"
));
runner
.
set_children
(
*
id
,
children
).
save
();
}
if
item
.
running
&&
item
.
watch
.
enabled
{
let
path
=
item
.
path
.
join
(
item
.
watch
.
path
.
clone
());
let
hash
=
hash
::
create
(
path
);
if
hash
!=
item
.
watch
.
hash
{
runner
.
restart
(
item
.
id
,
false
);
log
!
(
"[daemon] watch reload"
,
"name"
=>
item
.
name
,
"hash"
=>
"hash"
);
continue
;
}
}
if
!
item
.
running
&&
pid
::
running
(
item
.
pid
as
i32
)
{
Runner
::
new
().
set_status
(
*
id
,
Status
::
Running
);
log
!
(
"[daemon] process fix status"
,
"name"
=>
item
.
name
,
"id"
=>
id
);
continue
;
}
then
!
(
!
item
.
running
||
pid
::
running
(
item
.
pid
as
i32
),
continue
);
if
item
.
running
&&
item
.
crash
.
value
==
config
::
read
().
daemon
.
restarts
{
log
!
(
"[daemon] process has crashed"
,
"name"
=>
item
.
name
,
"id"
=>
id
);
runner
.
stop
(
item
.
id
);
runner
.
set_crashed
(
*
id
).
save
();
continue
;
}
else
{
runner
.
get
(
item
.
id
).
crashed
();
log
!
(
"[daemon] restarted"
,
"name"
=>
item
.
name
,
"id"
=>
id
,
"crashes"
=>
item
.
crash
.
value
);
}
}
}
pub
fn
health
(
format
:
&
String
)
{
let
mut
pid
:
Option
<
i32
>
=
None
;
let
mut
cpu_percent
:
Option
<
f64
>
=
None
;
let
mut
uptime
:
Option
<
DateTime
<
Utc
>>
=
None
;
let
mut
memory_usage
:
Option
<
MemoryInfo
>
=
None
;
let
mut
runner
:
Runner
=
file
::
read_object
(
global
!
(
"pmc.dump"
));
#[derive(Clone, Debug, Tabled)]
struct
Info
{
#[tabled(rename =
"pid file"
)]
pid_file
:
String
,
#[tabled(rename =
"fork path"
)]
path
:
String
,
#[tabled(rename =
"cpu percent"
)]
cpu_percent
:
String
,
#[tabled(rename =
"memory usage"
)]
memory_usage
:
String
,
#[tabled(rename =
"daemon type"
)]
external
:
String
,
#[tabled(rename =
"process count"
)]
process_count
:
usize
,
uptime
:
String
,
pid
:
String
,
status
:
ColoredString
,
}
impl
Serialize
for
Info
{
fn
serialize
<
S
:
serde
::
Serializer
>
(
&
self
,
serializer
:
S
)
->
Result
<
S
::
Ok
,
S
::
Error
>
{
let
trimmed_json
=
json
!
({
"pid_file"
:
&
self
.
pid_file
.
trim
(),
"path"
:
&
self
.
path
.
trim
(),
"cpu"
:
&
self
.
cpu_percent
.
trim
(),
"mem"
:
&
self
.
memory_usage
.
trim
(),
"process_count"
:
&
self
.
process_count
.
to_string
(),
"uptime"
:
&
self
.
uptime
.
trim
(),
"pid"
:
&
self
.
pid
.
trim
(),
"status"
:
&
self
.
status
.
0.
trim
(),
});
trimmed_json
.
serialize
(
serializer
)
}
}
if
pid
::
exists
()
{
if
let
Ok
(
process_id
)
=
pid
::
read
()
{
if
let
Ok
(
process
)
=
Process
::
new
(
process_id
.
get
::
<
u32
>
())
{
pid
=
Some
(
process
.
pid
()
as
i32
);
uptime
=
Some
(
pid
::
uptime
().
unwrap
());
memory_usage
=
process
.
memory_info
().
ok
();
cpu_percent
=
Some
(
pmc
::
service
::
get_process_cpu_usage_percentage
(
process_id
.
get
::
<
i64
>
()));
}
}
}
let
cpu_percent
=
match
cpu_percent
{
Some
(
percent
)
=>
format!
(
"{:.2}%"
,
percent
),
None
=>
string
!
(
"0.00%"
),
};
let
memory_usage
=
match
memory_usage
{
Some
(
usage
)
=>
helpers
::
format_memory
(
usage
.
rss
()),
None
=>
string
!
(
"0b"
),
};
let
uptime
=
match
uptime
{
Some
(
uptime
)
=>
helpers
::
format_duration
(
uptime
),
None
=>
string
!
(
"none"
),
};
let
pid
=
match
pid
{
Some
(
pid
)
=>
string
!
(
pid
),
None
=>
string
!
(
"n/a"
),
};
let
data
=
vec!
[
Info
{
pid
:
pid
,
cpu_percent
,
memory_usage
,
uptime
:
uptime
,
path
:
global
!
(
"pmc.base"
),
external
:
global
!
(
"pmc.daemon.kind"
),
process_count
:
runner
.
count
(),
pid_file
:
format
!
(
"{} "
,
global
!
(
"pmc.pid"
)),
status
:
ColoredString
(
ternary
!
(
pid
::
exists
(),
"online"
.
green
().
bold
(),
"stopped"
.
red
().
bold
())),
}];
let
table
=
Table
::
new
(
data
.
clone
())
.
with
(
Rotate
::
Left
)
.
with
(
Style
::
rounded
().
remove_horizontals
())
.
with
(
Colorization
::
exact
([
Color
::
FG_CYAN
],
Columns
::
first
()))
.
with
(
BorderColor
::
filled
(
Color
::
FG_BRIGHT_BLACK
))
.
to_string
();
if
let
Ok
(
json
)
=
serde_json
::
to_string
(
&
data
[
0
])
{
match
format
.
as_str
()
{
"raw"
=>
println!
(
"{:?}"
,
data
[
0
]),
"json"
=>
println!
(
"{json}"
),
"default"
=>
{
println!
(
"{}
\n
{table}
\n
"
,
format!
(
"PMC daemon information"
).
on_bright_white
().
black
());
println!
(
" {}"
,
format!
(
"Use `pmc daemon restart` to restart the daemon"
).
white
());
println!
(
" {}"
,
format!
(
"Use `pmc daemon reset` to clean process id values"
).
white
());
}
_
=>
{}
};
};
}
pub
fn
stop
()
{
if
pid
::
exists
()
{
println!
(
"{} Stopping PMC daemon"
,
*
helpers
::
SUCCESS
);
match
pid
::
read
()
{
Ok
(
pid
)
=>
{
pmc
::
service
::
stop
(
pid
.
get
());
pid
::
remove
();
log
!
(
"[daemon] stopped"
,
"pid"
=>
pid
);
println!
(
"{} PMC daemon stopped"
,
*
helpers
::
SUCCESS
);
}
Err
(
err
)
=>
crashln
!
(
"{} Failed to read PID file: {}"
,
*
helpers
::
FAIL
,
err
),
}
}
else
{
crashln
!
(
"{} The daemon is not running"
,
*
helpers
::
FAIL
)
}
}
pub
fn
start
(
verbose
:
bool
)
{
let
external
=
match
global
!
(
"pmc.daemon.kind"
).
as_str
()
{
"external"
=>
true
,
"default"
=>
false
,
"rust"
=>
false
,
"cc"
=>
true
,
_
=>
false
,
};
println!
(
"{} Spawning PMC daemon (pmc_base={})"
,
*
helpers
::
SUCCESS
,
global
!
(
"pmc.base"
));
if
ENABLE_API
.
load
(
Ordering
::
Acquire
)
{
println!
(
"{} API server started (address={}, webui={})"
,
*
helpers
::
SUCCESS
,
config
::
read
().
fmt_address
(),
ENABLE_WEBUI
.
load
(
Ordering
::
Acquire
)
);
}
if
pid
::
exists
()
{
match
pid
::
read
()
{
Ok
(
pid
)
=>
then
!
(
!
pid
::
running
(
pid
.
get
()),
pid
::
remove
()),
Err
(
_
)
=>
crashln
!
(
"{} The daemon is already running"
,
*
helpers
::
FAIL
),
}
}
#[inline]
#[tokio::main]
async
extern
"C"
fn
init
()
{
pid
::
name
(
"PMC Restart Handler Daemon"
);
let
config
=
config
::
read
().
daemon
;
let
api_enabled
=
ENABLE_API
.
load
(
Ordering
::
Acquire
);
let
ui_enabled
=
ENABLE_WEBUI
.
load
(
Ordering
::
Acquire
);
unsafe
{
libc
::
signal
(
libc
::
SIGTERM
,
handle_termination_signal
as
usize
)
};
DAEMON_START_TIME
.
set
(
Utc
::
now
().
timestamp_millis
()
as
f64
);
pid
::
write
(
process
::
id
());
log
!
(
"[daemon] new fork"
,
"pid"
=>
process
::
id
());
if
api_enabled
{
log
!
(
"[api] server queued"
,
"address"
=>
config
::
read
().
fmt_address
());
tokio
::
spawn
(
async
move
{
api
::
start
(
ui_enabled
).
await
});
}
loop
{
if
api_enabled
{
if
let
Ok
(
process
)
=
Process
::
new
(
process
::
id
())
{
DAEMON_CPU_PERCENTAGE
.
observe
(
pmc
::
service
::
get_process_cpu_usage_percentage
(
process
.
pid
()
as
i64
));
DAEMON_MEM_USAGE
.
observe
(
process
.
memory_info
().
ok
().
unwrap
().
rss
()
as
f64
);
}
}
then
!
(
!
Runner
::
new
().
is_empty
(),
restart_process
());
sleep
(
Duration
::
from_millis
(
config
.
interval
));
}
}
println!
(
"{} PMC Successfully daemonized (type={})"
,
*
helpers
::
SUCCESS
,
global
!
(
"pmc.daemon.kind"
));
if
external
{
let
callback
=
pmc
::
Callback
(
init
);
pmc
::
service
::
try_fork
(
false
,
verbose
,
callback
);
}
else
{
match
daemon
(
false
,
verbose
)
{
Ok
(
Fork
::
Parent
(
_
))
=>
{}
Ok
(
Fork
::
Child
)
=>
init
(),
Err
(
err
)
=>
crashln
!
(
"{} Daemon creation failed with code {err}"
,
*
helpers
::
FAIL
),
}
}
}
pub
fn
restart
(
api
:
&
bool
,
webui
:
&
bool
,
verbose
:
bool
)
{
if
pid
::
exists
()
{
stop
();
}
let
config
=
config
::
read
().
daemon
;
if
config
.
web
.
ui
||
*
webui
{
ENABLE_API
.
store
(
true
,
Ordering
::
Release
);
ENABLE_WEBUI
.
store
(
true
,
Ordering
::
Release
);
}
else
if
config
.
web
.
api
{
ENABLE_API
.
store
(
true
,
Ordering
::
Release
);
}
else
{
ENABLE_API
.
store
(
*
api
,
Ordering
::
Release
);
}
start
(
verbose
);
}
pub
fn
reset
()
{
let
mut
runner
=
Runner
::
new
();
let
largest
=
runner
.
size
();
match
largest
{
Some
(
id
)
=>
runner
.
set_id
(
Id
::
from
(
str
!
(
id
.
to_string
()))),
None
=>
runner
.
set_id
(
Id
::
new
(
0
)),
}
println!
(
"{} Successfully reset (index={})"
,
*
helpers
::
SUCCESS
,
runner
.
id
);
}
pub
mod
pid
;
File Metadata
Details
Attached
Mime Type
text/x-c
Expires
Fri, May 30, 2:15 AM (2 d)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
201475
Default Alt Text
mod.rs (9 KB)
Attached To
Mode
rPMC Process Management Controller
Attached
Detach File
Event Timeline
Log In to Comment