X7ROOT File Manager
Current Path:
/usr/local/cpanel/scripts
usr
/
local
/
cpanel
/
scripts
/
📁
..
📄
MirrorSearch_pingtest
(2.38 KB)
📄
activesync-invite-reply
(1.69 KB)
📄
add_dns
(2.36 KB)
📄
adddns
(2.36 KB)
📄
addpop
(6.08 KB)
📄
addsystemuser
(3.27 KB)
📄
adduser
(92 B)
📄
apachelimits
(4.31 KB)
📄
archive_sync_zones
(3.02 KB)
📄
auto-adjust-mysql-limits
(1.81 KB)
📄
autorepair
(1.24 KB)
📄
backups_clean_metadata_for_missing_backups
(1.57 KB)
📄
backups_create_metadata
(15.75 KB)
📄
backups_list_user_files
(4.56 KB)
📄
balance_linked_node_quotas
(2.58 KB)
📄
biglogcheck
(1.69 KB)
📄
build_bandwidthdb_root_cache_in_background
(1.52 KB)
📄
build_cpnat
(3.41 KB)
📄
build_mail_sni
(3.87 KB)
📄
build_maxemails_config
(1.14 KB)
📄
builddovecotconf
(6.76 KB)
📄
buildeximconf
(7 KB)
📄
buildhttpdconf
(2.6 KB)
📄
buildnsdconf
(1.01 KB)
📄
buildpureftproot
(539 B)
📄
ccs-check
(4.91 KB)
📄
check_cpanel_pkgs
(10.75 KB)
📄
check_cpanel_rpms
(218 B)
📄
check_domain_tls_service_domains.pl
(6.68 KB)
📄
check_immutable_files
(5.49 KB)
📄
check_mail_spamassassin_compiledregexps_body_0
(187 B)
📄
check_maxmem_against_domains_count
(3.57 KB)
📄
check_mount_procfs
(2.02 KB)
📄
check_mysql
(5.55 KB)
📄
check_security_advice_changes
(8.28 KB)
📄
check_unmonitored_enabled_services
(4.56 KB)
📄
check_unreliable_resolvers
(3.59 KB)
📄
check_users_my_cnf
(6.05 KB)
📄
check_valid_server_hostname
(7.66 KB)
📄
checkalldomainsmxs
(2.4 KB)
📄
checkbashshell
(1.18 KB)
📄
checkccompiler
(1.22 KB)
📄
checkexim.pl
(3.1 KB)
📄
checklink
(1.29 KB)
📄
checknsddirs
(1014 B)
📄
checkusers
(856 B)
📄
chkmydns
(561 B)
📄
chkpaths
(141 B)
📄
chpass
(416 B)
📄
ckillall
(1.11 KB)
📄
clean_dead_mailman_locks
(2.09 KB)
📄
clean_up_temp_wheel_users
(2.44 KB)
📄
clean_user_php_sessions
(4.76 KB)
📄
cleandns
(13.09 KB)
📄
cleandns8
(417 B)
📄
cleanmsglog
(735 B)
📄
cleanphpsessions
(932 B)
📄
cleanphpsessions.php
(658 B)
📄
cleanquotas
(1.61 KB)
📄
cleansessions
(5.91 KB)
📄
cleanupinterchange
(2.64 KB)
📄
cleanupmysqlprivs
(533 B)
📄
clear_cpaddon_ui_caches
(1.27 KB)
📄
clear_orphaned_virtfs_mounts
(3.56 KB)
📄
comparecdb
(1.52 KB)
📄
compilers
(2.86 KB)
📄
compilerscheck
(999 B)
📄
configure_firewall_for_cpanel
(520 B)
📄
configure_rh_firewall_for_cpanel
(520 B)
📄
configure_rh_ipv6_firewall_for_cpanel
(520 B)
📄
convert2dovecot
(682 B)
📄
convert_accesshash_to_token
(4.07 KB)
📄
convert_and_migrate_from_legacy_backup
(1.97 KB)
📄
convert_maildir_to_mdbox
(1.66 KB)
📄
convert_mdbox_to_maildir
(1.66 KB)
📄
convert_roundcube_mysql2sqlite
(25.28 KB)
📄
convert_to_dovecot_delivery
(4.33 KB)
📄
convert_whmxfer_to_sqlite
(1.46 KB)
📄
copy_user_mail_as_root
(1.25 KB)
📄
copy_user_mail_as_user
(1.34 KB)
📄
cpaddonsup
(3.25 KB)
📄
cpan_config
(2.8 KB)
📁
cpan_sandbox
📄
cpanel_initial_install
(67.36 KB)
📄
cpanelsync
(28.31 KB)
📄
cpanelsync_postprocessor
(1.62 KB)
📄
cpanpingtest
(965 B)
📄
cpbackup
(44.77 KB)
📄
cpbackup_transport_file
(5.65 KB)
📄
cpdig
(1.81 KB)
📄
cpfetch
(1.23 KB)
📄
cphulkdblacklist
(433 B)
📄
cphulkdwhitelist
(1.3 KB)
📄
cpservice
(2.87 KB)
📄
cpuser_port_authority
(19.29 KB)
📄
cpuser_service_manager
(10.85 KB)
📄
createacct
(24.57 MB)
📄
custom_backup_destination.pl.sample
(5.06 KB)
📄
custom_backup_destination.pl.skeleton
(2.84 KB)
📄
dav_change_hostname
(3.57 KB)
📄
dcpumon-wrapper
(850 B)
📄
delpop
(6.2 KB)
📄
detect_env_capabilities
(508 B)
📄
disable_prelink
(2.77 KB)
📄
disable_sqloptimizer
(1.49 KB)
📄
disablefileprotect
(2.09 KB)
📄
distro_changed_hook
(1.16 KB)
📄
dnscluster
(4.44 KB)
📄
dnsqueuecron
(1.29 KB)
📄
dnssec-cluster-keys
(3.75 KB)
📄
dovecot_maintenance
(7.93 KB)
📄
dovecot_set_defaults.pl
(984 B)
📄
dumpcdb
(866 B)
📄
dumpinodes
(687 B)
📄
dumpquotas
(616 B)
📄
dumpstor
(913 B)
📄
ea4_fresh_install
(2.64 KB)
📄
edit_cpanelsync_exclude_list
(2.58 KB)
📄
editquota
(3.44 KB)
📄
elevate-cpanel
(377.87 KB)
📄
email_archive_maintenance
(6.15 KB)
📄
email_hold_maintenance
(1.46 KB)
📄
enable_spf_dkim_globally
(8.83 KB)
📄
enable_sqloptimizer
(1.57 KB)
📄
enablefileprotect
(2.1 KB)
📄
ensure_autoenabled_features
(2.5 MB)
📄
ensure_conf_dir_crt_key
(4.82 KB)
📄
ensure_cpuser_file_ip
(2.55 KB)
📄
ensure_crontab_permissions
(1.08 KB)
📄
ensure_dovecot_memory_limits_meet_minimum
(3.13 KB)
📄
ensure_hostname_resolves
(2.57 KB)
📄
ensure_includes
(601 B)
📄
ensure_vhost_includes
(13.53 KB)
📄
exim_tidydb
(2.96 KB)
📄
eximconfgen
(1.32 KB)
📄
eximstats_spam_check
(867 B)
📄
export_horde_calendars_to_ics
(15.07 KB)
📄
export_horde_contacts_to_vcf
(13.94 KB)
📄
exportmydnsdb
(3.47 KB)
📄
expunge_expired_certificates_from_sslstorage
(3.56 KB)
📄
expunge_expired_pkgacct_sessions
(852 B)
📄
expunge_expired_transfer_sessions
(1.06 KB)
📄
fastmail
(5.16 KB)
📄
featuremod
(1.92 KB)
📄
fetchfile
(422 B)
📄
find_and_fix_rpm_issues
(6.99 KB)
📄
find_outdated_services
(5.41 KB)
📄
find_pids_with_inotify_watch_on_path
(3.66 KB)
📄
fix-cpanel-perl
(28.82 KB)
📄
fix-listen-on-localhost
(3.52 KB)
📄
fix-web-vhost-configuration
(6.15 KB)
📄
fix_addon_permissions
(7.68 KB)
📄
fix_dns_zone_ttls
(1.34 KB)
📄
fix_innodb_tables
(4.05 KB)
📄
fix_pear_registry
(4.07 KB)
📄
fix_reseller_acls
(10.88 KB)
📄
fixetchosts
(4.32 KB)
📄
fixheaders
(572 B)
📄
fixmailinglistperms
(1008 B)
📄
fixmailman
(2.09 KB)
📄
fixnamedviews
(1.22 KB)
📄
fixndc
(413 B)
📄
fixquotas
(17.45 KB)
📄
fixrelayd
(1.74 KB)
📄
fixrndc
(16.48 KB)
📄
fixtar
(503 B)
📄
fixtlsversions
(4.7 KB)
📄
fixvaliases
(2 KB)
📄
fixwebalizer
(966 B)
📄
forcelocaldomain
(895 B)
📄
ftpfetch
(2.2 KB)
📄
ftpquotacheck
(8.31 KB)
📄
ftpsfetch
(2.36 KB)
📄
ftpupdate
(261 B)
📄
gather_update_log_stats
(4.25 KB)
📄
gather_update_logs_setupcrontab
(5.45 KB)
📄
gemwrapper
(1.74 KB)
📄
gencrt
(6.26 KB)
📄
generate_account_suspension_include
(5.7 KB)
📄
generate_google_drive_credentials
(1.11 KB)
📄
generate_google_drive_oauth_uri
(984 B)
📄
generate_maildirsize
(13.94 KB)
📄
gensysinfo
(1.16 KB)
📄
get_locale_from_legacy_name_info
(1.99 KB)
📄
getremotecpmove
(12.67 KB)
📄
grpck
(1.19 KB)
📄
hackcheck
(3.02 KB)
📄
hook
(1.45 KB)
📄
httpspamdetect
(2.66 KB)
📄
hulk-unban-ip
(3.93 MB)
📄
import_exim_data
(8.39 KB)
📄
importmydnsdb
(11.34 KB)
📄
increase_filesystem_limits
(891 B)
📄
initacls
(4.99 KB)
📄
initfpsuexec
(444 B)
📄
initquotas
(19.47 KB)
📄
initsuexec
(4.03 KB)
📄
install_cpanel_analytics
(1.93 KB)
📄
install_dovecot_fts
(1.57 KB)
📄
install_plugin
(2.8 KB)
📄
installpkg
(575 B)
📄
installpostgres
(6.56 KB)
📄
installsqlite3
(1.82 KB)
📄
ipcheck
(3.93 KB)
📄
ipusage
(7.45 KB)
📄
isdedicatedip
(602 B)
📄
jetbackup-check
(3.69 KB)
📄
killdns
(422 B)
📄
killdns-dnsadmin
(1.15 KB)
📄
killmysqluserprivs
(433 B)
📄
killmysqlwildcard
(1.15 KB)
📄
killpvhost
(853 B)
📄
killspamkeys
(937 B)
📄
link_3rdparty_binaries
(1.24 KB)
📄
linksubemailtomainacct
(3.17 KB)
📄
listcheck
(538 B)
📄
listsubdomains
(1.05 KB)
📄
litespeed-check
(3.86 KB)
📄
locale_export
(4.82 KB)
📄
locale_import
(4.35 KB)
📄
locale_info
(3.99 KB)
📄
logo.dat
(205 B)
📄
magicloader
(1.94 KB)
📄
maildir_converter
(6.08 KB)
📄
mailperm
(16.58 KB)
📄
mailscannerupdate
(2.42 KB)
📄
mainipcheck
(10 KB)
📄
maintenance
(46.62 KB)
📄
make_config
(407 B)
📄
make_hostname_unowned
(1.16 KB)
📄
manage_extra_marketing
(12.41 KB)
📄
manage_greylisting
(16.19 KB)
📄
manage_mysql_profiles
(20.02 KB)
📄
migrate-pdns-conf
(9.83 KB)
📄
migrate_local_ini_to_php_ini
(7.41 KB)
📄
migrate_whmtheme_file_to_userdata
(2.95 KB)
📄
mkwwwacctconf
(2.33 KB)
📄
modify_accounts
(4.09 KB)
📄
modify_featurelist
(9.2 KB)
📄
modify_packages
(3.65 KB)
📄
modsec_vendor
(15.63 KB)
📄
mysqlconnectioncheck
(6.55 KB)
📄
mysqlpasswd
(4.09 KB)
📄
named.ca
(1.57 KB)
📄
named.rfc1912.zones
(774 B)
📄
nixstatsagent.sh
(58.67 KB)
📄
notify_expiring_certificates
(9.37 KB)
📄
notify_expiring_certificates_on_linked_nodes
(1.33 KB)
📄
oopscheck
(1.12 KB)
📄
optimize_eximstats
(3.88 KB)
📄
patch_mail_spamassassin_compiledregexps_body_0
(2.39 KB)
📄
patchfdsetsize
(2.72 KB)
📄
pedquota
(2.26 KB)
📄
perform_sqlite_auto_rebuild_db_maintenance
(2.2 KB)
📄
perlinstaller
(528 B)
📄
perlmods
(1.18 KB)
📄
php_fpm_config
(9.73 KB)
📁
php_sandbox
📄
phpini_tidy
(687 B)
📄
pkgacct
(87.82 KB)
📄
post_snapshot
(3.16 KB)
📄
post_sync_cleanup
(6.09 KB)
📄
primary_virtual_host_migration
(2.44 KB)
📄
process_pending_cpanel_php_pear_registration
(3.49 KB)
📄
process_site_templates
(7.27 KB)
📄
proxydomains
(9.34 KB)
📄
ptycheck
(724 B)
📄
purge_modsec_log
(1.53 KB)
📄
purge_old_config_caches
(2.08 KB)
📄
pwck
(708 B)
📄
quickdnslookup
(1.13 KB)
📄
quickwhoisips
(2.29 KB)
📄
quota_auto_fix
(1.41 KB)
📄
quotacheck
(22.36 KB)
📄
rawchpass
(460 B)
📄
rdate
(4.8 KB)
📄
realadduser
(5.61 KB)
📄
realchpass
(3.26 KB)
📄
realperlinstaller
(5.67 KB)
📄
realrawchpass
(425 B)
📄
rebuild_available_addons_packages_cache
(1.27 KB)
📄
rebuild_available_rpm_addons_cache
(1.27 KB)
📄
rebuild_bandwidthdb_root_cache
(1.45 KB)
📄
rebuild_dbmap
(5.8 KB)
📄
rebuild_provider_openid_connect_links_db
(1.01 KB)
📄
rebuild_whm_chrome
(2.22 KB)
📄
rebuilddnsconfig
(26.21 KB)
📄
rebuildhttpdconf
(2.6 KB)
📄
rebuildinstalledssldb
(2.85 KB)
📄
rebuildippool
(509 B)
📄
rebuildnsdzones
(1.14 KB)
📄
rebuilduserssldb
(948 B)
📄
refresh-dkim-validity-cache
(5.97 KB)
📄
regenerate_tokens
(2.18 KB)
📄
reloadnsd
(821 B)
📄
remote_log_transfer
(11.6 KB)
📄
remove_dovecot_index_files
(5.89 KB)
📄
removeacct
(20.91 MB)
📄
rescan_user_dovecot_fts
(2.98 KB)
📄
reset_mail_quotas_to_sane_values
(6.82 KB)
📄
resetmailmanurls
(2.03 KB)
📄
resetquotas
(4.68 KB)
📄
restartsrv
(3.23 KB)
📄
restartsrv_apache
(422 B)
📄
restartsrv_apache_php_fpm
(9.9 MB)
📄
restartsrv_base
(9.9 MB)
📄
restartsrv_bind
(9.9 MB)
📄
restartsrv_chkservd
(427 B)
📄
restartsrv_clamd
(9.9 MB)
📄
restartsrv_cpanalyticsd
(9.9 MB)
📄
restartsrv_cpanel_php_fpm
(9.9 MB)
📄
restartsrv_cpanellogd
(9.9 MB)
📄
restartsrv_cpdavd
(9.9 MB)
📄
restartsrv_cpgreylistd
(9.9 MB)
📄
restartsrv_cphulkd
(9.9 MB)
📄
restartsrv_cpipv6
(9.9 MB)
📄
restartsrv_cpsrvd
(9.9 MB)
📄
restartsrv_crond
(9.9 MB)
📄
restartsrv_dnsadmin
(9.9 MB)
📄
restartsrv_dovecot
(9.9 MB)
📄
restartsrv_exim
(9.9 MB)
📄
restartsrv_eximstats
(504 B)
📄
restartsrv_ftpd
(426 B)
📄
restartsrv_ftpserver
(911 B)
📄
restartsrv_httpd
(9.9 MB)
📄
restartsrv_imap
(437 B)
📄
restartsrv_inetd
(2.47 KB)
📄
restartsrv_ipaliases
(9.9 MB)
📄
restartsrv_lmtp
(437 B)
📄
restartsrv_mailman
(9.9 MB)
📄
restartsrv_mydns
(9.9 MB)
📄
restartsrv_mysql
(9.9 MB)
📄
restartsrv_named
(777 B)
📄
restartsrv_nscd
(9.9 MB)
📄
restartsrv_nsd
(9.9 MB)
📄
restartsrv_p0f
(9.9 MB)
📄
restartsrv_pdns
(9.9 MB)
📄
restartsrv_pop3
(437 B)
📄
restartsrv_postgres
(427 B)
📄
restartsrv_postgresql
(9.9 MB)
📄
restartsrv_powerdns
(442 B)
📄
restartsrv_proftpd
(9.9 MB)
📄
restartsrv_pureftpd
(9.9 MB)
📄
restartsrv_queueprocd
(9.9 MB)
📄
restartsrv_rsyslog
(9.9 MB)
📄
restartsrv_rsyslogd
(437 B)
📄
restartsrv_spamd
(9.9 MB)
📄
restartsrv_sshd
(9.9 MB)
📄
restartsrv_syslogd
(2.4 KB)
📄
restartsrv_tailwatchd
(9.9 MB)
📄
restartsrv_unknown
(9.9 MB)
📄
restartsrv_xinetd
(422 B)
📄
restorecpuserfromcache
(1.96 KB)
📄
restorepkg
(36.49 MB)
📄
rfc1912_zones.tar
(10 KB)
📄
rpmup
(4.77 KB)
📄
rsync-user-homedir.pl
(5.76 KB)
📄
run_if_exists
(512 B)
📄
run_plugin_lifecycle
(3.53 KB)
📄
runstatsonce
(440 B)
📄
runweblogs
(1.02 KB)
📄
sa-update_wrapper
(3.34 KB)
📄
safetybits.pl
(844 B)
📄
secureit
(4.72 KB)
📄
securemysql
(4.54 KB)
📄
securerailsapps
(3.58 KB)
📄
securetmp
(15.99 KB)
📄
sendicq
(474 B)
📄
servicedomains
(9.34 KB)
📄
set_mailman_archive_perms
(1.75 KB)
📄
set_php_memory_limits
(3.67 KB)
📄
setpostgresconfig
(6.04 KB)
📄
setup_greylist_db
(16.19 KB)
📄
setup_modsec_db
(1.3 KB)
📄
setup_systemd_timer_for_plugins
(3.92 KB)
📄
setupftpserver
(10.47 KB)
📄
setupmailserver
(9.55 KB)
📄
setupnameserver
(13.75 KB)
📄
shrink_modsec_ip_database
(12.97 KB)
📄
simpleps
(3.05 KB)
📄
slurp_exim_mainlog
(5.78 KB)
📄
smartcheck
(15.13 KB)
📄
smtpmailgidonly
(8.15 KB)
📄
snapshot_prep
(5.88 KB)
📄
spamassassin_dbm_cleaner
(5.85 KB)
📄
spamassassindisable
(3.74 KB)
📄
spamboxdisable
(2.27 KB)
📄
sshcontrol
(14.38 KB)
📄
ssl_crt_status
(3.84 KB)
📄
suspendacct
(18.01 KB)
📄
suspendmysqlusers
(4.42 KB)
📄
swapip
(3.82 KB)
📄
sync-mysql-users-from-grants
(1.2 KB)
📄
sync_child_accounts
(1.77 KB)
📄
sync_contact_emails_to_cpanel_users_files
(1.14 KB)
📄
synccpaddonswithsqlhost
(6.59 KB)
📄
synctransfers
(1.92 KB)
📄
syslog_check
(1.36 KB)
📄
sysup
(645 B)
📄
test_sa_compiled
(1.07 KB)
📄
transfer_account_as_user
(2.34 KB)
📄
transfer_accounts_as_root
(4.76 KB)
📄
transfer_in_progress
(3.08 KB)
📄
transfer_in_progress.pod
(312 B)
📄
transfermysqlusers
(9.53 MB)
📄
try-later
(7.95 KB)
📄
unblockip
(667 B)
📄
uninstall_cpanel_analytics
(1.2 KB)
📄
uninstall_dovecot_fts
(562 B)
📄
uninstall_plugin
(2.84 KB)
📄
unlink_service_account
(2.62 KB)
📄
unpkgacct
(4.6 KB)
📄
unslavenamedconf
(863 B)
📄
unsuspendacct
(17.8 KB)
📄
unsuspendmysqlusers
(6.71 KB)
📄
upcp
(31.56 KB)
📄
upcp-running
(2.7 KB)
📄
upcp.static
(708.6 KB)
📄
update-packages
(4.77 KB)
📄
update_apachectl
(480 B)
📄
update_db_cache
(430 B)
📄
update_dkim_keys
(1.45 KB)
📄
update_exim_rejects
(1.21 KB)
📄
update_existing_mail_quotas_for_account
(4.78 KB)
📄
update_known_proxy_ips
(1002 B)
📄
update_local_rpm_versions
(4.56 KB)
📄
update_mailman_cache
(8.34 KB)
📄
update_mysql_systemd_config
(1.25 KB)
📄
update_neighbor_netblocks
(487 B)
📄
update_sa_config
(2.14 KB)
📄
update_spamassassin_config
(10.73 KB)
📄
update_users_jail
(691 B)
📄
update_users_vhosts
(801 B)
📄
updatedomainips
(605 B)
📄
updatenameserverips
(1.66 KB)
📄
updatenow
(5.18 KB)
📄
updatenow.static
(1.91 MB)
📄
updatesigningkey
(1.95 KB)
📄
updatessldomains
(1.81 KB)
📄
updatesupportauthorizations
(2.49 KB)
📄
updateuserdatacache
(2.47 KB)
📄
updateuserdomains
(774 B)
📄
upgrade_bandwidth_dbs
(2.22 KB)
📄
upgrade_subaccount_databases
(2.73 KB)
📄
userdata_wildcard_cleanup
(5.74 KB)
📄
userdirctl
(5.01 KB)
📄
validate_sshkey_passphrase
(1.21 KB)
📄
verify_api_spec_files
(757 B)
📄
verify_pidfile
(1.96 KB)
📄
verify_vhost_includes
(7.34 KB)
📄
vps_optimizer
(7.82 KB)
📄
vzzo-fixer
(725 B)
📄
whmlogin
(2.33 KB)
📄
whoowns
(1.13 KB)
📄
wpt_license
(6.27 MB)
📄
wwwacct
(24.57 MB)
📄
wwwacct2
(88 B)
📄
xfer_rcube_schema_migrate.pl
(2.4 KB)
📄
xfer_rcube_uid_resolver.pl
(1.8 KB)
📄
xferpoint
(3.13 KB)
📄
xfertool
(16.14 KB)
📄
zoneexists
(800 B)
Editing: convert_roundcube_mysql2sqlite
#!/usr/local/cpanel/3rdparty/bin/perl # cpanel - scripts/convert_roundcube_mysql2sqlite Copyright 2022 cPanel, L.L.C. # All rights reserved. # copyright@cpanel.net http://cpanel.net # This code is subject to the cPanel license. Unauthorized copying is prohibited package Script::RCube::Mysql2Sqlite; use strict; ## no critic (RequireUseWarnings) use Try::Tiny; use Cpanel::AccessIds::ReducedPrivileges (); use Cpanel::Config::LoadCpConf (); use Cpanel::MysqlUtils (); use Cpanel::DbUtils (); use Cpanel::MysqlUtils::Connect (); use Cpanel::MysqlUtils::Command (); use Cpanel::MysqlRun (); use Cpanel::Config::LoadUserDomains (); use Cpanel::Email::RoundCube (); use Cpanel::Email::RoundCube::DBI (); use Cpanel::Quota::Temp (); use Cpanel::PwCache (); use Cpanel::Logger (); use Cpanel::Filesys::Home (); use File::Basename (); use File::Copy (); use File::Path (); use File::Slurper (); use DBI; use XML::Simple (); use IPC::Open3 (); $XML::Simple::PREFERRED_PARSER = "XML::SAX::PurePerl"; my $sqlite_table_file = '/usr/local/cpanel/base/3rdparty/roundcube/SQL/sqlite.initial.sql'; my $log_file = '/usr/local/cpanel/logs/roundcube_sqlite_convert_log'; my %opts = ( 'alternate_logfile' => $log_file ); my $logger = Cpanel::Logger->new( \%opts ); my $mysql_dbname = 'roundcube'; my $dbh; # If this updates, then do_all_rcube_xml_to_db() needs to be updated to match my @dumptables = qw(users identities contacts contactgroups contactgroupmembers calendars caldav_calendars events caldav_events attachments caldav_attachments itipinvitations responses); my $time = time(); ## if invoked as a script, there is nothing in the call stack my $invoked_as_script = !caller(); __PACKAGE__->script(@ARGV) if ($invoked_as_script); my $mysql_plus_sqlite; sub script { my ( $package, $opt_user, $opt_dbname ) = @_; my $cpconf = Cpanel::Config::LoadCpConf::loadcpconf(); my $db_type = 'mysql'; if ( exists $cpconf->{'roundcube_db'} ) { $db_type = 'sqlite' if $cpconf->{'roundcube_db'} eq 'sqlite'; $mysql_plus_sqlite = $cpconf->{'roundcube_db'} eq 'mysql_plus_sqlite'; } if ( defined $opt_user ) { ## the optional user arg feature is currently *only* called as a script; ## otherwise, all these exits would check $invoked_as_script ## Transfers.pm is the only user of $opt_user, which it uses in conjunction with $opt_user; ## this changes the global variable $opt_dbname just in time to be used in &init_check if ( defined $opt_dbname ) { $mysql_dbname = $opt_dbname; } if ( $db_type eq 'sqlite' && init_check($cpconf) ) { my $wasSuccess = convert_mysql_roundcube_to_sqlite($opt_user); if ($wasSuccess) { ## Transfers.pm operates over a different dbname, and is responsible for the ## removal of the temporary database unless ( defined $opt_dbname ) { Cpanel::Email::RoundCube::archive_and_drop_mysql_roundcube($logger); } return 1; } die "Conversion for user '$opt_user' was not successful.\n"; } ## SOMEDAY @GOLIVE: update-roundcube-sqlite-db gets moved to update-roundcube-db. Update this message and the conditional @GOIVE. die "The optional user argument is to be used only when Roundcube has been converted to sqlite. Use bin/update-roundcube-sqlite-db Exiting.\n"; } ## Has the roundcube conversion already happened? ## note: this check is here, and not in init_check, because Transfers.pm calls into this ## from a different context. The src server uses MySQL but the dest machine has done ## the sqlite conversion. if ( $db_type eq 'sqlite' ) { $logger->info("Roundcube conversion already occurred, bailing out."); if ($invoked_as_script) { exit(0); } return 1; } unless ( init_check($cpconf) ) { if ($invoked_as_script) { exit(0); } return 1; } my %TRUEDOMAINS; Cpanel::Config::LoadUserDomains::loadtrueuserdomains( \%TRUEDOMAINS ); my @users = sort values %TRUEDOMAINS; my $success_cnt = 0; my $success_verify = scalar @users; for my $user (@users) { $success_cnt += convert_mysql_roundcube_to_sqlite($user); } unless ( $success_cnt == $success_verify ) { $logger->warn("Roundcube Mysql to sqlite conversion was not completely successful. Please check $log_file for details."); if ($invoked_as_script) { exit(1); } return; } ## Conversion is a success, so switch Roundcube configuration to use SQLite before dropping the database. my $DIR = '/usr/local/cpanel/base/3rdparty'; Cpanel::Email::RoundCube::generate_roundcube_config_sqlite( $DIR, $logger ); ## if the conversion was a complete success (as checked above), we can safely archive and ## delete the MySQL roundcube database. This will prevent future problems with transfers, ## and in fact is the final solution for the original Roundcube case 12162! ## already asserted via &init_check my $archive_success = Cpanel::Email::RoundCube::archive_and_drop_mysql_roundcube($logger); require Cpanel::Config::CpConfGuard; my $cpconf_guard = Cpanel::Config::CpConfGuard->new(); $cpconf->{'roundcube_db'} = $cpconf_guard->{'data'}->{'roundcube_db'} = 'sqlite'; $cpconf_guard->save(); if ($invoked_as_script) { Cpanel::Email::RoundCube::restart_cpsrvd(); exit(0); } return 1; } sub convert_mysql_roundcube_to_sqlite { my ($user) = @_; my @domains = Cpanel::Email::RoundCube::collect_domains($user); my @uid_name_pairs = collect_roundcube_user_info( \@domains, $user ); ## returning 1: no conversion necessary return 1 unless ( scalar @uid_name_pairs ); ## e.g. '/home' my $best_mnt_point = Cpanel::Filesys::Home::get_homematch_with_most_free_space(); my $tmpdir_root_base = "$best_mnt_point/roundcube_convert"; if ( -d $tmpdir_root_base ) { rename( $tmpdir_root_base, "$tmpdir_root_base.$time" ); } mkdir($tmpdir_root_base); my $tmpdir_root_sys = "$tmpdir_root_base/$user"; mkdir($tmpdir_root_sys); my @pwinfo = Cpanel::PwCache::getpwnam($user); my ( $mmuid, $mmgid, $homedir ) = @pwinfo[ 2, 3, 7 ]; my $tmpdir_emailuser_base = "$homedir/tmp/roundcube_convert"; my $success_cnt = 0; my %has_sqlite_trapper_keeper; for my $email_user_info (@uid_name_pairs) { my ( $uid, $email_user ) = @$email_user_info; if ($mysql_plus_sqlite) { my $db_path = _roundcube_db_path( $user, $homedir, $email_user ); $has_sqlite_trapper_keeper{$db_path} = -f $db_path; next if $has_sqlite_trapper_keeper{$db_path}; } my $tmpdir_root_sys_emailuser = "$tmpdir_root_sys/$email_user"; mkdir($tmpdir_root_sys_emailuser); do_mysqldumps_for_user( $uid, $tmpdir_root_sys_emailuser ); } ## ensures that $best/rcube/$sysuser/$emailuser is readable by $sysuser, ## so that $sysuser can move it to their $homedir after the setuid my $tempquota = Cpanel::Quota::Temp->new( user => $user, log => 1 ); $tempquota->disable(); my $rv_chmod = system("chmod -R 700 $tmpdir_root_sys"); my $rv_chown = system("chown -Rf $mmuid:$mmgid $tmpdir_root_sys"); for my $email_user_info (@uid_name_pairs) { my ( $uid, $email_user ) = @$email_user_info; if ($mysql_plus_sqlite) { my $db_path = _roundcube_db_path( $user, $homedir, $email_user ); if ( $has_sqlite_trapper_keeper{$db_path} ) { $logger->info("$email_user already has an SQLite DB, skipping..."); $success_cnt++; next; } } chdir($tmpdir_root_base); my $tmpdir_root_sys_emailuser = "$tmpdir_root_sys/$email_user"; my $rv_convert = xml_to_sqlite( $user, $tmpdir_root_sys_emailuser, $tmpdir_emailuser_base, $email_user, $homedir ); unless ($rv_convert) { $logger->warn("Conversion was not successful for user '$user'."); } $success_cnt += $rv_convert; ## note: no need to "rmdir($tmpdir_root_sys_emailuser)", as this dir was moved ## to $homedir/tmp/rcube/$emailuser (during the setuid) ## SOMEDAY @GOLIVE: remove the mysql roundcube database (steal mysql/roundcube backup/archive ## clauses from update-roundcube-db, the mysql version) } $tempquota->restore(); rmdir($tmpdir_root_sys); rmdir($tmpdir_root_base); return $success_cnt == scalar(@uid_name_pairs); } sub collect_roundcube_user_info { my ( $ar_domains, $user ) = @_; my $regexp = '@(' . join( '|', map { s/\./\\./gr } @$ar_domains ) . ')$'; my @ids; try { @ids = $dbh->selectall_array( "SELECT user_id, username FROM users WHERE username REGEXP ? or username = ?", {}, $regexp, $user ); }; return @ids; } ## Stolen from pkgacct; needs to be modularized! ## SOMEDAY: this version has converged; move pkgacct version to module, and get rid of below sub mysqldumpdb { my ($args) = @_; my @options = @{ $args->{'options'} }; my $db = $args->{'db'}; my $table = $args->{'table'}; my $file = $args->{'file'}; my $file_mode = $args->{'append'} ? '>>' : '>'; # for testing my @extra_options = _get_extra_options(); my $mysqldump = Cpanel::DbUtils::find_mysqldump(); my @db = ($db); if ($table) { push @db, $table; } my $pid = IPC::Open3::open3( my $w, my $r, '', $mysqldump, @extra_options, @options, @db ); my $first_line = 1; if ( open( my $fh, $file_mode, $file ) ) { while (<$r>) { if ( $first_line && ( !$_ || m/^mysqldump:/ ) ) { warn join( '.', @db ) . ': ' . $_; close $w; close $r; waitpid( $pid, 0 ); $first_line = 0; my $mysqlcheck = Cpanel::DbUtils::find_mysqlcheck(); system( $mysqlcheck, '--repair', @extra_options, @db ); $pid = IPC::Open3::open3( $w, $r, '', $mysqldump, @extra_options, @options, @db ); } else { print {$fh} $_; } } } close $w; close $r; waitpid( $pid, 0 ); return; } sub do_mysqldumps_for_user { my ( $uid, $tmp_convertdir ) = @_; for my $tbl (@dumptables) { ## these two tables do not key off user_id next if grep { $tbl eq $_ } qw{contactgroupmembers events ical_events caldav_events attachments ical_attachments caldav_attachments caldav_calendars}; ## XML output, not suppressing table structure, with a where clause on user_id list my @opts = ( '--xml', '-w', qq{user_id = $uid} ); mysqldumpdb( { 'options' => [@opts], 'db' => $mysql_dbname, 'file' => "$tmp_convertdir/rcube.$tbl.xml", 'table' => $tbl } ); } ## contactgroupmembers is gathered in two steps, otherwise would be a LEFT JOIN that ## &mysqldumpdb can not currently support my $contacts = _get_ids_from_col( 'contactgroups', 'contactgroup_id', $uid, 'user_id' ); _dump_stuff_in( 'contactgroupmembers', 'contactgroup_id', $contacts, $tmp_convertdir ) if scalar(@$contacts); # Handle events & attachments for calendaring, as those are identified by # calendar & event respectively. foreach my $type (qw{default ical caldav}) { my $prefix = $type eq 'default' ? '' : "${type}_"; my $cals = _get_ids_from_col( "${prefix}calendars", 'calendar_id', $uid, 'user_id' ); next if !scalar(@$cals); my @events2export; my @attachments2export; foreach my $cal (@$cals) { my $events = _get_ids_from_col( "${prefix}events", 'event_id', $cal, 'calendar_id' ); # Can't have attachments if no events exist to attach em to, so # next here is appropriate. next if !scalar(@$events); push @events2export, @$events; my $attachments = _get_ids_from_col( "${prefix}attachments", 'attachment_id', $events, 'event_id' ); next if !scalar(@$attachments); push @attachments2export, @$attachments; } _dump_stuff_in( "${prefix}events", 'event_id', \@events2export, $tmp_convertdir ) if @events2export; _dump_stuff_in( "${prefix}attachments", 'attachment_id', \@attachments2export, $tmp_convertdir ) if @attachments2export; } return undef; } # Only here so that I don't "repeat myself". sub _get_ids_from_col { my ( $tbl, $col, $search_term, $filter_on, $filter_mode ) = @_; my @ids; try { # Quoting anything other than values here is actually harmful. # Don't do it as such. Same for IN statement here. my $where_clause = "WHERE $filter_on = ?"; if ( ref $search_term eq 'ARRAY' ) { my $count = scalar(@$search_term); die "Bad call to _get_ids_from_col: no search term specified!" if !$count; my $qs = ( '?,' x ( $count - 1 ) ) . '?'; $where_clause = "WHERE $filter_on IN ($qs)"; @ids = map { $_->[0] } $dbh->selectall_array( "SELECT $col FROM $tbl $where_clause", {}, @$search_term ); } else { @ids = map { $_->[0] } $dbh->selectall_array( "SELECT $col FROM $tbl $where_clause", {}, $search_term ); } } catch { $logger->warn( "Attempt to find necessary data from $col in $tbl failed: " . $dbh->errstr() ); }; return \@ids; } # Only here so that I don't "repeat myself". sub _dump_stuff_in { my ( $tbl, $col, $in, $tmp_convertdir ) = @_; my $csv = join( ',', @$in ); mysqldumpdb( { 'options' => [ '--xml', '-w', qq{$col IN ($csv)} ], 'db' => $mysql_dbname, 'file' => "$tmp_convertdir/rcube.$tbl.xml", 'table' => $tbl, } ); return; } sub create_rcube_sqlite_tables { my ($dbh) = @_; Cpanel::Email::RoundCube::DBI::ensure_schema_update( $dbh, 'sqlite' ); return undef; } # This used to do all sorts of things like telling rcube_xml_to_db what needed # quoting, what needed to be set to NULL, etc. -- thankfully all that is # absolutely unnecessary with sqlite3. sub do_all_rcube_xml_to_db { my ( $dbh, $xmldir ) = @_; my $success_cnt = 0; ## verify the number of XML docs to convert to SQL my $success_verify = scalar(@dumptables); { ## note: there is a disparity in the sqlite.initial.sql, and the "table_structure" for "users"; ## preferences is not NULLABLE. my @_cols = qw(user_id username mail_host created last_login language preferences); $success_cnt += rcube_xml_to_db( $xmldir, 'users', $dbh, \@_cols ); } { my @_cols = qw( identity_id user_id changed del standard name organization email reply-to bcc signature html_signature); $success_cnt += rcube_xml_to_db( $xmldir, 'identities', $dbh, \@_cols ); } { my @_cols = qw(contact_id user_id changed del name email firstname surname vcard); $success_cnt += rcube_xml_to_db( $xmldir, 'contacts', $dbh, \@_cols ); } ## new contactgroups table { my @_cols = qw( contactgroup_id user_id changed del name ); $success_cnt += rcube_xml_to_db( $xmldir, 'contactgroups', $dbh, \@_cols ); } ## new contactgroupmembers table ## guard clause: contactgroupmembers is gathered conditionally, so it is the only file which ## may not exist if ( -e _xml_file( $xmldir, 'contactgroupmembers' ) ) { my @_cols = qw( contactgroup_id contact_id created ); $success_cnt += rcube_xml_to_db( $xmldir, 'contactgroupmembers', $dbh, \@_cols ); } else { $success_cnt++; } # Restore calendar data foreach my $cal_tbl (qw{calendars caldav_calendars}) { unless ( -e _xml_file( $xmldir, $cal_tbl ) ) { $success_cnt++; next; } my @_cols = qw( calendar_id user_id name color showalarms ); push( @_cols, qw{ical_url ical_user ical_pass ical_last_change} ) if $cal_tbl eq 'ical_calendars'; push( @_cols, qw{readonly caldav_url caldav_tag caldav_user caldav_pass caldav_oauth_provider caldav_last_change} ) if $cal_tbl eq 'caldav_calendars'; $success_cnt += rcube_xml_to_db( $xmldir, $cal_tbl, $dbh, \@_cols ); } foreach my $evt_tbl (qw{events caldav_events}) { unless ( -e _xml_file( $xmldir, $evt_tbl ) ) { $success_cnt++; next; } my @_cols = qw( event_id calendar_id recurrence_id uid instance isexception created changed sequence start end recurrence title description location categories url all_day free_busy priority sensitivity status alarms attendees notifyat ); push( @_cols, qw{ical_url ical_last_change} ) if $evt_tbl eq 'ical_events'; push( @_cols, qw{caldav_url caldav_tag caldav_last_change} ) if $evt_tbl eq 'caldav_events'; $success_cnt += rcube_xml_to_db( $xmldir, $evt_tbl, $dbh, \@_cols ); } foreach my $atc_tbl (qw{attachments caldav_attachments}) { unless ( -e _xml_file( $xmldir, $atc_tbl ) ) { $success_cnt++; next; } my @_cols = qw( attachment_id event_id filename mimetype size data ); $success_cnt += rcube_xml_to_db( $xmldir, $atc_tbl, $dbh, \@_cols ); } if ( -e _xml_file( $xmldir, 'itipinvitations' ) ) { my @_cols = qw(token event_uid user_id event expires cancelled); $success_cnt += rcube_xml_to_db( $xmldir, 'itipinvitations', $dbh, \@_cols ); } else { $success_cnt++; } ## Restore responses table, if available if ( -e _xml_file( $xmldir, 'responses' ) ) { my @_cols = qw( response_id user_id name data is_html changed del ); $success_cnt += rcube_xml_to_db( $xmldir, 'responses', $dbh, \@_cols ); } else { $success_cnt++; } return $success_cnt == $success_verify; } sub _xml_file { my ( $xmldir, $tbl ) = @_; return "$xmldir/rcube.$tbl.xml"; } sub rcube_xml_to_db { my ( $xmldir, $tbl, $dbh, $ar_cols ) = @_; my $xml_fname = _xml_file( $xmldir, $tbl ); my %common_opts = ( ForceArray => 1, KeyAttr => [], ContentKey => '__content' ); ## $ref->{database}->[0]->{table_data}->[0]->{row}->[$x]->{field}->[$x]->{__content} my $ref; my $err; try { $ref = XML::Simple::XMLin( $xml_fname, %common_opts ); } catch { $err = $_; }; if ($err) { $logger->warn("Failed to load xml for $tbl: $err"); return; } my $rows = $ref->{'database'}->[0]->{'table_data'}->[0]->{'row'}; ## FWIW, Perl does not complain if $rows is undef, when called in a for loop context. for my $row (@$rows) { # XXX Placeholders don't work on columns, but stuff with dashes # still needs quoting. As such, quote all col names. # Sorry, couldn't find a way around it. my $update_hr = { map { $dbh->quote_identifier( $_->{'name'} ) => $_->{'__content'} || '' } @{ $row->{'field'} } }; my $rv = $dbh->do( rcube_make_sql( $tbl, $update_hr ) ); unless ($rv) { $logger->warn( "Conversion was not successful: " . $dbh->errstr() ); return; } } return 1; } sub rcube_make_sql { my ( $table, $update_hr ) = @_; # Lets hope these actually are nonzero my @values = values(%$update_hr); my $qs = ( '?,' x ( scalar(@values) - 1 ) ) . '?'; return ( "INSERT OR REPLACE INTO $table (" . join( ",", keys(%$update_hr) ) . ") VALUES ($qs)", undef, @values, ); } sub check_mysqlup { ## note: this call is cached with a ttl of 600 my $isrunning = Cpanel::MysqlRun::running(); if ($isrunning) { return 1; } return undef; } sub ensure_roundcube_tables { my ($dbh) = @_; my %tables = map { $_ => undef } $dbh->tables(); # system table is created during a migration my @expected = qw(session cache system); push( @expected, @dumptables ); for my $exp (@expected) { if ( !exists $tables{$exp} && !exists $tables{qq{"main"."$exp"}} ) { $logger->warn("ERROR: missing $exp table"); return; } } return 1; } sub init_check { my ($cpconf) = @_; ## Ensure root return unless ( 0 == $> ); unless ( -e $sqlite_table_file ) { $logger->info("Roundcube is not installed; conversion is irrelevant."); return; } if ( exists $cpconf->{'skiproundcube'} and $cpconf->{'skiproundcube'} ) { $logger->info("Roundcube should be skipped, bailing out."); return; } my $mysqlup = check_mysqlup(); unless ($mysqlup) { $logger->warn("Mysql not currently running, bailing out."); return; } ## Does this installation use Roundcube/MySQL? unless ( Cpanel::MysqlUtils::Command::db_exists($mysql_dbname) ) { $logger->warn("mysql database $mysql_dbname is missing"); return; } my $error; try { my $dbc = Cpanel::MysqlUtils::Connect->new( database => $mysql_dbname ); $dbh = $dbc->db_handle(); } catch { $error = $_; }; if ($error) { $logger->warn("Failed to connect to MySQL database '$mysql_dbname'."); return; } return 1; } ############################################################## # These functions allow tests to use a temporary MySQL server sub _get_extra_options_string { if ( scalar _get_extra_options() ) { return join( ' ', _get_extra_options() ); } return; } sub _get_extra_options { return @Cpanel::MysqlUtils::_EXTRA_MYSQL_ARGS; } # ############################################################## sub xml_to_sqlite { my ( $user, $tmpdir_root, $tmpdir_convert_base, $email_user, $homedir ) = @_; my $code_ref = _closure_maker( $user, $tmpdir_root, $tmpdir_convert_base, $email_user, $homedir ); #fixup a user who's purposefully deleted their "etc" dir Cpanel::AccessIds::ReducedPrivileges::call_as_user( sub { File::Path::make_path( File::Basename::dirname( _roundcube_db_path( $user, $homedir, $email_user ) ), { chmod => 0750 } ); }, $user, 'mail' ); my $rv = Cpanel::AccessIds::ReducedPrivileges::call_as_user( $code_ref, $user ); return $rv; } sub _roundcube_db_path { my ( $system_user, $homedir, $mail_account ) = @_; my ( $mail_user, $mail_domain ) = split( '@', $mail_account ); my $db_path = ( defined $mail_domain ) ? "$homedir/etc/$mail_domain/$mail_user.rcube.db" : "$homedir/etc/$mail_user.rcube.db"; return $db_path; } ## defining as a closure, as ::run_as_user does not currently handle sub args sub _closure_maker { my ( $system_user, $tmpdir_root, $tmpdir_convert_base, $email_user, $homedir ) = @_; my $code_ref = sub { my $tmp_convertdir = "$tmpdir_convert_base/$email_user"; if ( -d $tmp_convertdir ) { rename( $tmp_convertdir, "$tmp_convertdir.$time" ); } # Copy the XML files over to the user's tempdir. # Hopefully these are simple renames. But directories may be on different # mount points requiring a copy. File::Path::make_path($tmp_convertdir); for my $table (@dumptables) { my $src = _xml_file( $tmpdir_root, $table ); my $dest = _xml_file( $tmp_convertdir, $table ); if ( -e $src ) { File::Copy::move( $src, $dest ); } } # Clean up empty source directory. # This will fail if the above moves failed. rmdir($tmpdir_root); ## note: similar clause in cpsrvd to set up $ENV{'_RCUBE'} my ( $mail_user, $mail_domain ) = split( '@', $email_user ); my $rcube_sqlite_loc8 = _roundcube_db_path( $system_user, $homedir, $email_user ); if ( -e $rcube_sqlite_loc8 ) { rename( $rcube_sqlite_loc8, "$rcube_sqlite_loc8.$time" ); } my $dbh = DBI->connect( "dbi:SQLite:dbname=$rcube_sqlite_loc8", "", "" ) or do { $logger->info("Connection error to $rcube_sqlite_loc8: $!"); return; }; ## note: $dbh->{sqlite_version} is confirmed 2.8.15 create_rcube_sqlite_tables($dbh); return unless ensure_roundcube_tables($dbh); unless ( do_all_rcube_xml_to_db( $dbh, $tmp_convertdir ) ) { $logger->info("Conversion of roundcube XML dump to sqlite database failed"); return; } $dbh->disconnect(); chmod( 0600, $rcube_sqlite_loc8 ); ## the return values on these rm calls are not critical ## only remove the /tmp directory if the XML converion has been successful thus far try { File::Path::remove_tree($tmp_convertdir); }; ## attempt to rmdir, which will intentionally fail if any of the converts did not succeed rmdir($tmpdir_convert_base); return 1; }; return $code_ref; } 1;
Upload File
Create Folder