first commit
commit
84bcda6f0b
@ -0,0 +1,16 @@
|
|||||||
|
|
||||||
|
|
||||||
|
include $(TOPDIR)/rules.mk
|
||||||
|
|
||||||
|
|
||||||
|
LUCI_TITLE:=LuCI support for seafile
|
||||||
|
LUCI_PKGARCH:=all
|
||||||
|
LUCI_DEPENDS:=+lsblk +docker +luci-lib-taskd
|
||||||
|
|
||||||
|
define Package/luci-app-seafile/conffiles
|
||||||
|
/etc/config/seafile
|
||||||
|
endef
|
||||||
|
|
||||||
|
include $(TOPDIR)/feeds/luci/luci.mk
|
||||||
|
|
||||||
|
# call BuildPackage - OpenWrt buildroot signature
|
@ -0,0 +1,15 @@
|
|||||||
|
|
||||||
|
module("luci.controller.seafile", package.seeall)
|
||||||
|
|
||||||
|
function index()
|
||||||
|
entry({"admin", "apps"}, firstchild(), _("Apps") , 45).dependent = false
|
||||||
|
if not nixio.fs.access("/etc/config/seafile") then
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
|
local page = entry({"admin", "apps", "seafile"}, cbi("seafile"), _("seafile"))
|
||||||
|
page.order = 10
|
||||||
|
page.dependent = true
|
||||||
|
page.acl_depends = { "luci-app-seafile" }
|
||||||
|
entry({"admin","apps","seafile","status"},call("act_status")).leaf=true
|
||||||
|
end
|
@ -0,0 +1,47 @@
|
|||||||
|
--[[
|
||||||
|
LuCI - Lua Configuration Interface
|
||||||
|
]]--
|
||||||
|
|
||||||
|
local taskd = require "luci.model.tasks"
|
||||||
|
local seafile_model = require "luci.model.seafile"
|
||||||
|
local m, s, o
|
||||||
|
|
||||||
|
m = taskd.docker_map("seafile", "seafile", "/usr/libexec/apps/seafile/seafile.sh",
|
||||||
|
translate("seafile"),
|
||||||
|
translate("Seafile is an open-source, cross-platform file-hosting software system. Files are stored on a central server and can be synchronized with personal computers and mobile devices through apps.")
|
||||||
|
.. translate("Official website:") .. ' <a href=\"https://www.seafile.com/en/home/\" target=\"_blank\">https://www.seafile.com/en/home/</a>')
|
||||||
|
|
||||||
|
s = m:section(SimpleSection, translate("Service Status"), translate("seafile status:"))
|
||||||
|
s:append(Template("seafile/status"))
|
||||||
|
|
||||||
|
s = m:section(TypedSection, "seafile", translate("Setup"), translate("Default user: me@example.com Default pass: asecret"))
|
||||||
|
s.addremove=false
|
||||||
|
s.anonymous=true
|
||||||
|
|
||||||
|
o = s:option(Value, "port", translate("Port").."<b>*</b>")
|
||||||
|
o.rmempty = false
|
||||||
|
o.default = "8160"
|
||||||
|
o.datatype = "port"
|
||||||
|
|
||||||
|
o = s:option(Value, "image_name", translate("Image").."<b>*</b>")
|
||||||
|
o.rmempty = false
|
||||||
|
o.datatype = "string"
|
||||||
|
o:value("seafileltd/seafile-mc:latest", "seafileltd/seafile-mc:latest")
|
||||||
|
o.default = "seafileltd/seafile-mc:latest"
|
||||||
|
|
||||||
|
local blocks = seafile_model.blocks()
|
||||||
|
local home = seafile_model.home()
|
||||||
|
|
||||||
|
o = s:option(Value, "config_path", translate("Config path").."<b>*</b>")
|
||||||
|
o.rmempty = false
|
||||||
|
o.datatype = "string"
|
||||||
|
o:value("/opt/docker2/compose/Config/Seafile", "/opt/docker2/compose/Config/Seafile")
|
||||||
|
o.default = "/opt/docker2/compose/Config/Seafile"
|
||||||
|
|
||||||
|
local paths, default_path = seafile_model.find_paths(blocks, home, "Configs")
|
||||||
|
for _, val in pairs(paths) do
|
||||||
|
o:value(val, val)
|
||||||
|
end
|
||||||
|
o.default = default_path
|
||||||
|
|
||||||
|
return m
|
@ -0,0 +1,55 @@
|
|||||||
|
local util = require "luci.util"
|
||||||
|
local jsonc = require "luci.jsonc"
|
||||||
|
|
||||||
|
local seafile = {}
|
||||||
|
|
||||||
|
seafile.blocks = function()
|
||||||
|
local f = io.popen("lsblk -s -f -b -o NAME,FSSIZE,MOUNTPOINT --json", "r")
|
||||||
|
local vals = {}
|
||||||
|
if f then
|
||||||
|
local ret = f:read("*all")
|
||||||
|
f:close()
|
||||||
|
local obj = jsonc.parse(ret)
|
||||||
|
for _, val in pairs(obj["blockdevices"]) do
|
||||||
|
local fsize = val["fssize"]
|
||||||
|
if fsize ~= nil and string.len(fsize) > 10 and val["mountpoint"] then
|
||||||
|
-- fsize > 1G
|
||||||
|
vals[#vals+1] = val["mountpoint"]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return vals
|
||||||
|
end
|
||||||
|
|
||||||
|
seafile.home = function()
|
||||||
|
local uci = require "luci.model.uci".cursor()
|
||||||
|
local home_dirs = {}
|
||||||
|
home_dirs["main_dir"] = uci:get_first("quickstart", "main", "main_dir", "/root")
|
||||||
|
home_dirs["Configs"] = uci:get_first("quickstart", "main", "conf_dir", home_dirs["main_dir"].."/Configs")
|
||||||
|
home_dirs["Public"] = uci:get_first("quickstart", "main", "pub_dir", home_dirs["main_dir"].."/Public")
|
||||||
|
home_dirs["Downloads"] = uci:get_first("quickstart", "main", "dl_dir", home_dirs["Public"].."/Downloads")
|
||||||
|
home_dirs["Caches"] = uci:get_first("quickstart", "main", "tmp_dir", home_dirs["main_dir"].."/Caches")
|
||||||
|
return home_dirs
|
||||||
|
end
|
||||||
|
|
||||||
|
seafile.find_paths = function(blocks, home_dirs, path_name)
|
||||||
|
local default_path = ''
|
||||||
|
local configs = {}
|
||||||
|
|
||||||
|
default_path = home_dirs[path_name] .. "/seafile"
|
||||||
|
if #blocks == 0 then
|
||||||
|
table.insert(configs, default_path)
|
||||||
|
else
|
||||||
|
for _, val in pairs(blocks) do
|
||||||
|
table.insert(configs, val .. "/" .. path_name .. "/seafile")
|
||||||
|
end
|
||||||
|
local without_conf_dir = "/root/" .. path_name .. "/seafile"
|
||||||
|
if default_path == without_conf_dir then
|
||||||
|
default_path = configs[1]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
return configs, default_path
|
||||||
|
end
|
||||||
|
|
||||||
|
return seafile
|
@ -0,0 +1,31 @@
|
|||||||
|
<%
|
||||||
|
local util = require "luci.util"
|
||||||
|
local container_status = util.trim(util.exec("/usr/libexec/apps/seafile/seafile.sh status"))
|
||||||
|
local container_install = (string.len(container_status) > 0)
|
||||||
|
local container_running = container_status == "running"
|
||||||
|
-%>
|
||||||
|
<div class="cbi-value">
|
||||||
|
<label class="cbi-value-title"><%:Status%></label>
|
||||||
|
<div class="cbi-value-field">
|
||||||
|
<% if container_running then %>
|
||||||
|
<button class="cbi-button cbi-button-success" disabled="true"><%:seafile is running%></button>
|
||||||
|
<% else %>
|
||||||
|
<button class="cbi-button cbi-button-negative" disabled="true"><%:seafile is not running%></button>
|
||||||
|
<% end %>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<%
|
||||||
|
if container_running then
|
||||||
|
local port=util.trim(util.exec("/usr/libexec/apps/seafile/seafile.sh port"))
|
||||||
|
if port == "" then
|
||||||
|
port="8160"
|
||||||
|
end
|
||||||
|
-%>
|
||||||
|
<div class="cbi-value cbi-value-last">
|
||||||
|
<label class="cbi-value-title"> </label>
|
||||||
|
<div class="cbi-value-field">
|
||||||
|
|
||||||
|
<input type="button" class="btn cbi-button cbi-button-apply" name="start" value="<%:Open seafile%>" onclick="window.open('http://'+location.hostname+':<%=port%>/', '_blank')">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<% end %>
|
@ -0,0 +1,5 @@
|
|||||||
|
config seafile
|
||||||
|
option 'port' '8160'
|
||||||
|
option 'config_path' '/opt/docker2/Conf/seafile'
|
||||||
|
option 'image_name' 'seafileltd/seafile-mc:latest'
|
||||||
|
|
@ -0,0 +1,14 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
config_dir=`uci -q get seafile.@seafile[0].config_path`
|
||||||
|
|
||||||
|
data_dir=`docker inspect --format '{{.Mounts}}' seafile | grep -Eom1 '[^ ]+/_data /var/www/html local true ' | cut -d' ' -f1`
|
||||||
|
|
||||||
|
if [ -n "$data_dir" -a "$data_dir" != "$config_dir" ]; then
|
||||||
|
uci -q batch <<-EOF >/dev/null
|
||||||
|
set seafile.@seafile[0].config_path="$data_dir"
|
||||||
|
commit seafile
|
||||||
|
EOF
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit 0
|
@ -0,0 +1,138 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
ACTION=${1}
|
||||||
|
shift 1
|
||||||
|
|
||||||
|
get_image() {
|
||||||
|
IMAGE_NAME="seafileltd/seafile-mc:latest"
|
||||||
|
}
|
||||||
|
|
||||||
|
do_install_detail() {
|
||||||
|
local config=`uci get seafile.@seafile[0].config_path 2>/dev/null`
|
||||||
|
local port=`uci get seafile.@seafile[0].port 2>/dev/null`
|
||||||
|
local IMAGE_NAME=`uci get seafile.@seafile[0].image_name 2>/dev/null`
|
||||||
|
local username=`uci get seafile.@seafile[0].username 2>/dev/null`
|
||||||
|
local password=`uci get seafile.@seafile[0].password 2>/dev/null`
|
||||||
|
|
||||||
|
#Generate the generic environment variables for the docker-compose
|
||||||
|
|
||||||
|
GEN_PASS=$(< /dev/urandom tr -dc A-Za-z0-9 2>/dev/null | head -c14; echo)
|
||||||
|
GEN_PASS2=$(< /dev/urandom tr -dc A-Za-z0-9 2>/dev/null | head -c14; echo)
|
||||||
|
|
||||||
|
# Get our local LAN IP Address
|
||||||
|
LAN_IP=$(uci get network.lan.ipaddr)
|
||||||
|
# Strip trailing network mask
|
||||||
|
LAN_IP="${LAN_IP%/*}"
|
||||||
|
|
||||||
|
if [ -z "$config" ]; then
|
||||||
|
echo "config path is empty!"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
[ -z "$port" ] && port=8160
|
||||||
|
[ -z "$IMAGE_NAME" ] && IMAGE_NAME=seafileltd/seafile-mc:latest
|
||||||
|
|
||||||
|
|
||||||
|
# Create Docker Compose file with custom variables
|
||||||
|
|
||||||
|
# Create Docker Compose file with custom variables
|
||||||
|
touch docker-compose.yml
|
||||||
|
cat > docker-compose.yml <<EOF
|
||||||
|
version: '2.0'
|
||||||
|
services:
|
||||||
|
db:
|
||||||
|
image: mariadb:10.5
|
||||||
|
container_name: seafile-mysql
|
||||||
|
environment:
|
||||||
|
- MYSQL_ROOT_PASSWORD=${GEN_PASS} # Requested, set the root's password of MySQL service.
|
||||||
|
- MYSQL_LOG_CONSOLE=true
|
||||||
|
volumes:
|
||||||
|
- ./seafile-mysql/db:/var/lib/mysql # Requested, specifies the path to MySQL data persistent store.
|
||||||
|
networks:
|
||||||
|
- seafile-net
|
||||||
|
|
||||||
|
memcached:
|
||||||
|
image: memcached:1.6
|
||||||
|
container_name: seafile-memcached
|
||||||
|
entrypoint: memcached -m 256
|
||||||
|
networks:
|
||||||
|
- seafile-net
|
||||||
|
|
||||||
|
seafile:
|
||||||
|
image: seafileltd/seafile-mc:latest
|
||||||
|
container_name: seafile
|
||||||
|
ports:
|
||||||
|
- "${port}:80"
|
||||||
|
# - "4643:443" # If https is enabled, cancel the comment.
|
||||||
|
volumes:
|
||||||
|
- ./seafile-data:/shared # Requested, specifies the path to Seafile data persistent store.
|
||||||
|
environment:
|
||||||
|
- DB_HOST=db
|
||||||
|
- DB_ROOT_PASSWD=${GEN_PASS} # Requested, the value shuold be root's password of MySQL service.
|
||||||
|
- TIME_ZONE=Etc/UTC # Optional, default is UTC. Should be uncomment and set to your local time zone.
|
||||||
|
- SEAFILE_ADMIN_EMAIL=me@example.com # Specifies Seafile admin user, default is 'me@example.com'.
|
||||||
|
- SEAFILE_ADMIN_PASSWORD=asecret # Specifies Seafile admin password, default is 'asecret'.
|
||||||
|
- SEAFILE_SERVER_LETSENCRYPT=false # Whether to use https or not.
|
||||||
|
# - SEAFILE_SERVER_HOSTNAME=docs.seafile.com # Specifies your host name if https is enabled.
|
||||||
|
depends_on:
|
||||||
|
- db
|
||||||
|
- memcached
|
||||||
|
networks:
|
||||||
|
- seafile-net
|
||||||
|
|
||||||
|
networks:
|
||||||
|
seafile-net:
|
||||||
|
labels:
|
||||||
|
plugsy.name: "Seafile"
|
||||||
|
plugsy.category: "Home"
|
||||||
|
plugsy.icon: "@styled-icons/icomoon/Seafile"
|
||||||
|
plugsy.link: "http://${LAN_IP}:${port}"
|
||||||
|
EOF
|
||||||
|
|
||||||
|
docker-compose up -d
|
||||||
|
|
||||||
|
# Add a new list option to the "shortcutmenu" configuration file
|
||||||
|
uci add shortcutmenu lists
|
||||||
|
uci set shortcutmenu.@lists[-1].webname="$IMAGE_NAME"
|
||||||
|
uci set shortcutmenu.@lists[-1].weburl="$LAN_IP:$port"
|
||||||
|
uci set shortcutmenu.@lists[-1].webpath="/"
|
||||||
|
uci commit shortcutmenu
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
usage() {
|
||||||
|
echo "usage: $0 sub-command"
|
||||||
|
echo "where sub-command is one of:"
|
||||||
|
echo " install Install seafile"
|
||||||
|
echo " upgrade Upgrade seafile"
|
||||||
|
echo " rm/start/stop/restart Remove/Start/Stop/Restart seafile"
|
||||||
|
echo " status seafile status"
|
||||||
|
echo " port seafile port"
|
||||||
|
}
|
||||||
|
|
||||||
|
case ${ACTION} in
|
||||||
|
"install")
|
||||||
|
get_image
|
||||||
|
do_install_detail
|
||||||
|
;;
|
||||||
|
"upgrade")
|
||||||
|
get_image
|
||||||
|
do_install_detail
|
||||||
|
;;
|
||||||
|
"rm")
|
||||||
|
docker rm -f seafile
|
||||||
|
;;
|
||||||
|
"start" | "stop" | "restart")
|
||||||
|
docker ${ACTION} seafile
|
||||||
|
;;
|
||||||
|
"status")
|
||||||
|
docker ps --all -f 'name=seafile' --format '{{.State}}'
|
||||||
|
;;
|
||||||
|
"port")
|
||||||
|
docker ps --all -f 'name=seafile' --format '{{.Ports}}' | grep -om1 '0.0.0.0:[0-9]*' | sed 's/0.0.0.0://'
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
usage
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
esac
|
@ -0,0 +1,11 @@
|
|||||||
|
{
|
||||||
|
"luci-app-seafile": {
|
||||||
|
"description": "Grant UCI access for luci-app-seafile",
|
||||||
|
"read": {
|
||||||
|
"uci": [ "seafile" ]
|
||||||
|
},
|
||||||
|
"write": {
|
||||||
|
"uci": [ "seafile" ]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue