{"id":305955,"date":"2026-05-29T21:16:59","date_gmt":"2026-05-29T21:16:59","guid":{"rendered":"https:\/\/ca.wordpress.org\/plugins\/permission-manager\/"},"modified":"2026-05-29T22:45:22","modified_gmt":"2026-05-29T22:45:22","slug":"plugseal","status":"publish","type":"plugin","link":"https:\/\/tir.wordpress.org\/plugins\/plugseal\/","author":13803083,"comment_status":"closed","ping_status":"closed","template":"","meta":{"version":"0.1.0","stable_tag":"0.1.0","tested":"7.0","requires":"6.6","requires_php":"8.2","requires_plugins":null,"header_name":"PlugSeal","header_author":"Marc Armengou","header_description":"Control what each active plugin is allowed to do. Allow or deny specific permissions per plugin, with immediate effect.","assets_banners_color":"","last_updated":"2026-05-29 22:45:22","external_support_url":"","external_repository_url":"","donate_link":"","header_plugin_uri":"https:\/\/wordpress.org\/plugins\/plugseal\/","header_author_uri":"https:\/\/www.marcarmengou.com\/","rating":0,"author_block_rating":0,"active_installs":0,"downloads":30,"num_ratings":0,"support_threads":0,"support_threads_resolved":0,"author_block_count":0,"sections":["description","installation","faq","changelog"],"tags":{"0.1.0":{"tag":"0.1.0","author":"Marc4","date":"2026-05-29 22:45:22"}},"upgrade_notice":[],"ratings":[],"assets_icons":{"icon-256x256.jpg":{"filename":"icon-256x256.jpg","revision":3554199,"resolution":"256x256","location":"assets","locale":"","width":1024,"height":1024}},"assets_banners":[],"assets_blueprints":{"blueprint.json":{"filename":"blueprint.json","revision":3554243,"resolution":false,"location":"assets","locale":"","contents":"{\"landingPage\":\"\\\/wp-admin\\\/options-general.php?page=plugseal\",\"preferredVersions\":{\"php\":\"8.3\",\"wp\":\"latest\"},\"phpExtensionBundles\":[\"kitchen-sink\"],\"features\":{\"networking\":true},\"steps\":[{\"step\":\"installPlugin\",\"options\":{\"activate\":true},\"pluginData\":{\"resource\":\"wordpress.org\\\/plugins\",\"slug\":\"plugseal\"}},{\"step\":\"login\",\"username\":\"admin\",\"password\":\"password\"}]}"}},"all_blocks":[],"tagged_versions":["0.1.0"],"block_files":[],"assets_screenshots":[],"screenshots":[]},"plugin_section":[],"plugin_tags":[1912,31093,895,600],"plugin_category":[54],"plugin_contributors":[221849],"plugin_business_model":[],"class_list":["post-305955","plugin","type-plugin","status-publish","hentry","plugin_tags-access-control","plugin_tags-hardening","plugin_tags-permissions","plugin_tags-security","plugin_category-security-and-spam-protection","plugin_contributors-marc4","plugin_committers-marc4"],"banners":[],"icons":{"svg":false,"icon":"https:\/\/ps.w.org\/plugseal\/assets\/icon-256x256.jpg?rev=3554199","icon_2x":"https:\/\/ps.w.org\/plugseal\/assets\/icon-256x256.jpg?rev=3554199","generated":false},"screenshots":[],"raw_content":"<!--section=description-->\n<p>PlugSeal gives administrators granular control over what each active plugin is allowed to do, inspired by Android app permissions and Flatseal for Flatpak. Each active plugin is listed in the settings page. For each plugin, administrators can allow or deny individual permissions with immediate effect. All permissions are allowed by default, so no existing functionality is broken until an administrator explicitly restricts it.<\/p>\n\n<p><strong>Permissions covered:<\/strong><\/p>\n\n<ul>\n<li><code>db:read<\/code> \/ <code>db:write<\/code> \u2014 database queries via $wpdb<\/li>\n<li><code>db:read:users<\/code> \/ <code>db:write:users<\/code> \u2014 read and write access to user data (also covers wp_delete_user and wp_update_user)<\/li>\n<li><code>http:outbound<\/code> \u2014 outbound HTTP requests via the WordPress HTTP API<\/li>\n<li><code>options:read<\/code> \/ <code>options:write<\/code> \u2014 WordPress options via get_option \/ update_option (see limitations)<\/li>\n<li><code>email:send<\/code> \u2014 sending email via wp_mail()<\/li>\n<li><code>cron:write<\/code> \u2014 scheduling events via wp_schedule_event()<\/li>\n<li><code>transients:write<\/code> \u2014 writing transients via set_transient()<\/li>\n<li><code>users:create<\/code> \u2014 creating users via wp_create_user() (updates and deletes are covered by db:write:users)<\/li>\n<li><code>rest:register<\/code> \u2014 registering REST API endpoints via register_rest_route()<\/li>\n<li><code>shortcode:register<\/code> \u2014 registering shortcodes via add_shortcode()<\/li>\n<li><code>rewrite:register<\/code> \u2014 registering rewrite rules via add_rewrite_rule()<\/li>\n<li><code>admin:menu<\/code> \u2014 adding entries to the admin menu and submenus<\/li>\n<li><code>dashboard:widget<\/code> \u2014 adding dashboard widgets via wp_add_dashboard_widget()<\/li>\n<li><code>hooks:frontend<\/code> \u2014 hooking into frontend hooks (wp_head, wp_footer, the_content, wp_enqueue_scripts...)<\/li>\n<li><code>hooks:admin<\/code> \u2014 hooking into admin hooks (admin_head, admin_notices, admin_enqueue_scripts...)<\/li>\n<li><code>hooks:auth<\/code> \u2014 hooking into authentication hooks (wp_login, wp_logout, user_register, authenticate...)<\/li>\n<li><code>hooks:content<\/code> \u2014 hooking into content hooks (save_post, delete_post, pre_get_posts, wp_handle_upload...)<\/li>\n<li><code>hooks:lifecycle<\/code> \u2014 hooking into plugin and theme lifecycle hooks (activated_plugin, deactivated_plugin, switch_theme...)<\/li>\n<\/ul>\n\n<p><strong>Honest limitations:<\/strong><\/p>\n\n<p>This plugin intercepts official WordPress APIs by identifying the calling plugin via the PHP call stack. It cannot intercept calls made by WordPress core on behalf of a plugin \u2014 for example, when WordPress processes a settings form via <code>options.php<\/code>, the call stack contains core files rather than the plugin files.<\/p>\n\n<p>Specific limitations:<\/p>\n\n<ul>\n<li><code>options:read<\/code> \/ <code>options:write<\/code> \u2014 work when a plugin calls these APIs directly from its own code (hooks, AJAX, cron). Do not block standard WordPress settings forms processed by <code>options.php<\/code>.<\/li>\n<li>Filesystem access (<code>file_get_contents<\/code>, <code>fopen<\/code>, etc.) is not intercepted.<\/li>\n<li>Direct <code>mysqli<\/code> connections, <code>eval()<\/code>, and raw PHP file functions bypass all interceptors.<\/li>\n<li><code>wp_update_user()<\/code> and <code>wp_delete_user()<\/code> are covered by <code>db:write:users<\/code> since they write directly to the users table.<\/li>\n<li><code>admin_init<\/code> is intentionally excluded from <code>hooks:admin<\/code> as it is too critical to block safely.<\/li>\n<\/ul>\n\n<!--section=installation-->\n<ol>\n<li>Upload the <code>plugseal<\/code> folder to <code>\/wp-content\/plugins\/<\/code>.<\/li>\n<li>Activate the plugin through the <strong>Plugins<\/strong> menu.<\/li>\n<li>Go to <strong>Settings \u2192 PlugSeal<\/strong>.<\/li>\n<li>Select a plugin and toggle individual permissions on or off.<\/li>\n<\/ol>\n\n<!--section=faq-->\n<dl>\n<dt id=\"does%20this%20work%20with%20multisite%3F\"><h3>Does this work with Multisite?<\/h3><\/dt>\n<dd><p>No. Multisite is not supported in this version.<\/p><\/dd>\n<dt id=\"what%20happens%20to%20my%20data%20if%20i%20uninstall%20the%20plugin%3F\"><h3>What happens to my data if I uninstall the plugin?<\/h3><\/dt>\n<dd><p>Data is preserved by default. To delete all data on uninstall, enable the option in the settings page before deleting the plugin.<\/p><\/dd>\n<dt id=\"can%20a%20plugin%20bypass%20this%20system%3F\"><h3>Can a plugin bypass this system?<\/h3><\/dt>\n<dd><p>Yes, if a plugin makes direct database connections or filesystem calls without using WordPress APIs, or if WordPress core processes actions on its behalf. These are known limitations documented above.<\/p><\/dd>\n\n<\/dl>\n\n<!--section=changelog-->\n<h4>0.1.0 - 2025-04-25<\/h4>\n\n<ul>\n<li>Initial release.<\/li>\n<\/ul>","raw_excerpt":"Control what each active plugin is allowed to do. Allow or deny specific permissions per plugin, with immediate effect.","jetpack_sharing_enabled":true,"_links":{"self":[{"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/plugin\/305955","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/plugin"}],"about":[{"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/types\/plugin"}],"replies":[{"embeddable":true,"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/comments?post=305955"}],"author":[{"embeddable":true,"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wporg\/v1\/users\/marc4"}],"wp:attachment":[{"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/media?parent=305955"}],"wp:term":[{"taxonomy":"plugin_section","embeddable":true,"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/plugin_section?post=305955"},{"taxonomy":"plugin_tags","embeddable":true,"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/plugin_tags?post=305955"},{"taxonomy":"plugin_category","embeddable":true,"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/plugin_category?post=305955"},{"taxonomy":"plugin_contributors","embeddable":true,"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/plugin_contributors?post=305955"},{"taxonomy":"plugin_business_model","embeddable":true,"href":"https:\/\/tir.wordpress.org\/plugins\/wp-json\/wp\/v2\/plugin_business_model?post=305955"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}