<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: puppet</title>
    <description>The latest articles tagged 'puppet' on DEV Community.</description>
    <link>https://dev.to/t/puppet</link>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/tag/puppet"/>
    <language>en</language>
    <item>
      <title>RHEL 10 support now available in Puppet SCE for Linux</title>
      <dc:creator>Jason St-Cyr</dc:creator>
      <pubDate>Thu, 18 Jun 2026 16:16:41 +0000</pubDate>
      <link>https://dev.to/puppet/rhel-10-support-now-available-in-puppet-sce-for-linux-4opi</link>
      <guid>https://dev.to/puppet/rhel-10-support-now-available-in-puppet-sce-for-linux-4opi</guid>
      <description>&lt;p&gt;Version 2.7.0 of Security Compliance Enforcement (SCE) for Linux is now &lt;a href="https://forge.puppet.com/modules/puppetlabs/sce_linux/readme" rel="noopener noreferrer"&gt;available for download from the Forge&lt;/a&gt;!&lt;/p&gt;

&lt;h2&gt;
  
  
  Support for the RHEL 10 family
&lt;/h2&gt;

&lt;p&gt;This release adds Red Hat Enterprise Linux (RHEL) 10 CIS benchmarks (v1.0.1, Server Levels 1 and 2). Teams adopting RHEL 10 or a compatible platform can bring those systems into compliance using the same trusted standards already in place across earlier RHEL versions.&lt;/p&gt;

&lt;h2&gt;
  
  
  Other improvements
&lt;/h2&gt;

&lt;p&gt;Some other issues were also addressed, including logging issues with the rsyslog configuration file and intrusion detection on RHEL 9.&lt;/p&gt;

&lt;p&gt;For all the details, make sure to read the &lt;a href="https://help.puppet.com/sce/current/linux/scel_relnotes_270.htm" rel="noopener noreferrer"&gt;full release notes&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;&lt;a href="https://forge.puppet.com/modules/puppetlabs/sce_linux/readme" class="crayons-btn crayons-btn--primary" rel="noopener noreferrer"&gt;SCE Module on Puppet Forge&lt;/a&gt;
&lt;/p&gt;

</description>
      <category>puppet</category>
      <category>security</category>
      <category>devops</category>
    </item>
    <item>
      <title>Security Compliance Management 3.8.0 Is Now Available</title>
      <dc:creator>Jason St-Cyr</dc:creator>
      <pubDate>Tue, 16 Jun 2026 13:31:38 +0000</pubDate>
      <link>https://dev.to/puppet/security-compliance-management-380-is-now-available-1o26</link>
      <guid>https://dev.to/puppet/security-compliance-management-380-is-now-available-1o26</guid>
      <description>&lt;p&gt;Security Compliance Management (SCM) 3.8.0 is here, with updates focused on keeping compliance scans running reliably with less manual intervention and &lt;strong&gt;an important license update&lt;/strong&gt;.&lt;/p&gt;

&lt;p&gt;This release introduces automatic cleanup for stuck scans, improved control over scan behavior with configurable timeouts, extended CIS-CAT® Pro Assessor license support, and updated benchmark content. It also includes important security fixes across core components.&lt;/p&gt;

&lt;p&gt;⚠️ We recommend upgrading to SCM 3.8.0 before &lt;strong&gt;June 21, 2026&lt;/strong&gt; to avoid disruption, as the CIS-CAT Pro Assessor license included in SCM 3.7.1 expires on that date.&lt;/p&gt;




&lt;h2&gt;
  
  
  What’s changing in SCM 3.8.0
&lt;/h2&gt;

&lt;h3&gt;
  
  
  CIS-CAT Pro Assessor licensing and version
&lt;/h3&gt;

&lt;ul&gt;
&lt;li&gt;SCM 3.8.0 now contains CIS-CAT Pro Assessor v4.63.0&lt;/li&gt;
&lt;li&gt;The bundled &lt;strong&gt;CIS-CAT® Pro Assessor license&lt;/strong&gt; is now valid for &lt;strong&gt;one year&lt;/strong&gt;

&lt;ul&gt;
&lt;li&gt;The license shipped with SCM 3.8.0 is valid until &lt;strong&gt;June 2027&lt;/strong&gt;
&lt;/li&gt;
&lt;li&gt;The license included in &lt;strong&gt;SCM 3.7.1 expires on June 21, 2026&lt;/strong&gt;
&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;You can now also update the license without upgrading SCM:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;New &lt;code&gt;license_path&lt;/code&gt; parameter

&lt;ul&gt;
&lt;li&gt;Allows updating the CIS-CAT Pro Assessor license independently
&lt;/li&gt;
&lt;li&gt;Documentation: &lt;a href="https://help.puppet.com/scm/current/Content/UserGuide/SCM/update_assessor_license.htm" rel="noopener noreferrer"&gt;https://help.puppet.com/scm/current/Content/UserGuide/SCM/update_assessor_license.htm&lt;/a&gt;
&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  Scan management, configuration, and reliability
&lt;/h3&gt;

&lt;ul&gt;
&lt;li&gt;&lt;p&gt;Added a &lt;strong&gt;background scan sweeper&lt;/strong&gt; to detect and cancel scans stuck in a "running" state&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;Fixed a race condition where timed-out Puppet Enterprise job status polls could leave scans permanently stuck&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;New &lt;code&gt;assessor_scan_timeout&lt;/code&gt; option controls task timeout for &lt;strong&gt;Windows Server 2022 domain controllers&lt;/strong&gt; (Note: this isn't set by default)&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;Increased default &lt;strong&gt;Max GraphQL requests limit&lt;/strong&gt; to &lt;strong&gt;300 requests&lt;/strong&gt;&lt;/p&gt;&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  Benchmark coverage updates
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;New benchmarks added for Amazon Linux 2023 STIG, Microsoft Windows 11 STIG, Oracle Linux 9 STIG, RHEL 10 STIG, and SUSE 16&lt;/li&gt;
&lt;li&gt;Updated benchmarks for: Amazon Linux 2, macOS, Debian, Windows, and Ubuntu (see the release notes for specific benchmark updates)&lt;/li&gt;
&lt;li&gt;Removed benchmarks for:

&lt;ul&gt;
&lt;li&gt;Azure Compute Windows Server 2019 v1.0.1
&lt;/li&gt;
&lt;li&gt;Azure Compute Windows Server 2022 v1.0.0
&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  Security fixes
&lt;/h2&gt;

&lt;p&gt;This release includes updates to address 40 vulnerabilities across several components. The following components were updated to address the vulnerabilities:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Gorm.io&lt;/li&gt;
&lt;li&gt;Keycloak&lt;/li&gt;
&lt;li&gt;netty-codec&lt;/li&gt;
&lt;li&gt;netty-codec-http&lt;/li&gt;
&lt;li&gt;netty-codec-http2&lt;/li&gt;
&lt;li&gt;netty-codec-haproxy&lt;/li&gt;
&lt;li&gt;netty-handler&lt;/li&gt;
&lt;li&gt;Protobuf&lt;/li&gt;
&lt;li&gt;react-router&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Refer to the full release notes for the complete list of CVEs.&lt;/p&gt;




&lt;h2&gt;
  
  
  Upgrade guidance
&lt;/h2&gt;

&lt;p&gt;To avoid scan interruptions, upgrade to &lt;strong&gt;SCM 3.8.0 before June 21, 2026&lt;/strong&gt;. This ensures continued use of the CIS-CAT Pro Assessor, access to updated benchmark content, improved security posture, and improvements in scan processing.&lt;/p&gt;




&lt;h2&gt;
  
  
  Learn more
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;Full release notes:
&lt;a href="https://help.puppet.com/scm/current/Content/UserGuide/SCM/Release_notes/release_notes.htm#SecurityComplianceManagement380" rel="noopener noreferrer"&gt;https://help.puppet.com/scm/current/Content/UserGuide/SCM/Release_notes/release_notes.htm#SecurityComplianceManagement380&lt;/a&gt;
&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;If you have questions or need assistance upgrading, reach out to Puppet Support.&lt;/p&gt;

&lt;h2&gt;
  
  
  🤖 AI Disclosure
&lt;/h2&gt;

&lt;p&gt;This article was written and reviewed by the author, with the help of AI to assist in pulling together the details from multiple sources and general brand voice alignment.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;How did I do that?&lt;/strong&gt; For this particular article, I provided Microsoft 365 Copilot with the original release notes, my previous release announcement for 3.7.0, our company brand voice guidelines, and the official product release announcement that went out to customers. The LLM can then pull together the list of things that were updated and create a skeleton of an article. I then rewrite the content as needed to meet with my own tone of voice and get rid of the over-list-based approach that LLMs often take. It's also important to actually check back against the original release notes because sometimes the LLM will change certain words or remove words that change the meaning of what was in the release. I hope this helps if you are also writing with LLMs!&lt;/p&gt;

</description>
      <category>puppet</category>
      <category>security</category>
      <category>devops</category>
      <category>infrastructureascode</category>
    </item>
    <item>
      <title>Selenium vs Puppeteer vs Playwright: Complete Guide for Web Scraping, Testing &amp; AI Automation</title>
      <dc:creator>IPFoxy</dc:creator>
      <pubDate>Tue, 16 Jun 2026 12:23:25 +0000</pubDate>
      <link>https://dev.to/ipfoxy/selenium-vs-puppeteer-vs-playwright-complete-guide-for-web-scraping-testing-ai-automation-2619</link>
      <guid>https://dev.to/ipfoxy/selenium-vs-puppeteer-vs-playwright-complete-guide-for-web-scraping-testing-ai-automation-2619</guid>
      <description>&lt;p&gt;In 2026, browser automation has evolved far beyond traditional QA testing and web scraping. With the rise of LLM-powered AI Agents, automation frameworks have become the foundation for autonomous web browsing, task execution, and large-scale AI workflows.&lt;/p&gt;

&lt;p&gt;This guide compares Selenium, Puppeteer, and Playwright across architecture, performance, scalability, AI automation compatibility, and proxy integration.&lt;/p&gt;

&lt;p&gt;**&lt;/p&gt;

&lt;h2&gt;
  
  
  I. Introduction to the Three Automation Frameworks
&lt;/h2&gt;

&lt;p&gt;**&lt;br&gt;
&lt;strong&gt;1. Selenium: The Most Established Browser Automation Framework&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Selenium was introduced in 2004 and remains the industry standard for browser automation. With Selenium 5.0, the framework has undergone a major architectural upgrade.&lt;/p&gt;

&lt;p&gt;Key Features:&lt;br&gt;
• Supports Chrome, Firefox, Edge, Safari, and other major browsers.&lt;br&gt;
• Supports Java, Python, C#, JavaScript, and more.&lt;br&gt;
• Large ecosystem and strong enterprise adoption.&lt;/p&gt;

&lt;p&gt;Best For:&lt;br&gt;
Automated testing, enterprise workflows, and projects requiring broad browser compatibility.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;2. Playwright&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Playwright was open-sourced by Microsoft in 2020 and has become the fastest-growing browser automation framework.&lt;/p&gt;

&lt;p&gt;Key Features:&lt;br&gt;
• Native support for Chromium, Firefox, and WebKit.&lt;br&gt;
• Built-in Auto-Wait mechanism.&lt;br&gt;
• Browser Context support for large-scale concurrency.&lt;/p&gt;

&lt;p&gt;Best For:&lt;br&gt;
Web scraping, AI Agent automation, eCommerce data collection, social media automation, and large-scale browser operations.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;3. Puppeteer: Google's Native Chrome Automation Tool&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Released by the Chrome team in 2017, Puppeteer popularized the Chrome DevTools Protocol (CDP).&lt;/p&gt;

&lt;p&gt;Key Features:&lt;br&gt;
• Simple and intuitive API.&lt;br&gt;
• Fast execution speed.&lt;br&gt;
• Popular in web scraping and browser automation.&lt;/p&gt;

&lt;p&gt;Best For:&lt;br&gt;
Chrome/Chromium automation, web scraping, screenshots, and PDF generation.&lt;/p&gt;
&lt;h2&gt;
  
  
  II. Real-World Performance Comparison
&lt;/h2&gt;

&lt;p&gt;&lt;strong&gt;1. Startup Speed&lt;/strong&gt;&lt;br&gt;
In raw tests launching a headless browser, Puppeteer and Playwright achieve cold startup speeds of 100-300 milliseconds because they connect directly to the underlying binary protocols via WebSockets. Selenium, however, typically takes 1-2 seconds due to the need to initialize the corresponding Driver process and handle layered HTTP handshakes. &lt;/p&gt;

&lt;p&gt;&lt;strong&gt;2. Concurrency&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Selenium launches a new browser process for each isolated environment, resulting in significant memory usage.&lt;/p&gt;

&lt;p&gt;Playwright uses BrowserContext, allowing dozens of fully isolated environments to run within a single browser process while reducing resource consumption by over 70%.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;3. Stability&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;Modern React and Vue applications render content asynchronously. Selenium often requires manual waits, while Playwright automatically performs Actionability Checks before interacting with elements, improving reliability significantly.&lt;br&gt;
**&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;Compatibility**
If your enterprise requires testing specific rendering issues on Safari within a physical Mac environment, or if your development team relies strictly on Java/C#, Selenium’s multi-language support and deep-rooted history make it irreplaceable. &lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;&lt;strong&gt;5. AI Agent Compatibility&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;In 2026, AI Agent + Browser Automation has become one of the hottest directions. Projects like Claude Code, OpenAI Agent, and OpenManus all prioritize Playwright by default for three key reasons: &lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Native MCP Protocol Support: Playwright can output the page’s Accessibility Tree directly to the LLM, eliminating the need to parse raw DOM text. &lt;/li&gt;
&lt;li&gt;Highly Efficient Token Usage: Compared to traditional DOM serialization, Playwright's optimized approach reduces token consumption for long sessions by 3 to 4 times. &lt;/li&gt;
&lt;li&gt;Execution Stability: Playwright’s auto-wait and auto-retry mechanisms drastically lower the failure rates of complex agent actions. &lt;/li&gt;
&lt;/ul&gt;
&lt;h2&gt;
  
  
  III. Proxy Integration and Anti-Detection Capabilities
&lt;/h2&gt;

&lt;p&gt;For modern web scraping, accessing a page is only the beginning. Websites commonly evaluate:&lt;/p&gt;

&lt;p&gt;• IP Reputation&lt;br&gt;
• Request Frequency&lt;br&gt;
• Browser Fingerprints&lt;br&gt;
• Geographic Consistency&lt;/p&gt;

&lt;p&gt;Residential Proxies provide a more authentic browsing environment than datacenter IPs and help reduce blocking risks.&lt;br&gt;
IPFoxy offers residential proxies covering 200+ countries, static ISP proxies, mobile 4G/5G proxies, and supports both HTTP(S) and SOCKS5 protocols. It integrates seamlessly with Selenium, Puppeteer, and Playwright to significantly lower blocking risks for long-running scraper projects. &lt;/p&gt;

&lt;p&gt;Here is how to configure an IPFoxy proxy across the three frameworks:&lt;br&gt;
Selenium Proxy Configuration (Python)&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight python"&gt;&lt;code&gt;&lt;span class="kn"&gt;from&lt;/span&gt; &lt;span class="n"&gt;selenium&lt;/span&gt; &lt;span class="kn"&gt;import&lt;/span&gt; &lt;span class="n"&gt;webdriver&lt;/span&gt;
&lt;span class="kn"&gt;from&lt;/span&gt; &lt;span class="n"&gt;selenium.webdriver.chrome.options&lt;/span&gt; &lt;span class="kn"&gt;import&lt;/span&gt; &lt;span class="n"&gt;Options&lt;/span&gt;

&lt;span class="n"&gt;chrome_options&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nc"&gt;Options&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;
&lt;span class="c1"&gt;# Configure IPFoxy proxy address (with user/pass authentication)
# Static ISP or rotating residential proxies are recommended for the lowest fraud scores
&lt;/span&gt;&lt;span class="n"&gt;proxy_server&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;http://username:password@proxy.ipfoxy.com:port&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;
&lt;span class="n"&gt;chrome_options&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;add_argument&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="sa"&gt;f&lt;/span&gt;&lt;span class="sh"&gt;'&lt;/span&gt;&lt;span class="s"&gt;--proxy-server=&lt;/span&gt;&lt;span class="si"&gt;{&lt;/span&gt;&lt;span class="n"&gt;proxy_server&lt;/span&gt;&lt;span class="si"&gt;}&lt;/span&gt;&lt;span class="sh"&gt;'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;

&lt;span class="c1"&gt;# Launch driver
&lt;/span&gt;&lt;span class="n"&gt;driver&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="n"&gt;webdriver&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nc"&gt;Chrome&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;options&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="n"&gt;chrome_options&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="n"&gt;driver&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;get&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;https://ipinfo.io&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;  &lt;span class="c1"&gt;# Verify the IP has switched to a clean residential IP from IPFoxy
&lt;/span&gt;&lt;span class="nf"&gt;print&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="n"&gt;driver&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="n"&gt;title&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;span class="n"&gt;driver&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;quit&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Puppeteer Proxy Configuration (JavaScript)&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight javascript"&gt;&lt;code&gt;&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;puppeteer&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;require&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;puppeteer&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;proxy&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;http://user:pass@ipfoxy-proxy:port&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;;&lt;/span&gt;

&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;browser&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;puppeteer&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;launch&lt;/span&gt;&lt;span class="p"&gt;({&lt;/span&gt;
  &lt;span class="na"&gt;args&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="s2"&gt;`--proxy-server=&lt;/span&gt;&lt;span class="p"&gt;${&lt;/span&gt;&lt;span class="nx"&gt;proxy&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="s2"&gt;`&lt;/span&gt;&lt;span class="p"&gt;]&lt;/span&gt;
&lt;span class="p"&gt;});&lt;/span&gt;
&lt;span class="kd"&gt;const&lt;/span&gt; &lt;span class="nx"&gt;page&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;browser&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;newPage&lt;/span&gt;&lt;span class="p"&gt;();&lt;/span&gt;
&lt;span class="k"&gt;await&lt;/span&gt; &lt;span class="nx"&gt;page&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;goto&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="s1"&gt;https://example.com&lt;/span&gt;&lt;span class="dl"&gt;'&lt;/span&gt;&lt;span class="p"&gt;);&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Playwright Proxy Configuration (Python)&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight python"&gt;&lt;code&gt;&lt;span class="kn"&gt;from&lt;/span&gt; &lt;span class="n"&gt;playwright.sync_api&lt;/span&gt; &lt;span class="kn"&gt;import&lt;/span&gt; &lt;span class="n"&gt;sync_playwright&lt;/span&gt;

&lt;span class="k"&gt;with&lt;/span&gt; &lt;span class="nf"&gt;sync_playwright&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt; &lt;span class="k"&gt;as&lt;/span&gt; &lt;span class="n"&gt;p&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;
    &lt;span class="n"&gt;browser&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="n"&gt;p&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="n"&gt;chromium&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;launch&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;
        &lt;span class="n"&gt;proxy&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;
            &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;server&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;http://ipfoxy-proxy:port&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
            &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;username&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;user&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
            &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;password&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt; &lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;pass&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;
        &lt;span class="p"&gt;}&lt;/span&gt;
    &lt;span class="p"&gt;)&lt;/span&gt;
    &lt;span class="n"&gt;page&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="n"&gt;browser&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;new_page&lt;/span&gt;&lt;span class="p"&gt;()&lt;/span&gt;
&lt;span class="n"&gt;page&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;goto&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="s"&gt;https://example.com&lt;/span&gt;&lt;span class="sh"&gt;"&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  IV. Which Framework Should You Choose in 2026?
&lt;/h2&gt;

&lt;p&gt;Your technical choice should align directly with your business pain points and engineering stack: &lt;br&gt;
&lt;strong&gt;Choose Playwright if:&lt;/strong&gt;&lt;br&gt;
• You are building a new web scraping, data collection, or AI Agent project.&lt;br&gt;
• You need high concurrency and efficient resource usage.&lt;br&gt;
• You work primarily with Python or TypeScript.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Choose Puppeteer if:&lt;/strong&gt;&lt;br&gt;
• You are a Node.js-focused developer.&lt;br&gt;
• Your project only targets Chrome/Chromium.&lt;br&gt;
• You require direct access to advanced CDP capabilities.&lt;/p&gt;

&lt;p&gt;&lt;strong&gt;Choose Selenium if:&lt;/strong&gt;&lt;br&gt;
• Your organization relies heavily on Java or C#.&lt;br&gt;
• You already have large-scale Selenium Grid infrastructure.&lt;br&gt;
• Cross-browser testing is a top priority.&lt;/p&gt;

&lt;h2&gt;
  
  
  V. Conclusion
&lt;/h2&gt;

&lt;p&gt;Selenium, Puppeteer, and Playwright are all excellent browser automation frameworks, but they serve different needs.&lt;/p&gt;

&lt;p&gt;For web scraping, AI Agents, multi-account automation, and large-scale data collection, Playwright is often the most future-proof choice. Combined with high-quality residential proxies, it can significantly improve scraping success rates and long-term stability.&lt;/p&gt;

</description>
      <category>webdev</category>
      <category>selenium</category>
      <category>puppet</category>
      <category>playwright</category>
    </item>
    <item>
      <title>Remediating 18 OpenSSL CVEs at Scale with Puppet</title>
      <dc:creator>Jason St-Cyr</dc:creator>
      <pubDate>Mon, 15 Jun 2026 14:10:21 +0000</pubDate>
      <link>https://dev.to/puppet/remediating-18-openssl-cves-at-scale-with-puppet-1abo</link>
      <guid>https://dev.to/puppet/remediating-18-openssl-cves-at-scale-with-puppet-1abo</guid>
      <description>&lt;blockquote&gt;
&lt;p&gt;&lt;strong&gt;&lt;em&gt;Written by &lt;a href="https://www.puppet.com/author/paul-reed" rel="noopener noreferrer"&gt;Paul Reed&lt;/a&gt;.&lt;/em&gt;&lt;/strong&gt;&lt;/p&gt;
&lt;/blockquote&gt;

&lt;p&gt;The June 2026 OpenSSL advisory is a big one. &lt;a href="https://www.puppet.com/blog/openssl-cve-2026-45447-patching" rel="noopener noreferrer"&gt;18 vulnerabilities, one rated high severity&lt;/a&gt; with remote code execution potential, and a disclosure credited in part to &lt;a href="https://red.anthropic.com/2026/mythos-preview/" rel="noopener noreferrer"&gt;Anthropic's Mythos model&lt;/a&gt; working alongside researcher Alex Gaynor. Six of those CVEs trace back to that collaboration.&lt;/p&gt;

&lt;p&gt;When an advisory like the OpenSSL one lands, the first question is always the same: where are we exposed? If you run Puppet, you can answer that question across the entire fleet right now, patch it through one mechanism that handles every platform for you, and have it stay patched without anyone watching afterwards. The rest of this article is how.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Vulnerability: What CVE-2026-45447 Actually Does
&lt;/h2&gt;

&lt;p&gt;CVE-2026-45447 is a heap use-after-free in &lt;code&gt;PKCS7_verify()&lt;/code&gt;. The bug fires when OpenSSL processes a PKCS#7 or S/MIME signed message where the &lt;code&gt;SignedData.digestAlgorithms&lt;/code&gt; field is an empty ASN.1 SET.&lt;/p&gt;

&lt;p&gt;When OpenSSL encounters this condition, OpenSSL frees a &lt;code&gt;BIO&lt;/code&gt; object that was passed in by the calling application and is still expected to be valid. The calling application then uses the freed pointer. Depending on heap layout, that results in heap corruption, a process crash, or with a controlled heap grooming primitive, code execution.&lt;/p&gt;

&lt;p&gt;Affected ranges:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;  OpenSSL 3.0.x through 3.3.x (patch to 3.5.1)&lt;/li&gt;
&lt;li&gt;  OpenSSL 1.1.1x (patch to corresponding 1.1.1 update)&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;The other 17 CVEs in the advisory cover authentication bypass via forged certificates (moderate, roughly a 1-in-256 success rate), ciphertext forgery, private key recovery, root CA replacement, and several DoS vectors. None are trivial in regulated environments.&lt;/p&gt;

&lt;h2&gt;
  
  
  Step 1: Query Your Actual Exposure
&lt;/h2&gt;

&lt;p&gt;Knowing where you're exposed is where Puppet earns its keep on day zero. There's no scanner to stand up and no spreadsheet to chase round the teams. The data is already sitting in PuppetDB.&lt;/p&gt;

&lt;p&gt;Package inventory is fed by Puppet's resource abstraction layer, the same machinery behind &lt;code&gt;puppet resource package&lt;/code&gt;. It enumerates every package provider Puppet Enterprise supports. This means package inventory sees well beyond the system package manager: OS packages across apt, dnf/yum and zypper, and language managers like gem and pip alongside them. One query, every node:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;puppet query &lt;span class="s1"&gt;'package_inventory[certname, package_name, version, provider] {
  package_name ~ "(?i)openssl$|libssl$|libcrypto$"
  and
  version ~ "^(3\\.[0-3]\\.|1\\.[0-1]\\.)"
}'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Run the query against a live environment to review the results. On one fleet:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;  system libraries across multiple providers (&lt;code&gt;libopenssl3&lt;/code&gt;/&lt;code&gt;libopenssl1_1&lt;/code&gt; via zypper, &lt;code&gt;openssl&lt;/code&gt; and &lt;code&gt;openssl-libs&lt;/code&gt; via dnf/yum, &lt;code&gt;openssl&lt;/code&gt; via apt)&lt;/li&gt;
&lt;li&gt;  the Ruby &lt;code&gt;openssl&lt;/code&gt; gem at several versions&lt;/li&gt;
&lt;li&gt;  &lt;code&gt;openssl&lt;/code&gt; via &lt;code&gt;puppet_gem&lt;/code&gt; on every agent node, because Puppet's own Ruby ships it&lt;/li&gt;
&lt;li&gt;  &lt;code&gt;pyOpenSSL&lt;/code&gt; and &lt;code&gt;python3-openssl&lt;/code&gt; via pip and the OS package manager&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Most tools miss the bulk of that, because they only ever look at the system package manager. Here, anything a package manager put on the box is in scope, system libraries and language bindings together.&lt;/p&gt;

&lt;p&gt;Scope the query to an environment by filtering against the inventory endpoint:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;puppet query &lt;span class="s1"&gt;'package_inventory[certname, package_name, version, provider] {
  package_name ~ "(?i)openssl$|libssl$|libcrypto$"
  and
  version ~ "^(3\\.[0-3]\\.|1\\.[0-1]\\.)"
  and
  certname in inventory[certname] { environment = "production" }
}'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Step 2: Patch It (recommended for the actual remediation)
&lt;/h2&gt;

&lt;p&gt;You don't hand-write a &lt;code&gt;package&lt;/code&gt; resource per platform, and you shouldn't. The package name varies (&lt;code&gt;openssl-libs&lt;/code&gt; on RHEL, &lt;code&gt;libssl3&lt;/code&gt; on Debian, &lt;code&gt;libopenssl3&lt;/code&gt; on SUSE), the versions differ again, and the &lt;code&gt;openssl&lt;/code&gt; CLI isn't even the vulnerable piece, the runtime library is. Let the tooling that already models your estate carry that.&lt;/p&gt;

&lt;p&gt;Use the patching framework: &lt;code&gt;pe_patch&lt;/code&gt; on Puppet Enterprise, &lt;code&gt;os_patching&lt;/code&gt; for open source and Bolt. Classify the class and each node reports its pending updates, including which are security updates. You patch through the PE console or a task, scoped to security updates only if you want the change tight.&lt;/p&gt;

&lt;p&gt;The OS package manager applies the vendor's security update, so the correct library package and version are chosen per platform without you encoding any of it. Reboots, update ordering, and patch and blackout windows are the framework's job. A box that can't be touched in business hours is a blackout window in config, not a workaround.&lt;/p&gt;

&lt;p&gt;The exposure query doubles as your target list. The orchestrator takes PQL directly with &lt;code&gt;-q&lt;/code&gt;, so there's no glue script to write. You hand the orchestrator the query and let it resolve the nodes:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;puppet task run pe_patch::patch_server &lt;span class="nt"&gt;-q&lt;/span&gt; &lt;span class="s1"&gt;'package_inventory[certname]{
  package_name ~ "(?i)openssl$|libssl$|libcrypto$"
  and version ~ "^(3\\.[0-3]\\.|1\\.[0-1]\\.)"
}'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;That's &lt;code&gt;pe_patch::patch_server&lt;/code&gt; on Puppet Enterprise, &lt;code&gt;os_patching::patch_server&lt;/code&gt; for open source and Bolt. The orchestration runs the task against exactly the nodes the query returned and nothing else. Add &lt;code&gt;security_only=true&lt;/code&gt; to keep the run tight, so a node that's already current is a no-op. If you'd rather not touch the command line, wire the same query into a node group in the PE console and drive it from there.&lt;/p&gt;

&lt;h2&gt;
  
  
  Step 3: Enforce It In Code
&lt;/h2&gt;

&lt;p&gt;If you'd rather declare the state and have Puppet hold the state on every run, use the &lt;code&gt;openssl&lt;/code&gt; module's class. The module knows the package names per platform:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight puppet"&gt;&lt;code&gt;&lt;span class="k"&gt;class&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="s1"&gt;'openssl'&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;
  &lt;span class="py"&gt;package_ensure&lt;/span&gt;         &lt;span class="p"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="n"&gt;latest&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="py"&gt;ca_certificates_ensure&lt;/span&gt; &lt;span class="p"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="n"&gt;latest&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Using &lt;code&gt;latest&lt;/code&gt; keeps the library current; set &lt;code&gt;package_ensure&lt;/code&gt; to a specific version from Hiera if you want a pinned, auditable rollout.&lt;/p&gt;

&lt;p&gt;This enforcement step is something a one-off script and a scanner both miss. A patch run fixes the issue once; declaring the state is what makes it stay fixed. The agent runs on a schedule, every 30 minutes by default, and enforces desired state each time. So when a node drifts back to a vulnerable version, a VM reprovisioned off a stale image, say, or a change someone made by hand and forgot, the next run quietly puts the desired state back. A scanner would notice that regression on its next sweep and open you a ticket. Enforcement just doesn't let the gap stay open that long.&lt;/p&gt;

&lt;p&gt;The two paths aren't either/or. The patching framework is the quicker way to clear the initial backlog; enforcement is what keeps the backlog cleared. Run both and the framework does the first sweep while the module holds the line from then on.&lt;/p&gt;

&lt;h2&gt;
  
  
  Step 4: Restart Processes That Link the Library
&lt;/h2&gt;

&lt;p&gt;Patching the package is only half the job. A running process keeps the old &lt;code&gt;.so&lt;/code&gt; mapped until the process restarts, and a library swap under a live OpenSSL is exactly the case where that bites.&lt;/p&gt;

&lt;p&gt;If you patched with the framework, it already tracks this for you. &lt;code&gt;pe_patch&lt;/code&gt; reports the processes that require a restart in the node's fact, so you don't go hunting:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight json"&gt;&lt;code&gt;&lt;span class="nl"&gt;"reboots"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"app_restart_required"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"apps_needing_restart"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"1"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/usr/lib/systemd/systemd --switched-root --system --deserialize 31"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"586"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/usr/lib/systemd/systemd-journald"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"601"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/usr/lib/systemd/systemd-udevd"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"657"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/sbin/auditd"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"691"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/usr/bin/python3 -s /usr/sbin/firewalld --nofork --nopid"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"694"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/usr/sbin/rngd -f --fill-watermark=0 -x pkcs11 -x nist -x qrypt -x namedpipe -x jitter -D daemon:daemon"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"696"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/usr/lib/systemd/systemd-logind"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"766"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"/usr/sbin/NetworkManager --no-daemon"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
        &lt;/span&gt;&lt;span class="nl"&gt;"863"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"sshd: /usr/sbin/sshd -D [listener] 0 of 10-60 startups"&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="p"&gt;},&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"reboot_required"&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="err"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;&lt;code&gt;apps_needing_restart&lt;/code&gt; is the list of processes still mapping a file that's been replaced underneath them, and &lt;code&gt;reboot_required&lt;/code&gt; flags when a restart of individual services won't cut it. The patch run acts on these according to its reboot policy, so the same job that applies the update also clears the stale library, or tells you precisely which nodes still need a bounce. That's the manual &lt;code&gt;lsof&lt;/code&gt; check in the next section, done for the whole fleet as a fact.&lt;/p&gt;

&lt;p&gt;If you went the module route instead, chain the dependent services onto the class so they refresh when it changes the library:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight puppet"&gt;&lt;code&gt;&lt;span class="nc"&gt;Class&lt;/span&gt;&lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="s1"&gt;'openssl'&lt;/span&gt;&lt;span class="p"&gt;]&lt;/span&gt; &lt;span class="p"&gt;~&amp;gt;&lt;/span&gt; &lt;span class="nc"&gt;Service&lt;/span&gt;&lt;span class="p"&gt;[&lt;/span&gt;&lt;span class="s1"&gt;'nginx'&lt;/span&gt;&lt;span class="p"&gt;]&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Chaining on the class rather than a &lt;code&gt;Package['...']&lt;/code&gt; title means you don't have to know the package name the module picks per platform. The refresh fires only when something actually changes, so steady-state runs leave your services alone.&lt;/p&gt;

&lt;p&gt;A note on the language-level copies inventory turned up. Where a gem or &lt;code&gt;pyOpenSSL&lt;/code&gt; links the system library, patching &lt;code&gt;libssl&lt;/code&gt;/&lt;code&gt;libcrypto&lt;/code&gt; fixes the underlying crypto and these processes will pick up the patch on restart. The few that statically bundle their own copy get updated through their own toolchain (&lt;code&gt;gem update&lt;/code&gt;, &lt;code&gt;pip&lt;/code&gt;). The OS-packaged bindings ride along with the patch run either way.&lt;/p&gt;

&lt;h2&gt;
  
  
  Step 5: Verify the Process Has Reloaded the Library
&lt;/h2&gt;

&lt;p&gt;The &lt;code&gt;reboots&lt;/code&gt; fact above is your fleet-wide answer. When you want to confirm a single box by hand, or you're working somewhere the fact isn't available, &lt;code&gt;lsof&lt;/code&gt; against the actual PID tells you what that process has mapped:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;&lt;span class="nb"&gt;sudo &lt;/span&gt;lsof &lt;span class="nt"&gt;-p&lt;/span&gt; &lt;span class="si"&gt;$(&lt;/span&gt;pgrep nginx | &lt;span class="nb"&gt;head&lt;/span&gt; &lt;span class="nt"&gt;-1&lt;/span&gt;&lt;span class="si"&gt;)&lt;/span&gt; | &lt;span class="nb"&gt;grep &lt;/span&gt;libssl
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;If the &lt;code&gt;lsof&lt;/code&gt; query still shows the old path after the run, the restart didn't fire. Check the agent log or the patch run's reboot status. The &lt;code&gt;openssl&lt;/code&gt; CLI version won't help here, the version says nothing about what a long-running daemon has mapped.&lt;/p&gt;

&lt;h2&gt;
  
  
  Step 6: Confirm Convergence
&lt;/h2&gt;

&lt;p&gt;Re-run the inventory query to verify closure. This query doubles as audit evidence:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;puppet query &lt;span class="s1"&gt;'package_inventory[certname, package_name, version, provider] {
  package_name ~ "(?i)openssl$|libssl$|libcrypto$"
}'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Anything still on an affected version, across any provider, is your remaining work. Puppet Enterprise users can pull the same data from the compliance dashboard for audit review.&lt;/p&gt;

&lt;h2&gt;
  
  
  The Point Isn't OpenSSL
&lt;/h2&gt;

&lt;p&gt;OpenSSL is this month's fire drill. Next month it's something else, and the one after that hasn't been disclosed yet. None of the steps above were really about OpenSSL. They were about having a tool already in place that answers the questions every advisory asks, before the advisory lands.&lt;/p&gt;

&lt;p&gt;Look back at what each step actually was:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;The exposure query was inventory: what have I got, and where, as of the last check-in. &lt;/li&gt;
&lt;li&gt;The patch task, scoped from that same query, was remediation. The &lt;code&gt;reboots&lt;/code&gt; fact was reporting, telling you what's still exposed and what needs a bounce. &lt;/li&gt;
&lt;li&gt;Enforcement was the bit that holds the line afterwards, so a reprovisioned box can't slip back in unnoticed. &lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;That's the whole vulnerability-response loop, and the worst time to start building it is the morning a CVE lands with the clock already running.&lt;/p&gt;

&lt;p&gt;That's the case for putting the capability in now, while nothing's on fire. The next disclosure becomes a query and a patch run instead of a fortnight of spreadsheets and change tickets. The same loop applies well beyond library upgrades, too. &lt;a href="https://dev.to/puppet/handling-dirty-frag-and-copy-fail-with-puppet-6ff"&gt;We recently showed how to deal with dirty frag and copy-fail with Puppet&lt;/a&gt;, which walks the same detect-mitigate-remediate pattern on a different class of problem.&lt;/p&gt;

&lt;p&gt;Using Puppet provides a consistent way to address OpenSSL vulnerabilities across environments. It'll still be set up and ready when the next vulnerability lands.&lt;/p&gt;

&lt;h2&gt;
  
  
  🤖 AI Disclosure
&lt;/h2&gt;

&lt;p&gt;This article has been reviewed by a human expert in the subject matter and all code samples have been reviewed by Puppet technical experts. Initial structural content has been generated by Claude AI tools to assist with editing for clarity, structure, grammar, and maintaining brand voice, and then passed through human review.&lt;/p&gt;

</description>
      <category>puppet</category>
      <category>devops</category>
      <category>cybersecurity</category>
      <category>security</category>
    </item>
    <item>
      <title>What You Need to Know About the New puppetlabs-stdlib 10 in June 2026</title>
      <dc:creator>Jason St-Cyr</dc:creator>
      <pubDate>Thu, 11 Jun 2026 14:31:40 +0000</pubDate>
      <link>https://dev.to/puppet/what-you-need-to-know-about-the-new-puppetlabs-stdlib-10-in-june-2026-1jah</link>
      <guid>https://dev.to/puppet/what-you-need-to-know-about-the-new-puppetlabs-stdlib-10-in-june-2026-1jah</guid>
      <description>&lt;h2&gt;
  
  
  The&amp;nbsp;TL;DR&amp;nbsp;
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;  We are releasing&amp;nbsp;puppetlabs-stdlib&amp;nbsp;10.0.1&amp;nbsp;with a target date of&amp;nbsp;June 30, 2026.&lt;/li&gt;
&lt;li&gt;  This is a major version bump because we are dropping support for Puppet 7 (which&amp;nbsp;reached its end-of-life&amp;nbsp;in&amp;nbsp;February&amp;nbsp;2025) and requiring Ruby 3.1+.&lt;/li&gt;
&lt;li&gt;  If your modules pin&amp;nbsp;stdlib&amp;nbsp;to &amp;lt; 10.0.0, they will continue to work as-is on&amp;nbsp;stdlib&amp;nbsp;9.x.&amp;nbsp;&amp;nbsp;&lt;/li&gt;
&lt;li&gt;  You do not need to take any immediate action, but we wanted to give the community advance&amp;nbsp;notice&amp;nbsp;so module owners have time to plan.&amp;nbsp;&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Why a Major Release?&amp;nbsp;
&lt;/h2&gt;

&lt;p&gt;The&amp;nbsp;&lt;a href="https://forge.puppet.com/modules/puppetlabs/stdlib/readme" rel="noopener noreferrer"&gt;puppetlabs-stdlib&amp;nbsp;module&lt;/a&gt; is one of the most widely depended-upon&amp;nbsp;open source&amp;nbsp;modules in the Puppet ecosystem. Because of that, we take major version bumps seriously and want to be transparent about what is changing and why.&amp;nbsp;&lt;/p&gt;

&lt;p&gt;The last release of&amp;nbsp;stdlib&amp;nbsp;(9.7.0) was published in December 2024. Since then, maintenance work has accumulated in the main branch,&amp;nbsp;updates to Puppet Core tooling,&amp;nbsp;CentOS 9 support,&amp;nbsp;Rubocop&amp;nbsp;alignment, CI and testing infrastructure updates, and various bug fixes and enhancements contributed by both Puppet engineers and community members.&amp;nbsp;&lt;/p&gt;

&lt;p&gt;Among those accumulated changes is the&amp;nbsp;&lt;strong&gt;removal of Puppet 7 from the supported platforms&lt;/strong&gt;&amp;nbsp;in the module metadata. Since Puppet 7 reached end-of-life over a year ago, this is a natural and expected housekeeping step. However, because removing a previously supported Puppet version is a backwards-incompatible change under&amp;nbsp;semver, it requires a major version bump.&amp;nbsp;By making a major version release along with the removal of Puppet 7 support, this will make a clean delineation for any users still running Puppet 7.&amp;nbsp;&lt;/p&gt;

&lt;p&gt;Additionally, the&amp;nbsp;Rubocop&amp;nbsp;and tooling updates in the branch enforce Ruby 3.1+ syntax standards (such as the shorthand hash syntax). While these are primarily code-style changes, they mean the module codebase is no longer guaranteed to run on Ruby versions older than 3.1, which further supports the decision to move to a new major version.&amp;nbsp;&lt;/p&gt;

&lt;h2&gt;
  
  
  What’s Included in&amp;nbsp;stdlib&amp;nbsp;10&amp;nbsp;
&lt;/h2&gt;

&lt;p&gt;This release rolls up approximately 18 months of changes. The highlights include:&amp;nbsp;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Puppet 7 support removed:&lt;/strong&gt;&amp;nbsp;The module metadata now requires Puppet 8.0.0 or later. Puppet 7 has been end-of-life since early 2025.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Ruby 3.1+&amp;nbsp;required:&lt;/strong&gt;&amp;nbsp;Rubocop&amp;nbsp;and code style enforcement now targets Ruby 3.1 standards. Syntax that is incompatible with older Ruby versions has been adopted throughout the codebase.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;CentOS 9 support:&lt;/strong&gt;&amp;nbsp;The module metadata now&amp;nbsp;states&amp;nbsp;support for CentOS 9.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Bug fixes and enhancements:&lt;/strong&gt;&amp;nbsp;Multiple community-contributed fixes and improvements that have been pending release.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;CI and testing updates:&lt;/strong&gt;&amp;nbsp;Updated&amp;nbsp;testing infrastructure using Puppet Core tooling, updated nightly test matrices, and improved CI workflows.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;A full changelog will&amp;nbsp;accompany&amp;nbsp;the release on the Puppet Forge and on GitHub.&amp;nbsp;&lt;/p&gt;

&lt;h2&gt;
  
  
  What This Means for Module Authors&amp;nbsp;
&lt;/h2&gt;

&lt;p&gt;If you&amp;nbsp;maintain&amp;nbsp;a Puppet module that depends on&amp;nbsp;puppetlabs-stdlib, here is what you need to know.&amp;nbsp;&lt;/p&gt;

&lt;h3&gt;
  
  
  If You Do Nothing&amp;nbsp;
&lt;/h3&gt;

&lt;p&gt;Your module will continue to work. Most modules pin their stdlib&amp;nbsp;dependency with an upper bound like &lt;code&gt;"puppetlabs/stdlib": "&amp;gt;= 4.0.0 &amp;lt; 10.0.0"&lt;/code&gt;. The Puppet module resolver will keep you on the latest 9.x release. Nothing breaks, nothing changes.&amp;nbsp;&lt;/p&gt;

&lt;p&gt;If your module did not specify an upper bound, your module will be able to start using version 10 without any changes.&amp;nbsp;&lt;/p&gt;

&lt;h3&gt;
  
  
  If You Want to Adopt&amp;nbsp;stdlib&amp;nbsp;10&amp;nbsp;
&lt;/h3&gt;

&lt;p&gt;When you are ready, update the upper bound of your&amp;nbsp;stdlib&amp;nbsp;dependency in your&amp;nbsp;&lt;code&gt;metadata.json&lt;/code&gt;:&amp;nbsp;&lt;/p&gt;

&lt;p&gt;&lt;code&gt;"puppetlabs/stdlib": "&amp;gt;= 4.0.0 &amp;lt; 11.0.0"&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;This will allow your module to resolve either&amp;nbsp;stdlib&amp;nbsp;9.x or 10.x, giving your users flexibility. You should also confirm that your module no longer requires Puppet 7 support and that&amp;nbsp;your&amp;nbsp;testing and CI pipelines use Ruby 3.1 or later.&amp;nbsp;&lt;/p&gt;

&lt;h3&gt;
  
  
  Timing Is Up to You&amp;nbsp;
&lt;/h3&gt;

&lt;p&gt;There is no urgency to adopt&amp;nbsp;stdlib&amp;nbsp;10 on day one. The 9.x line will remain available on the Forge. We encourage module owners to update at their own pace, and we are providing this advance notice specifically so you can plan that work into your roadmap rather than being caught off-guard.&amp;nbsp;&lt;/p&gt;

&lt;h2&gt;
  
  
  What&amp;nbsp;Perforce&amp;nbsp;Puppet Is Doing to Prepare&amp;nbsp;
&lt;/h2&gt;

&lt;p&gt;We recognize that&amp;nbsp;stdlib&amp;nbsp;touches a huge&amp;nbsp;portion&amp;nbsp;of the module ecosystem. There are approximately&amp;nbsp;34&amp;nbsp;puppetlabs&amp;nbsp;modules&amp;nbsp;that depend on&amp;nbsp;stdlib. Between now and the release date, we will be updating&amp;nbsp;all of&amp;nbsp;those modules to expand their dependency bounds to accept&amp;nbsp;stdlib&amp;nbsp;10.x. These updated modules will be released to the Forge ahead of or alongside the&amp;nbsp;stdlib&amp;nbsp;10 release so that the transition is as smooth as possible.&amp;nbsp;&lt;/p&gt;

&lt;p&gt;Our goal is that by the time&amp;nbsp;stdlib&amp;nbsp;10 lands on the Forge, the&amp;nbsp;puppetlabs&amp;nbsp;module ecosystem will already be ready for it.&amp;nbsp;&lt;/p&gt;

&lt;h3&gt;
  
  
  Timeline&amp;nbsp;
&lt;/h3&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Date&lt;/th&gt;
&lt;th&gt;Milestone&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;Now&lt;/td&gt;
&lt;td&gt;This announcement. Community has advance notice.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;June 18–30&lt;/td&gt;
&lt;td&gt;Perforce will release updated dependency bounds across puppetlabs modules.&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;June 30, 2026&lt;/td&gt;
&lt;td&gt;&lt;strong&gt;Target release date for puppetlabs-stdlib 10 on the Puppet Forge.&lt;/strong&gt;&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;h2&gt;
  
  
  A Note on Communication&amp;nbsp;
&lt;/h2&gt;

&lt;p&gt;We want to acknowledge that an earlier attempt to release&amp;nbsp;stdlib&amp;nbsp;10.0.0 was made without sufficient advance communication to the community. We heard the feedback, rolled that release back, and are now doing this the right way: giving you notice, giving you time, and coordinating the broader module ecosystem before the release lands.&amp;nbsp;&lt;/p&gt;

&lt;p&gt;Major&amp;nbsp;stdlib&amp;nbsp;releases have historically been disruptive because of how deeply the module is embedded across the ecosystem. We are committed to making this transition as smooth as possible, and your feedback during this notice period is welcome and appreciated.&amp;nbsp;&lt;/p&gt;

&lt;h2&gt;
  
  
  How to Prepare&amp;nbsp;
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Review your dependency bounds:&lt;/strong&gt;&amp;nbsp;Check your&amp;nbsp;metadata.json&amp;nbsp;for your&amp;nbsp;stdlib&amp;nbsp;version constraint.&amp;nbsp; &lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Check your Puppet version support:&lt;/strong&gt;&amp;nbsp;If your module still claims Puppet 7 support, consider whether that is still necessary given Puppet 7 is end-of-life.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Check your Ruby version:&lt;/strong&gt;&amp;nbsp;Ensure your testing and CI environments use Ruby 3.1 or later.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Test against the main branch:&lt;/strong&gt;&amp;nbsp;If you want to verify compatibility ahead of the release, you can test your module against the main branch of&amp;nbsp;puppetlabs-stdlib&amp;nbsp;on GitHub.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Questions and Feedback&amp;nbsp;
&lt;/h2&gt;

&lt;p&gt;We want to hear from you. If you have questions, concerns, or feedback about this upcoming release, please reach out through any of the following channels:&amp;nbsp;&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Community Slack:&lt;/strong&gt;&amp;nbsp;The &lt;a href="https://slack.puppet.com" rel="noopener noreferrer"&gt;Puppet community Slack workspace&lt;/a&gt;, particularly the &lt;code&gt;#forge-modules&lt;/code&gt; channel where we have already had some discussion on this upcoming change.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Comments on this post:&lt;/strong&gt;&amp;nbsp;We will be&amp;nbsp;monitoring&amp;nbsp;and responding to comments here on dev.to.&amp;nbsp;&lt;/p&gt;&lt;/li&gt;
&lt;li&gt;&lt;p&gt;&lt;strong&gt;Perforce Forums:&lt;/strong&gt; Join the discussion on the&amp;nbsp;&lt;a href="https://portal.perforce.com/s/group/0F9PA000000085d0AA/puppet-product-discussion" rel="noopener noreferrer"&gt;Perforce Puppet forums&lt;/a&gt;, part of the official Perforce Portal.&lt;/p&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Thank you for being part of the Puppet community!&amp;nbsp;We appreciate your&amp;nbsp;collaboration&amp;nbsp;and your contributions, and we are looking forward to getting&amp;nbsp;this release&amp;nbsp;into your hands.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://forge.puppet.com/modules/puppetlabs/stdlib/readme" class="crayons-btn crayons-btn--primary" rel="noopener noreferrer"&gt;stdlib Forge Module&lt;/a&gt;
&lt;/p&gt;

</description>
      <category>puppet</category>
      <category>opensource</category>
      <category>devops</category>
    </item>
    <item>
      <title>Puppetlabs Modules Roundup – May 2026</title>
      <dc:creator>Jason St-Cyr</dc:creator>
      <pubDate>Wed, 03 Jun 2026 21:48:48 +0000</pubDate>
      <link>https://dev.to/puppet/puppetlabs-modules-roundup-may-2026-2gp2</link>
      <guid>https://dev.to/puppet/puppetlabs-modules-roundup-may-2026-2gp2</guid>
      <description>&lt;p&gt;This time around we look back at May 2026 and the 11 Puppetlabs module releases on the Forge, with an emphasis on the changes most likely to matter in active environments.&lt;/p&gt;

&lt;h2&gt;
  
  
  Highlighted Updates
&lt;/h2&gt;

&lt;h3&gt;
  
  
  New Windows audit policy module released!
&lt;/h3&gt;

&lt;p&gt;The new &lt;a href="https://forge.puppet.com/modules/puppetlabs/audit_policy/readme" rel="noopener noreferrer"&gt;audit_policy module&lt;/a&gt; has been released by Perforce as a Ruby replacement for the generated &lt;a href="https://forge.puppet.com/modules/dsc/auditpolicydsc/readme" rel="noopener noreferrer"&gt;DSC community auditpolicydsc module&lt;/a&gt;.&lt;/p&gt;

&lt;p&gt;This module uses Puppet Resources API for managing Windows audit policy using &lt;code&gt;auditpol.exe&lt;/code&gt;&lt;/p&gt;

&lt;h3&gt;
  
  
  ruby_task_helper Dependency Bound Update
&lt;/h3&gt;

&lt;p&gt;Five Bolt-adjacent modules all bumped the ruby_task_helper upper bound to &amp;lt; 2.0.0 in a coordinated maintenance pass, helping with dependency resolution failures when using Bolt 5.x.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Affected modules: vault, terraform, http_request, gcloud_inventory, azure_inventory.&lt;/li&gt;
&lt;/ul&gt;

&lt;h3&gt;
  
  
  CentOS 9 Support
&lt;/h3&gt;

&lt;p&gt;Multiple modules added explicit CentOS 9 compatibility, expanding the Linux platform coverage in line with the broader Puppet ecosystem push.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Affected modules: concat, inifile.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  What Updates Happened to Puppetlabs Modules in May 2026?
&lt;/h2&gt;

&lt;p&gt;The following is an alphabetical listing of modules which received updates in May 2026. If a module had multiple versions released, the updates are collected together, numbered with the "latest" version available.&lt;/p&gt;




&lt;h3&gt;
  
  
  apt 11.3.1
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-19 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/apt" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release introduced an explicit hash value syntax while also adding a param to support purging keyrings and other community contributions.&lt;/p&gt;

&lt;p&gt;Includes monthly releases: 11.3.1 (2026-05-19), 11.3.0 (2026-05-18).&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Use explicit hash value syntax instead of shorthand &lt;a href="https://github.com/puppetlabs/puppetlabs-apt/pull/1285" rel="noopener noreferrer"&gt;#1285&lt;/a&gt; (&lt;a href="https://github.com/SugatD" rel="noopener noreferrer"&gt;SugatD&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;Add param for purging keyrings &lt;a href="https://github.com/puppetlabs/puppetlabs-apt/pull/1266" rel="noopener noreferrer"&gt;#1266&lt;/a&gt; (&lt;a href="https://github.com/bwitt" rel="noopener noreferrer"&gt;bwitt&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;Include components when suite does not end with slash &lt;a href="https://github.com/puppetlabs/puppetlabs-apt/pull/1259" rel="noopener noreferrer"&gt;#1259&lt;/a&gt; (&lt;a href="https://github.com/bwitt" rel="noopener noreferrer"&gt;bwitt&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;Bugfix - sources format and ensure =&amp;gt; absent fails &lt;a href="https://github.com/puppetlabs/puppetlabs-apt/pull/1243" rel="noopener noreferrer"&gt;#1243&lt;/a&gt; (&lt;a href="https://github.com/traylenator" rel="noopener noreferrer"&gt;traylenator&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;fix: allow plus signs in ppa &lt;a href="https://github.com/puppetlabs/puppetlabs-apt/pull/1222" rel="noopener noreferrer"&gt;#1222&lt;/a&gt; (&lt;a href="https://github.com/moritz-makandra" rel="noopener noreferrer"&gt;moritz-makandra&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;Fix and improve DEB822-style template &lt;a href="https://github.com/puppetlabs/puppetlabs-apt/pull/1212" rel="noopener noreferrer"&gt;#1212&lt;/a&gt; (&lt;a href="https://github.com/smortex" rel="noopener noreferrer"&gt;smortex&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  audit_policy 1.0.0
&lt;/h3&gt;

&lt;p&gt;🌟 &lt;strong&gt;&lt;em&gt;New Module:&lt;/em&gt;&lt;/strong&gt; 2026-05-29 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/audit_policy" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This new module allows you to manage Windows audit policy with auditpol.exe as a replacement for the generated DSC community module. Initial release contains:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;audit_policy_subcategory: manage Windows audit policy subcategories by display name using auditpol.exe&lt;/li&gt;
&lt;li&gt;audit_policy_guid: manage Windows audit policy subcategories by GUID using auditpol.exe&lt;/li&gt;
&lt;li&gt;audit_policy_option: manage global Windows audit policy options (CrashOnAuditFail, FullPrivilegeAuditing, AuditBaseObjects, AuditBaseDirectories)&lt;/li&gt;
&lt;li&gt;audit_policy_csv: manage Windows audit policy by importing settings from an auditpol /backup CSV file&lt;/li&gt;
&lt;li&gt;Support for Windows Server 2016, 2019, 2022, and 2025&lt;/li&gt;
&lt;li&gt;Pure Ruby implementation — no PowerShell dependency; replaces the dsc-auditpolicydsc community module&lt;/li&gt;
&lt;li&gt;Puppet requirement pinned to &amp;gt;= 8.0.0 &amp;lt; 9.0.0&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  azure_inventory 0.5.1
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-14 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/azure_inventory" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release bumped the ruby_task_helper upper bound to &lt;strong&gt;&amp;lt; 2.0.0&lt;/strong&gt;.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Bump ruby_task_helper upper bound to &amp;lt; 2.0.0 (&lt;a href="https://github.com/puppetlabs/puppetlabs-azure_inventory/pull/16" rel="noopener noreferrer"&gt;#16&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  cd4peadm 5.15.1
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-07 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/cd4peadm" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;A few highlights from this release:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Fixed an issue where SAML logins were failing.&lt;/li&gt;
&lt;li&gt;Fixed an issue where CD would not use the configured HTTP timeouts when making calls to the Azure DevOps API, resulting in unexpected timeout failures.&lt;/li&gt;
&lt;li&gt;Fixed an issue where GitLab merge request updates that do not involve code changes would trigger CD pipelines.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;CVE-2026-42198.&lt;/strong&gt; Updated to address this vulnerability.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Check the official &lt;a href="https://help.puppet.com/cdpe/current/Content/UserGuide/CDPE/ReleaseNotes/cd_release_notes.htm#Version5151" rel="noopener noreferrer"&gt;release notes for cd4peadm 5.15.1&lt;/a&gt; for the full details.&lt;/p&gt;




&lt;h3&gt;
  
  
  concat 10.0.0
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-19 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/concat" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release added support for CentOS 9 while also addressing runner images for Ubuntu 24.04.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Redact sensitive content &lt;a href="https://github.com/puppetlabs/puppetlabs-concat/pull/828" rel="noopener noreferrer"&gt;#828&lt;/a&gt; (&lt;a href="https://github.com/smortex" rel="noopener noreferrer"&gt;smortex&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(CAT-2296) Update github runner image to ubuntu-24.04 &lt;a href="https://github.com/puppetlabs/puppetlabs-concat/pull/823" rel="noopener noreferrer"&gt;#823&lt;/a&gt; (&lt;a href="https://github.com/shubhamshinde360" rel="noopener noreferrer"&gt;shubhamshinde360&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(CAT-2152) Add support for CentOS 9 &lt;a href="https://github.com/puppetlabs/puppetlabs-concat/pull/818" rel="noopener noreferrer"&gt;#818&lt;/a&gt; (&lt;a href="https://github.com/skyamgarp" rel="noopener noreferrer"&gt;skyamgarp&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;Allow user defined tag or list of tags &lt;a href="https://github.com/puppetlabs/puppetlabs-concat/pull/790" rel="noopener noreferrer"&gt;#790&lt;/a&gt; (&lt;a href="https://github.com/Lightning-" rel="noopener noreferrer"&gt;Lightning-&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;Use explicit hash value syntax instead of shorthand &lt;a href="https://github.com/puppetlabs/puppetlabs-concat/pull/835" rel="noopener noreferrer"&gt;#835&lt;/a&gt; (&lt;a href="https://github.com/SugatD" rel="noopener noreferrer"&gt;SugatD&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  gcloud_inventory 0.3.1
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-14 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/gcloud_inventory" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release bumped the ruby_task_helper upper bound to &lt;strong&gt;&amp;lt; 2.0.0&lt;/strong&gt;.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Bump ruby_task_helper upper bound to &amp;lt; 2.0.0 (&lt;a href="https://github.com/puppetlabs/puppetlabs-gcloud_inventory/pull/14" rel="noopener noreferrer"&gt;#14&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  http_request 0.3.2
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-14 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/http_request" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release bumped the ruby_task_helper upper bound to &lt;strong&gt;&amp;lt; 2.0.0&lt;/strong&gt;.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Bump ruby_task_helper upper bound to &amp;lt; 2.0.0 (&lt;a href="https://github.com/puppetlabs/puppetlabs-http_request/pull/18" rel="noopener noreferrer"&gt;#18&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  inifile 6.4.0
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-19 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/inifile" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;The inifile module now supports multiple values per key while also adding support for CentOS 9.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Add support for multiple values per key &lt;a href="https://github.com/puppetlabs/puppetlabs-inifile/pull/555" rel="noopener noreferrer"&gt;#555&lt;/a&gt; (&lt;a href="https://github.com/bwitt" rel="noopener noreferrer"&gt;bwitt&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(CAT-2152) Add support for CentOS 9 &lt;a href="https://github.com/puppetlabs/puppetlabs-inifile/pull/549" rel="noopener noreferrer"&gt;#549&lt;/a&gt; (&lt;a href="https://github.com/skyamgarp" rel="noopener noreferrer"&gt;skyamgarp&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  puppet_agent 4.28.0
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-07 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/puppet_agent" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;Updates to new tooling (Bolt, PDK Templates) as well as support for MacOS 26 and some pre-work for the upcoming Puppet Core 9.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;(PA-7824) Use newest Bolt &lt;a href="https://github.com/puppetlabs/puppetlabs-puppet_agent/pull/829" rel="noopener noreferrer"&gt;#829&lt;/a&gt; (&lt;a href="https://github.com/mhashizume" rel="noopener noreferrer"&gt;mhashizume&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(PA-7897) Update to pdk-templates 3.6.1.1 &lt;a href="https://github.com/puppetlabs/puppetlabs-puppet_agent/pull/825" rel="noopener noreferrer"&gt;#825&lt;/a&gt; (&lt;a href="https://github.com/joshcooper" rel="noopener noreferrer"&gt;joshcooper&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(PA-8250) Allow installation of puppetcore9-nightly packages &lt;a href="https://github.com/puppetlabs/puppetlabs-puppet_agent/pull/822" rel="noopener noreferrer"&gt;#822&lt;/a&gt; (&lt;a href="https://github.com/joshcooper" rel="noopener noreferrer"&gt;joshcooper&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(PA-8238) Add support for MacOS 26 in install_shell.sh &lt;a href="https://github.com/puppetlabs/puppetlabs-puppet_agent/pull/821" rel="noopener noreferrer"&gt;#821&lt;/a&gt; (&lt;a href="https://github.com/shubhamshinde360" rel="noopener noreferrer"&gt;shubhamshinde360&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(PA-8250) Restore windows command to check puppet service &lt;a href="https://github.com/puppetlabs/puppetlabs-puppet_agent/pull/826" rel="noopener noreferrer"&gt;#826&lt;/a&gt; (&lt;a href="https://github.com/joshcooper" rel="noopener noreferrer"&gt;joshcooper&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(PA-8041) Fix puppetcore8-nightly installs on rpm and mac &lt;a href="https://github.com/puppetlabs/puppetlabs-puppet_agent/pull/824" rel="noopener noreferrer"&gt;#824&lt;/a&gt; (&lt;a href="https://github.com/joshcooper" rel="noopener noreferrer"&gt;joshcooper&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(PA-8247) Add guard against other ruby process when installing &lt;a href="https://github.com/puppetlabs/puppetlabs-puppet_agent/pull/823" rel="noopener noreferrer"&gt;#823&lt;/a&gt; (&lt;a href="https://github.com/AriaXLi" rel="noopener noreferrer"&gt;AriaXLi&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  terraform 0.7.2
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-14 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/terraform" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release bumped the ruby_task_helper upper bound to &lt;strong&gt;&amp;lt; 2.0.0&lt;/strong&gt;.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Bump ruby_task_helper upper bound to &amp;lt; 2.0.0 (&lt;a href="https://github.com/puppetlabs/puppetlabs-terraform/pull/37" rel="noopener noreferrer"&gt;#37&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  vault 0.4.1
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-05-14 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/vault" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release bumped the ruby_task_helper upper bound to &lt;strong&gt;&amp;lt; 2.0.0&lt;/strong&gt;.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Bump ruby_task_helper upper bound to &amp;lt; 2.0.0 (&lt;a href="https://github.com/puppetlabs/puppetlabs-vault/pull/19" rel="noopener noreferrer"&gt;#19&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Until Next Time!
&lt;/h2&gt;

&lt;p&gt;That closes out the May 2026 update set. For deeper implementation detail, the linked module pages and release notes remain the best source of truth.&lt;/p&gt;

&lt;p&gt;If you have feedback on the roundup format or want a deeper look at a specific module area, the Perforce Community Slack is still the best place to continue the conversation.&lt;/p&gt;

&lt;p&gt;See you next month with a roundup for June releases!&lt;/p&gt;

</description>
      <category>puppet</category>
      <category>infrastructureascode</category>
    </item>
    <item>
      <title>Puppet Core 8.19 and PDK 3.7: Security Updates, Dependency Changes, and Windows Fixes</title>
      <dc:creator>Jason St-Cyr</dc:creator>
      <pubDate>Thu, 21 May 2026 16:17:55 +0000</pubDate>
      <link>https://dev.to/puppet/puppet-core-819-and-pdk-37-security-updates-dependency-changes-and-windows-fixes-1j6n</link>
      <guid>https://dev.to/puppet/puppet-core-819-and-pdk-37-security-updates-dependency-changes-and-windows-fixes-1j6n</guid>
      <description>&lt;p&gt;Puppet Core &lt;strong&gt;8.19.0&lt;/strong&gt; focuses largely on security hardening, with some dependency cleanup and a small but important fix for Windows user management.&lt;/p&gt;

&lt;p&gt;If you already run Puppet Core 8, this release is primarily about &lt;strong&gt;keeping your runtime secure and predictable&lt;/strong&gt;, rather than introducing new workflows or configuration changes.&lt;/p&gt;

&lt;p&gt;➡️ Full details: &lt;a href="https://help.puppet.com/core/current/Content/PuppetCore/PuppetReleaseNotes/release_notes_puppet_x-8-19-0.htm" rel="noopener noreferrer"&gt;Puppet Core 8.19.0 release notes&lt;/a&gt;&lt;/p&gt;

&lt;p&gt;PDK 3.7.0 was also released to improve performance on Windows, update dependencies, and provide other updates for security and known issues.&lt;/p&gt;

&lt;p&gt;➡️ Full details: &lt;a href="https://help.puppet.com/pdk/current/topics/release_notes_pdk.htm#PDK370" rel="noopener noreferrer"&gt;PDK 3.7.0 release notes&lt;/a&gt;&lt;/p&gt;




&lt;h2&gt;
  
  
  CSV gem dependency removed
&lt;/h2&gt;

&lt;p&gt;Puppet Core no longer depends on the &lt;strong&gt;CSV&lt;/strong&gt; Ruby gem.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Removes an external dependency from the Puppet runtime&lt;/li&gt;
&lt;li&gt;Reduces overall dependency surface area&lt;/li&gt;
&lt;li&gt;Simplifies installation and long-term maintenance&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;This change does not alter Puppet DSL behavior or require configuration changes.&lt;/p&gt;




&lt;h2&gt;
  
  
  Security updates
&lt;/h2&gt;

&lt;p&gt;Puppet Core 8.19.0 updates several bundled runtime components to address recently disclosed security vulnerabilities. These updates apply automatically when you upgrade.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Ruby&lt;/strong&gt; updated to &lt;strong&gt;3.2.11&lt;/strong&gt; (CVE-2026-27820)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;OpenSSL&lt;/strong&gt; updated to &lt;strong&gt;3.0.20&lt;/strong&gt; (CVE-2026-28387, CVE-2026-28388, CVE-2026-28389, CVE-2026-28390, CVE-2026-31789, CVE-2026-31790)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;libxml2&lt;/strong&gt; updated to &lt;strong&gt;2.15.3&lt;/strong&gt; (CVE-2026-6732)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;curl&lt;/strong&gt; updated to &lt;strong&gt;8.20.0&lt;/strong&gt; (CVE-2026-6253, CVE-2026-6276, CVE-2026-6429, CVE-2026-7009, CVE-2026-7168)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;net-imap&lt;/strong&gt; updated to &lt;strong&gt;0.4.24&lt;/strong&gt; (CVE-2026-42245, CVE-2026-42246, CVE-2026-42256, CVE-2026-42257, CVE-2026-42258)&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;erb&lt;/strong&gt; updated to &lt;strong&gt;6.0.4&lt;/strong&gt; (CVE-2026-41316)&lt;/li&gt;
&lt;/ul&gt;

&lt;h3&gt;
  
  
  ℹ️ Important note for net-imap users
&lt;/h3&gt;

&lt;p&gt;If you use net-imap directly in custom Ruby code, the updated version enforces stricter argument validation. &lt;a href="https://help.puppet.com/core/current/Content/PuppetCore/PuppetReleaseNotes/release_notes_puppet_x-8-19-0.htm" rel="noopener noreferrer"&gt;Check the release notes&lt;/a&gt; for some details on how to manage this gem if you cannot use the upgraded version of net-imap.&lt;/p&gt;




&lt;h2&gt;
  
  
  Windows user passwords now allow colons
&lt;/h2&gt;

&lt;p&gt;When managing &lt;strong&gt;user resources on Windows&lt;/strong&gt;, Puppet Core no longer rejects passwords containing colons (&lt;code&gt;:&lt;/code&gt;).&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Affects Windows platforms only&lt;/li&gt;
&lt;li&gt;Behavior on other platforms is unchanged&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;This prevents unnecessary failures when managing Windows accounts with valid password formats.&lt;/p&gt;




&lt;h2&gt;
  
  
  Installation safeguards for Ruby versions
&lt;/h2&gt;

&lt;p&gt;Puppet Core now prevents installation on unsupported Ruby versions.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Puppet Core 8 enforces a &lt;strong&gt;maximum supported Ruby version of 3.x&lt;/strong&gt;
&lt;/li&gt;
&lt;li&gt;Prevents installation attempts using unsupported versions (such as Ruby 4)&lt;/li&gt;
&lt;li&gt;Protects against installation failures like &lt;code&gt;can't modify frozen Hash&lt;/code&gt;
&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;This applies when installing Puppet Core via &lt;strong&gt;bundler or gem commands&lt;/strong&gt;.&lt;/p&gt;




&lt;h2&gt;
  
  
  PDK 3.7.0 changes
&lt;/h2&gt;

&lt;p&gt;This version of PDK was updated to help prevent security issues and reduce test failures, and made macOS 15 downloads available.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Rexml updated to version 3.4.4 to address CVE-2025-58767&lt;/li&gt;
&lt;li&gt;macOS 15 downloads now available&lt;/li&gt;
&lt;li&gt;Windows performance has been improved&lt;/li&gt;
&lt;li&gt;YAML file validation issues should be resolved&lt;/li&gt;
&lt;li&gt;Several &lt;code&gt;puppet_forge&lt;/code&gt; gems and dependencies were updated&lt;/li&gt;
&lt;/ul&gt;




&lt;h2&gt;
  
  
  Should you upgrade?
&lt;/h2&gt;

&lt;p&gt;Upgrading to Puppet Core &lt;strong&gt;8.19.0&lt;/strong&gt; is recommended for Puppet Core 8.x users if:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;You want current security fixes for bundled runtime dependencies&lt;/li&gt;
&lt;li&gt;You manage Windows users with complex passwords&lt;/li&gt;
&lt;li&gt;You want to avoid accidental installation on unsupported Ruby versions&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Want to get started? Here are the install/upgrade guides:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;a href="https://help.puppet.com/core/current/Content/PuppetCore/installing_and_upgrading.htm" rel="noopener noreferrer"&gt;Install docs&lt;/a&gt;&lt;/li&gt;
&lt;li&gt;&lt;a href="https://help.puppet.com/core/current/Content/PuppetCore/upgrade.htm" rel="noopener noreferrer"&gt;Upgrade docs&lt;/a&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;For full details and CVE listings, see the release notes:&lt;/p&gt;

&lt;p&gt;➡️ &lt;a href="https://help.puppet.com/core/current/Content/PuppetCore/PuppetReleaseNotes/release_notes_puppet_x-8-19-0.htm" rel="noopener noreferrer"&gt;Puppet Core 8.19.0 release notes&lt;/a&gt;&lt;br&gt;
➡️ &lt;a href="https://help.puppet.com/pdk/current/topics/release_notes_pdk.htm#PDK370" rel="noopener noreferrer"&gt;PDK 3.7.0 release notes&lt;/a&gt;&lt;/p&gt;

</description>
      <category>puppet</category>
      <category>devops</category>
      <category>infrastructureascode</category>
    </item>
    <item>
      <title>Creating Successful Migration Workflows with Puppet</title>
      <dc:creator>Tony Green</dc:creator>
      <pubDate>Mon, 18 May 2026 10:11:59 +0000</pubDate>
      <link>https://dev.to/puppet/creating-successful-migration-workflows-with-puppet-4b8n</link>
      <guid>https://dev.to/puppet/creating-successful-migration-workflows-with-puppet-4b8n</guid>
      <description>&lt;h2&gt;
  
  
  The goal isn't to move things. It's to only move them once.
&lt;/h2&gt;

&lt;p&gt;I've been doing this for over thirty years.&lt;/p&gt;

&lt;p&gt;Sysadmin, ops lead, global teams, and more data centre migrations than I'd like to admit. Site to site, P2V, V2V, cloud, hybrid, all of it.&lt;/p&gt;

&lt;p&gt;Every migration gets sold as a clean, well-planned transition. None of them are.&lt;/p&gt;

&lt;p&gt;They go wrong in very predictable ways. Not because moving infrastructure is especially difficult, but because nobody ever has a clear, current view of what's actually running, what's changed, and what still matters. So people fall back to spreadsheets, SSH sessions, scripts written at 2am, and a lot of "we think this is right".&lt;/p&gt;

&lt;p&gt;That's where migrations fail. Not in the move itself, but in the loss of control around it.&lt;/p&gt;

&lt;h3&gt;
  
  
  Why migrations go wrong
&lt;/h3&gt;

&lt;p&gt;After you've done a few of these, the pattern is obvious. Three things show up every time.&lt;/p&gt;

&lt;p&gt;The first is drift. Most environments look well understood on paper. In reality, versions don't match, configurations have wandered off in their own directions, "identical" servers aren't, and there are systems no one owns but everyone is afraid to touch. I once worked on an estate where three "identical" app servers were running three different JVM versions, and nobody could tell me which one production traffic was actually landing on. You start the migration with an incomplete picture and everything after that is guesswork. It gets worse once the migration is underway, because people make manual fixes to keep things moving, and nothing lasts as long as a temporary solution.&lt;/p&gt;

&lt;p&gt;The second is manual execution. No matter how good the plan looks in the slide deck, at some point it turns into a human running commands on a list of hosts they pasted out of a spreadsheet. Sequencing becomes tribal knowledge. Parallelisation becomes guesswork. The rollback plan, if anyone wrote one down, is usually a single line that says "restore from backup".&lt;/p&gt;

&lt;p&gt;The third is lost accountability. During a migration more people have access, more changes are happening, and less of it gets tracked properly. So when something breaks at 3am, nobody is quite sure what changed, or who changed it, or how to undo it. By the time you've worked it out, the on-call engineer has lost an evening and the project has lost a week of trust.&lt;/p&gt;

&lt;p&gt;None of this is new. What's interesting is that all three problems have the same root cause. There's no consistent way to know what your estate looks like and operate it predictably while it's moving.&lt;/p&gt;

&lt;h3&gt;
  
  
  The four things you need to stay in control
&lt;/h3&gt;

&lt;p&gt;Fixing this comes down to four things. You need to know what's actually out there. You need a way to say what it should look like. You need to be able to operate it safely at scale. And you need to know who's allowed to do what. Most teams have bits of all four scattered across different tools, which is why nothing quite lines up when it matters. &lt;a href="https://www.puppet.com/products/puppet-enterprise" rel="noopener noreferrer"&gt;Puppet Enterprise&lt;/a&gt; gives you all four in one place. One model, one set of controls, one audit trail. Not four tools you have to glue together yourself and pray they agree with each other.&lt;/p&gt;

&lt;h4&gt;
  
  
  Visibility
&lt;/h4&gt;

&lt;p&gt;Most migrations start with a discovery phase that's stale before the spreadsheet is finished.&lt;/p&gt;

&lt;p&gt;What you actually want is continuous visibility, and that's what &lt;code&gt;facter&lt;/code&gt; does. &lt;code&gt;facter&lt;/code&gt; runs on every node, collects system information, and reports it back centrally dozens of times a day. Out of the box you get the obvious things (OS, kernel, memory, network) but the part that matters during a migration is &lt;a href="https://help.puppet.com/core/current/Content/PuppetCore/custom_facts.htm" rel="noopener noreferrer"&gt;custom facts&lt;/a&gt;. You can teach &lt;code&gt;facter&lt;/code&gt; to collect anything you care about.&lt;/p&gt;

&lt;p&gt;Here's a small one that reports the version of an in-house payments app by reading a file the deploy pipeline drops on disk:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight ruby"&gt;&lt;code&gt;&lt;span class="no"&gt;Facter&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;add&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="ss"&gt;:payments_version&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="k"&gt;do&lt;/span&gt;
  &lt;span class="n"&gt;setcode&lt;/span&gt; &lt;span class="k"&gt;do&lt;/span&gt;
    &lt;span class="k"&gt;if&lt;/span&gt; &lt;span class="no"&gt;File&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;exist?&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'/opt/payments/VERSION'&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
      &lt;span class="no"&gt;File&lt;/span&gt;&lt;span class="p"&gt;.&lt;/span&gt;&lt;span class="nf"&gt;read&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'/opt/payments/VERSION'&lt;/span&gt;&lt;span class="p"&gt;).&lt;/span&gt;&lt;span class="nf"&gt;strip&lt;/span&gt;
    &lt;span class="k"&gt;end&lt;/span&gt;
  &lt;span class="k"&gt;end&lt;/span&gt;
&lt;span class="k"&gt;end&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Once that fact exists, every node running the app reports its version on every Puppet run. Five minutes later you can ask Puppet Enterprise "show me every node where payments_version is older than 4.2" and get a real answer, not a guess. Multiply that across the dozen things you actually care about (JVM version, storage layout, mount points, whether a vendor agent is running, whether a system is genuinely in use) and you've replaced your discovery spreadsheet with something that updates itself.&lt;/p&gt;

&lt;p&gt;That's the bit most teams never quite get to. Half the battle in a migration is just knowing what you've got.&lt;/p&gt;

&lt;h4&gt;
  
  
  State
&lt;/h4&gt;

&lt;p&gt;Once you know what you've got, the next problem is making sure it behaves the way you expect.&lt;/p&gt;

&lt;p&gt;This is the part Puppet has been doing for years. You define the desired state in code, and that code gets applied consistently across the old data centre, the new one, and any cloud environment you're bringing into the mix. No golden images quietly drifting. No "we rebuilt it by hand and it's nearly the same".&lt;/p&gt;

&lt;p&gt;The reason this matters during a migration is that it lets you stand up the new environment alongside the old one and keep them honest. Same roles, same profiles, same definitions. The question stops being "did we build this correctly?" and becomes "does it match the defined state?". The second question has an answer. The first one rarely does.&lt;/p&gt;

&lt;h4&gt;
  
  
  Execution
&lt;/h4&gt;

&lt;p&gt;At some point you actually have to do things. Stop services, drain load balancers, run pre-flight checks, kick off data sync, validate the result, move on. This is the bit that usually collapses into SSH and good intentions.&lt;/p&gt;

&lt;p&gt;&lt;a href="https://www.puppet.com/docs/pe/2025.0/running_jobs_with_puppet_orchestrator_overview.html" rel="noopener noreferrer"&gt;Puppet Tasks and Plans&lt;/a&gt; give you a way to do it properly. Tasks run actions across large numbers of nodes. Plans sequence those actions and react to what they return. You target nodes by what Puppet already knows about them, not by a hand-built host list, which means the targeting stays correct as the estate changes underneath you.&lt;/p&gt;

&lt;p&gt;Here's roughly what a cutover step looks like as a Plan:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight puppet"&gt;&lt;code&gt;&lt;span class="n"&gt;plan&lt;/span&gt; &lt;span class="nf"&gt;migration::drain_web&lt;/span&gt; &lt;span class="p"&gt;(&lt;/span&gt;
  &lt;span class="nc"&gt;TargetSpec&lt;/span&gt; &lt;span class="nv"&gt;$nodes&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;)&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt;
  &lt;span class="nv"&gt;$targets&lt;/span&gt; &lt;span class="o"&gt;=&lt;/span&gt; &lt;span class="nf"&gt;get_targets&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="nv"&gt;$nodes&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="nf"&gt;run_command&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'systemctl stop nginx'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nv"&gt;$targets&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="nf"&gt;run_task&lt;/span&gt;&lt;span class="p"&gt;(&lt;/span&gt;&lt;span class="s1"&gt;'healthcheck::wait_for_drain'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="nv"&gt;$targets&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="py"&gt;timeout&lt;/span&gt; &lt;span class="p"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="m"&gt;300&lt;/span&gt;&lt;span class="p"&gt;)&lt;/span&gt;
  &lt;span class="n"&gt;return&lt;/span&gt; &lt;span class="s2"&gt;"drained &lt;/span&gt;&lt;span class="err"&gt;$&lt;/span&gt;&lt;span class="s2"&gt;{targets.size} nodes"&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Nothing clever. The point is that it's the same plan whether you're draining four nodes or four hundred, it runs through the orchestrator with the same RBAC and the same logging as everything else, and you can call it from another plan that drains the web tier, then the app tier, then flips DNS. The chaos of cutover night turns into something you can rehearse.&lt;/p&gt;

&lt;h4&gt;
  
  
  Control
&lt;/h4&gt;

&lt;p&gt;The last piece is the one most people only think about after something has already broken.&lt;/p&gt;

&lt;p&gt;During a migration more people have access, more changes are happening, and the blast radius of a mistake is bigger. You need to know who can define behaviour, who can execute it, and who can see what's going on. Puppet Enterprise handles that through &lt;a href="https://docs.puppet.com/pe/latest/rbac_intro.html" rel="noopener noreferrer"&gt;RBAC&lt;/a&gt; and its reporting layer. You can let a team run specific tasks against specific systems without handing them the keys to the kingdom, and you can give read-only visibility to the people who need to know what's happening without putting them in the room.&lt;/p&gt;

&lt;p&gt;The same controls apply no matter how the work is triggered. A change ticket in ServiceNow, a step in a CI/CD pipeline, an approval workflow, or someone clicking a button in the console all run the same plans against the same nodes. Same RBAC checks, same audit trail. That matters during a migration because cutover work doesn't all come from one place. Some of it is planned change, some is automated, some is "the network team needs us to drain that rack in the next ten minutes", and you don't want each route to have its own access model and its own log.&lt;/p&gt;

&lt;p&gt;If you're running PE Advanced, the same model extends into &lt;a href="https://www.puppet.com/products/security-compliance-enforcement" rel="noopener noreferrer"&gt;compliance&lt;/a&gt;. Continuous reporting, CIS benchmarks running on every node, and the ability to show an auditor that the new environment was in the expected state on the day you cut over. During a migration that evidence is the difference between "we think it went well" and "here's the report".&lt;/p&gt;

&lt;h4&gt;
  
  
  A few things that come along for the ride
&lt;/h4&gt;

&lt;p&gt;If part of your migration is moving workloads into the cloud, PE Advanced ships with CloudOps and FinOps capabilities that are genuinely useful in flight, not just once you've landed.&lt;/p&gt;

&lt;p&gt;Once your cloud accounts are connected, you get visibility of what's actually running, what it's costing, and what's been left switched on by mistake. Migrations are very good at producing all three: orphaned instances, oversized VMs that someone picked "to be safe", and test environments that nobody remembered to turn off.&lt;/p&gt;

&lt;p&gt;The bit that makes this useful rather than just another dashboard is that it's tied to the same node inventory you're already managing with Puppet. So the cost story doesn't end up as a separate project six months after cutover, run by people who don't know which servers were meant to be there in the first place.&lt;/p&gt;

&lt;p&gt;A couple of other things in PE Advanced are worth knowing about for the same reason. The &lt;a href="https://www.puppet.com/products/observability" rel="noopener noreferrer"&gt;data connector&lt;/a&gt; pushes Puppet's facts, reports, and events into whatever SIEM or observability platform you already run, so the migration shows up in the same dashboards your SOC and SREs are already watching, instead of being a parallel universe nobody looks at. And the &lt;a href="https://www.puppet.com/integrations/servicenow" rel="noopener noreferrer"&gt;ServiceNow integration&lt;/a&gt; syncs bi-directionally with your CMDB, which means the live view Puppet has of the estate stops drifting away from the system of record the rest of the business is using. Both of those matter more during a migration than at any other point in the life of an environment, because both are usually where the wheels come off after cutover.&lt;/p&gt;

&lt;h3&gt;
  
  
  You don't have to throw away what you've got
&lt;/h3&gt;

&lt;p&gt;One thing worth saying clearly, because it comes up in every conversation I have about this. You don't need to rip out your existing tooling to do any of it.&lt;/p&gt;

&lt;p&gt;Most environments I see are a mix of on-prem and cloud, multiple operating systems, and whatever automation has grown over the years. Usually that includes Ansible, a pile of homegrown scripts, and a fair bit of "don't touch that, it works". Trying to standardise all of that during a migration is a mistake. It adds risk at exactly the moment you want less of it.&lt;/p&gt;

&lt;p&gt;A more practical approach is to orchestrate what you already have. Puppet Tasks can call existing scripts and Ansible playbooks, sequence them alongside Puppet-managed actions, and target the right systems based on facts and roles. Instead of maintaining multiple inventories and disconnected workflows, you get one control layer driving everything you already run.&lt;/p&gt;

&lt;p&gt;Puppet isn't replacing those tools. It's coordinating them. That lets you keep what works, cut the duplication, and converge on a more consistent model over time, which is a much safer place to be in the middle of a migration than mid-rewrite.&lt;/p&gt;

&lt;h3&gt;
  
  
  Final thought
&lt;/h3&gt;

&lt;p&gt;A data centre migration isn't really a logistics problem. It's a control problem.&lt;/p&gt;

&lt;p&gt;If you know what's out there, what state it's in, and who's changing it, you can move systems safely, validate them properly, and recover when things don't go to plan. If you can't, it doesn't matter how good the plan looked at kickoff. You'll be discovering the gaps the hard way, one outage at a time.&lt;/p&gt;

&lt;p&gt;The goal of a migration isn't to move things. It's to only have to move them once.&lt;/p&gt;

</description>
      <category>puppet</category>
      <category>automation</category>
      <category>cloud</category>
    </item>
    <item>
      <title>Handling Dirty Frag and Copy Fail with Puppet</title>
      <dc:creator>Tony Green</dc:creator>
      <pubDate>Wed, 13 May 2026 21:00:55 +0000</pubDate>
      <link>https://dev.to/puppet/handling-dirty-frag-and-copy-fail-with-puppet-6ff</link>
      <guid>https://dev.to/puppet/handling-dirty-frag-and-copy-fail-with-puppet-6ff</guid>
      <description>&lt;h2&gt;
  
  
  Do you know which of your Linux servers are vulnerable right now?
&lt;/h2&gt;

&lt;p&gt;Two critical Linux kernel vulnerabilities are being actively exploited. Dirty Frag targets the IP fragment reassembly modules that almost every Linux server has loaded by default. Copy Fail targets the AF_ALG cryptographic subsystem's AEAD interface. Both can allow an attacker to gain escalated local privileges, and both have interim mitigations you can deploy right now, before the kernel patches land.&lt;/p&gt;

&lt;p&gt;The first question every team needs to answer is not "how do we fix it?" but "how many of our systems are actually exposed?" Not tomorrow, after someone runs a scanner. Not next week, when the security team finishes their audit. Right now.&lt;/p&gt;

&lt;p&gt;If you are running Puppet, the answer is already within reach. You just need a fact.&lt;/p&gt;

&lt;h3&gt;
  
  
  Add these to your Puppetfile. That is it.
&lt;/h3&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight ruby"&gt;&lt;code&gt;&lt;span class="n"&gt;mod&lt;/span&gt; &lt;span class="s1"&gt;'albatrossflavour-dirty_frag'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="s1"&gt;'1.0.1'&lt;/span&gt;
&lt;span class="n"&gt;mod&lt;/span&gt; &lt;span class="s1"&gt;'albatrossflavour-copy_fail'&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt; &lt;span class="s1"&gt;'1.0.0'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Each of these open-source modules, that I have published to the Forge, ship a custom structured fact that reports vulnerability exposure on every node. No class to include. No Hiera changes. No manifest edits. Next Puppet run, the data is there.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;a href="https://github.com/albatrossflavour/dirty_frag" rel="noopener noreferrer"&gt;dirty_frag&lt;/a&gt; reports exposure to Dirty Frag (&lt;a href="https://nvd.nist.gov/vuln/detail/CVE-2026-43284" rel="noopener noreferrer"&gt;CVE-2026-43284&lt;/a&gt; and &lt;a href="https://nvd.nist.gov/vuln/detail/CVE-2026-43500" rel="noopener noreferrer"&gt;CVE-2026-43500&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;
&lt;a href="https://github.com/albatrossflavour/copy_fail" rel="noopener noreferrer"&gt;copy_fail&lt;/a&gt; reports exposure to Copy Fail (&lt;a href="https://nvd.nist.gov/vuln/detail/CVE-2026-31431" rel="noopener noreferrer"&gt;CVE-2026-31431&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;

&lt;h3&gt;
  
  
  See where you stand
&lt;/h3&gt;

&lt;p&gt;Once deployed, every node reports its exposure state as structured data. Here is what the output looks like.&lt;/p&gt;

&lt;h4&gt;
  
  
  dirty_frag
&lt;/h4&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight json"&gt;&lt;code&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"esp4"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"loaded"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"blocked"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"available"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="p"&gt;},&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"esp6"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"loaded"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"blocked"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"available"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="p"&gt;},&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"rxrpc"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"loaded"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"blocked"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"available"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="p"&gt;},&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"vulnerable"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"reboot_required"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h4&gt;
  
  
  copy_fail
&lt;/h4&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight json"&gt;&lt;code&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"algif_aead"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="p"&gt;{&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"type"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="s2"&gt;"builtin"&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"loaded"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"active"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"blocked"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
    &lt;/span&gt;&lt;span class="nl"&gt;"available"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="p"&gt;},&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"initcall_blacklisted"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"vulnerable"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"mitigated"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;&lt;span class="w"&gt;
  &lt;/span&gt;&lt;span class="nl"&gt;"reboot_required"&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;&lt;span class="w"&gt; &lt;/span&gt;&lt;span class="kc"&gt;false&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;span class="p"&gt;}&lt;/span&gt;&lt;span class="w"&gt;
&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Both facts surface the same two summary keys that give you the answers you actually care about. &lt;code&gt;vulnerable&lt;/code&gt; is &lt;code&gt;true&lt;/code&gt; if the exploitable module is active. &lt;code&gt;reboot_required&lt;/code&gt; is &lt;code&gt;true&lt;/code&gt; if you have applied the mitigation but the module is still sitting in kernel memory.&lt;/p&gt;

&lt;p&gt;That &lt;code&gt;reboot_required&lt;/code&gt; flag is the one that catches people. You apply the mitigation, see the block in place, and assume you are safe. But the module is still loaded and exploitable until the machine restarts. The facts make this gap visible.&lt;/p&gt;

&lt;h3&gt;
  
  
  Query your entire fleet in one line
&lt;/h3&gt;

&lt;p&gt;The facts land in PuppetDB like any other structured fact. No extra tooling, no scheduled scans, no agents phoning home to a separate platform. It is just data, maintained automatically every Puppet run.&lt;/p&gt;

&lt;p&gt;Show me every node vulnerable to Dirty Frag:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;puppet query &lt;span class="s1"&gt;'facts[certname, value] { name = "dirty_frag" and value.vulnerable = true }'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Show me every node vulnerable to Copy Fail:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;puppet query &lt;span class="s1"&gt;'facts[certname, value] { name = "copy_fail" and value.vulnerable = true }'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Show me nodes where the Dirty Frag block is applied but a reboot is still needed:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;puppet query &lt;span class="s1"&gt;'facts[certname, value] { name = "dirty_frag" and value.reboot_required = true }'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Show me nodes where &lt;code&gt;algif_aead&lt;/code&gt; is built into the kernel (the harder mitigation path):&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;puppet query &lt;span class="s1"&gt;'facts[certname, value] { name = "copy_fail" and value.algif_aead.type = "builtin" }'&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;This is the bit that makes Puppet's model shine. You are not running a point-in-time scan. The data updates every run, automatically. As nodes get patched, rebooted, or have blocks applied, the numbers go down on their own. You can watch your exposure shrink in real time without maintaining anything.&lt;/p&gt;

&lt;h2&gt;
  
  
  What makes a system vulnerable?
&lt;/h2&gt;

&lt;h3&gt;
  
  
  Dirty Frag
&lt;/h3&gt;

&lt;p&gt;The Dirty Frag vulnerability sits in three Linux kernel modules used for IP fragment reassembly: &lt;code&gt;esp4&lt;/code&gt;, &lt;code&gt;esp6&lt;/code&gt;, and &lt;code&gt;rxrpc&lt;/code&gt;. If any of those modules is loaded, the system is exposed. The two CVEs chain together to allow privilege escalation and remote code execution through these modules.&lt;/p&gt;

&lt;p&gt;These modules are almost universally loaded on production Linux servers. If you are running iptables, firewalld, Docker, Kubernetes, or any network policy enforcement, odds are at least &lt;code&gt;esp4&lt;/code&gt; is there. This one affects nearly everything.&lt;/p&gt;

&lt;h3&gt;
  
  
  Copy Fail
&lt;/h3&gt;

&lt;p&gt;The vulnerability is a use-after-free in the AF_ALG subsystem's AEAD interface (&lt;code&gt;algif_aead&lt;/code&gt;). An unprivileged user can trigger it by opening an AF_ALG socket and exercising the AEAD code path in a specific way.&lt;/p&gt;

&lt;p&gt;&lt;code&gt;algif_aead&lt;/code&gt; is present on most stock kernels but rarely used for legitimate work. The main consumers are some VPN implementations that offload crypto to the kernel, &lt;code&gt;kcapi-tools&lt;/code&gt;, and certain FIPS-certified configurations. On systems where the module is loadable, an attacker can force-load it themselves just by opening the socket. On systems where it is built-in, it may already be active regardless of use.&lt;/p&gt;

&lt;h3&gt;
  
  
  Why module presence, not kernel version?
&lt;/h3&gt;

&lt;p&gt;The real fix for both vulnerabilities is a kernel patch, but every distribution ships their own patched versions on their own schedules with their own version numbers. Tracking "which kernel version is safe" across Red Hat, Ubuntu, SUSE, Amazon Linux, and the rest is a moving target that goes stale the day you publish it. Module presence is the reliable signal: if the vulnerable module is active, you are exposed, regardless of kernel version.&lt;/p&gt;

&lt;p&gt;This is also what the vendor advisories recommend. Red Hat's &lt;a href="https://access.redhat.com/security/vulnerabilities/RHSB-2026-003" rel="noopener noreferrer"&gt;RHSB-2026-003&lt;/a&gt; and the equivalent Ubuntu and SUSE bulletins all point to the same interim mitigation: prevent the modules from loading using &lt;code&gt;install /bin/false&lt;/code&gt;.&lt;/p&gt;

&lt;h2&gt;
  
  
  When you want Puppet to enforce the block
&lt;/h2&gt;

&lt;p&gt;The facts give you visibility. If you also want Puppet to actively prevent the modules from loading, include the classes.&lt;/p&gt;

&lt;h3&gt;
  
  
  Dirty Frag
&lt;/h3&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight puppet"&gt;&lt;code&gt;&lt;span class="k"&gt;class&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="s1"&gt;'dirty_frag'&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;
  &lt;span class="py"&gt;mitigate_esp4&lt;/span&gt;  &lt;span class="p"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="py"&gt;mitigate_esp6&lt;/span&gt;  &lt;span class="p"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
  &lt;span class="py"&gt;mitigate_rxrpc&lt;/span&gt; &lt;span class="p"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h3&gt;
  
  
  Copy Fail
&lt;/h3&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight puppet"&gt;&lt;code&gt;&lt;span class="k"&gt;class&lt;/span&gt; &lt;span class="p"&gt;{&lt;/span&gt; &lt;span class="s1"&gt;'copy_fail'&lt;/span&gt;&lt;span class="p"&gt;:&lt;/span&gt;
  &lt;span class="py"&gt;mitigate_algif_aead&lt;/span&gt; &lt;span class="p"&gt;=&amp;gt;&lt;/span&gt; &lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="p"&gt;,&lt;/span&gt;
&lt;span class="p"&gt;}&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;Both classes write &lt;code&gt;install &amp;lt;module&amp;gt; /bin/false&lt;/code&gt; directives to modprobe.d configuration files. From that point on, any attempt to load those modules (whether by autoloading, a dependency chain, or an explicit &lt;code&gt;modprobe&lt;/code&gt;) will run &lt;code&gt;/bin/false&lt;/code&gt; instead of loading the actual module code. The module simply cannot get into the kernel.&lt;/p&gt;

&lt;p&gt;This is worth understanding because it is stronger than the kernel's &lt;code&gt;blacklist&lt;/code&gt; directive. A &lt;code&gt;blacklist&lt;/code&gt; entry only prevents autoloading, it does not stop explicit &lt;code&gt;modprobe&lt;/code&gt; calls. &lt;code&gt;install /bin/false&lt;/code&gt; intercepts the load at every entry point.&lt;/p&gt;

&lt;p&gt;If the module was already loaded before the block was applied, it stays in memory until the next reboot. The &lt;code&gt;reboot_required&lt;/code&gt; flag in both facts makes this gap visible.&lt;/p&gt;

&lt;p&gt;This is declarative and safe. Puppet manages the config file, ensures the desired state on every run, and the classes are entirely opt-in. If your patching process handles the fix, or you are rolling out kernel updates, you might not need them at all. The facts alone give you the visibility to track progress.&lt;/p&gt;

&lt;p&gt;Both modules ship with Hiera data defaults, so if you prefer to drive parameters through your hierarchy, see each module's README for examples.&lt;/p&gt;

&lt;h2&gt;
  
  
  The built-in module problem (Copy Fail only)
&lt;/h2&gt;

&lt;p&gt;There is a wrinkle with Copy Fail that Dirty Frag does not have. On most stock distribution kernels, &lt;code&gt;algif_aead&lt;/code&gt; is compiled directly into the kernel image as a built-in module. It is not a loadable &lt;code&gt;.ko&lt;/code&gt; file, so modprobe.d has no effect on it.&lt;/p&gt;

&lt;p&gt;For built-in modules, the mitigation is &lt;code&gt;initcall_blacklist=algif_aead_init&lt;/code&gt; on the kernel command line (via GRUB configuration). This prevents the module's init function from running on boot. It requires a reboot to take effect.&lt;/p&gt;

&lt;p&gt;The copy_fail module does not automate GRUB changes. Getting boot configuration wrong can render a system unbootable. You can manage GRUB with Puppet (using &lt;code&gt;file_line&lt;/code&gt; or &lt;code&gt;augeas&lt;/code&gt;), but that is out of scope for this module. The fact will report &lt;code&gt;initcall_blacklisted: true&lt;/code&gt; once the parameter is in place, and &lt;code&gt;reboot_required: true&lt;/code&gt; until the reboot completes.&lt;/p&gt;

&lt;p&gt;The &lt;code&gt;type&lt;/code&gt; field in the fact output tells you which mitigation path applies to each node:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;builtin&lt;/code&gt;&lt;/strong&gt;: needs &lt;code&gt;initcall_blacklist&lt;/code&gt;, modprobe.d has no effect&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;loadable&lt;/code&gt;&lt;/strong&gt;: the class and modprobe.d approach works&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;absent&lt;/code&gt;&lt;/strong&gt;: not vulnerable, nothing to do&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Force-unload a module right now (with care)
&lt;/h2&gt;

&lt;p&gt;Sometimes you cannot wait for a reboot. Both modules ship a Bolt task that removes the module from the running kernel immediately:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight shell"&gt;&lt;code&gt;bolt task run dirty_frag::unload &lt;span class="nv"&gt;module&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;esp4 &lt;span class="nt"&gt;--targets&lt;/span&gt; servers
bolt task run copy_fail::unload &lt;span class="nv"&gt;module&lt;/span&gt;&lt;span class="o"&gt;=&lt;/span&gt;algif_aead &lt;span class="nt"&gt;--targets&lt;/span&gt; servers
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;p&gt;A word of caution. Unloading a kernel module is not the same as flipping a config switch. If the module is actively in use (for example, &lt;code&gt;esp4&lt;/code&gt; underpins IPsec tunnels, &lt;code&gt;esp6&lt;/code&gt; handles IPv6 ESP, and &lt;code&gt;rxrpc&lt;/code&gt; is used by AFS), pulling it out from under a running service can drop connections, crash VPN tunnels, or cause dependent subsystems to fail. This is exactly why the Puppet classes do not do it automatically. The classes apply the block and wait for a reboot. The Bolt tasks give you the lever to pull when you have decided the tradeoff is worth it.&lt;/p&gt;

&lt;p&gt;Each task validates the module name, checks it is actually loaded, and gives you structured output. If the module is in use by another kernel subsystem and cannot be unloaded, the task will tell you, and you are back to "apply the block and schedule a reboot".&lt;/p&gt;

&lt;p&gt;Test on non-production nodes first. Know what the module is doing on that system before you yank it. If in doubt, the block-and-reboot path is always the safer option.&lt;/p&gt;

&lt;p&gt;The unload task only works for loadable modules. Built-in modules cannot be unloaded.&lt;/p&gt;

&lt;h2&gt;
  
  
  The workflow
&lt;/h2&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Add the modules to your Puppetfile.&lt;/strong&gt; Deploy. Done. Every node now reports its exposure.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Query PuppetDB&lt;/strong&gt; to see where you stand. One-liner, fleet-wide answer.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Decide on mitigation.&lt;/strong&gt; Apply the classes via Hiera for persistent blocking.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Use Tasks&lt;/strong&gt; to immediately unload modules on critical systems that cannot wait.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Watch the numbers drop.&lt;/strong&gt; The facts update every run. No manual tracking, no spreadsheets, no scheduled scans.&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;The whole approach is declarative. You tell Puppet what state you want, and it converges towards it. The facts keep reporting reality. The gap between the two is your exposure, and it is always visible.&lt;/p&gt;

&lt;h2&gt;
  
  
  Get started
&lt;/h2&gt;

&lt;p&gt;Both of these open source modules are available on GitHub:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;a href="https://github.com/albatrossflavour/dirty_frag" rel="noopener noreferrer"&gt;albatrossflavour/dirty_frag&lt;/a&gt; (Puppet 7/8, no dependencies)&lt;/li&gt;
&lt;li&gt;
&lt;a href="https://github.com/albatrossflavour/copy_fail" rel="noopener noreferrer"&gt;albatrossflavour/copy_fail&lt;/a&gt; (Puppet 7/8, no dependencies)&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Both modules have been built for RedHat, CentOS, Ubuntu, Debian, Amazon Linux, and SLES.&lt;/p&gt;

</description>
      <category>puppet</category>
      <category>vulnerabilities</category>
      <category>automation</category>
      <category>devops</category>
    </item>
    <item>
      <title>Puppetlabs Modules Roundup – April 2026</title>
      <dc:creator>Jason St-Cyr</dc:creator>
      <pubDate>Thu, 07 May 2026 16:31:43 +0000</pubDate>
      <link>https://dev.to/puppet/puppetlabs-modules-roundup-april-2026-358</link>
      <guid>https://dev.to/puppet/puppetlabs-modules-roundup-april-2026-358</guid>
      <description>&lt;p&gt;In April 2026, the Puppetlabs module lineup saw 8 Puppetlabs module releases, with the most notable updates collected here for a quick review.&lt;/p&gt;

&lt;p&gt;The overall pattern in these releases was event forwarding enhancements and security and compliance platform updates, which makes this month’s roundup a useful quick scan for teams planning upgrades or routine maintenance.&lt;/p&gt;

&lt;h2&gt;
  
  
  Highlighted Updates
&lt;/h2&gt;

&lt;h3&gt;
  
  
  Event Forwarding Enhancements
&lt;/h3&gt;

&lt;p&gt;Coordinated updates to Splunk HEC and PE Event Forwarding modules add support for orchestrator_plan event type, improving visibility into Puppet orchestrator activities with enhanced filtering and integration capabilities.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Affected modules: splunk_hec, pe_event_forwarding.&lt;/li&gt;
&lt;/ul&gt;

&lt;h3&gt;
  
  
  Security and Compliance Platform Updates
&lt;/h3&gt;

&lt;p&gt;Comply and Comply Admin modules received coordinated releases, advancing the Security Compliance Management platform capabilities.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Affected modules: comply, complyadm.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  What Updates Happened to Puppetlabs Modules in April 2026?
&lt;/h2&gt;

&lt;p&gt;The following is an alphabetical listing of modules which received updates in April 2026. If a module had multiple versions released, the updates are collected together, numbered with the "latest" version available.&lt;/p&gt;




&lt;h3&gt;
  
  
  cd4peadm 5.15.0
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-04-28 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/cd4peadm" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;A few highlights from this release:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Added a Hiera configuration option, external_webhook_url, that allows you to set the webhook URL that Continuous Delivery sends to your VCS provider. This is useful if you are using a proxy between your VCS and CD.&lt;/li&gt;
&lt;li&gt;Added an idle timeout to the CD console that logs users out after 30 minutes. Configure this using the Hiera option, web_session_idle_timeout_mins.&lt;/li&gt;
&lt;li&gt;Added CSRF protection to the DeleteUserAccount and SetSuperUser endpoints by restricting them to POST requests and validating CSRF tokens issued at login and expired on logout.&lt;/li&gt;
&lt;li&gt;20 CVEs addressed.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Check the official &lt;a href="https://help.puppet.com/cdpe/current/Content/UserGuide/CDPE/ReleaseNotes/cd_release_notes.htm#Version5150" rel="noopener noreferrer"&gt;release notes for cd4peadm 5.15.0&lt;/a&gt; for the full details.&lt;/p&gt;




&lt;h3&gt;
  
  
  comply 3.7.1
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-04-28 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/comply" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;A few highlights from this release:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Fixed a stale image which resulted in checksum and benchmark issues upon install.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;CVE-2026-33815, CVE-2026-33816.&lt;/strong&gt; Updated gorm.io to v5.9.2 to address these vulnerabilities.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Check the official &lt;a href="https://help.puppet.com/scm/current/Content/UserGuide/SCM/Release_notes/release_notes.htm#SecurityComplianceManagement371" rel="noopener noreferrer"&gt;release notes for comply 3.7.1&lt;/a&gt; for the full details.&lt;/p&gt;




&lt;h3&gt;
  
  
  complyadm 3.7.1
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-04-28 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/complyadm" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;A few highlights from this release:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Fixed a stale image which resulted in checksum and benchmark issues upon install.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;CVE-2026-33815, CVE-2026-33816.&lt;/strong&gt; Updated gorm.io to v5.9.2 to address these vulnerabilities.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Check the official &lt;a href="https://help.puppet.com/scm/current/Content/UserGuide/SCM/Release_notes/release_notes.htm#SecurityComplianceManagement371" rel="noopener noreferrer"&gt;release notes for complyadm 3.7.1&lt;/a&gt; for the full details.&lt;/p&gt;




&lt;h3&gt;
  
  
  lvm 4.0.1
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-04-30 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/lvm" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release addresses two bug fixes. A race condition where &lt;code&gt;lvcreate&lt;/code&gt; returns before udev finishes processing device-add events could cause a subsequent &lt;code&gt;filesystem&lt;/code&gt; resource targeting the same logical volume to fail with "device or resource busy" — the fix calls &lt;code&gt;udevadm settle&lt;/code&gt; after a successful &lt;code&gt;lvcreate&lt;/code&gt;. The release also corrects an AIX-specific issue where boolean filesystem parameters such as &lt;code&gt;isnapshot&lt;/code&gt; were passed to &lt;code&gt;crfs&lt;/code&gt; as &lt;code&gt;true&lt;/code&gt;/&lt;code&gt;false&lt;/code&gt; instead of the required &lt;code&gt;yes&lt;/code&gt;/&lt;code&gt;no&lt;/code&gt;, causing the command to reject them outright.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;(MODULES-11756) Wait for udev to settle after lvcreate &lt;a href="https://github.com/puppetlabs/puppetlabs-lvm/pull/380" rel="noopener noreferrer"&gt;#380&lt;/a&gt; (&lt;a href="https://github.com/imaqsood" rel="noopener noreferrer"&gt;imaqsood&lt;/a&gt;)&lt;/li&gt;
&lt;li&gt;(MODULES-11788) Pass converted boolean parameter &lt;a href="https://github.com/puppetlabs/puppetlabs-lvm/pull/379" rel="noopener noreferrer"&gt;#379&lt;/a&gt; (&lt;a href="https://github.com/joshcooper" rel="noopener noreferrer"&gt;joshcooper&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  pe_event_forwarding 2.3.0
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-04-09 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/pe_event_forwarding" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release adds orchestrator plan-job collection controls and fixes duplicate forwarding behavior in job collection.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Added plan job data collection from the &lt;code&gt;orchestrator/v1/plan_jobs&lt;/code&gt; API, with progress tracked in a dedicated &lt;code&gt;pe_event_forwarding_plan_index.yaml&lt;/code&gt; state file.&lt;/li&gt;
&lt;li&gt;Added the &lt;code&gt;pe_event_forwarding::skip_plans&lt;/code&gt; parameter to disable plan job collection when needed.&lt;/li&gt;
&lt;li&gt;Fixed &lt;code&gt;get_jobs&lt;/code&gt; behavior where the first page could return more records than newly available jobs, which could cause duplicate forwarded data.&lt;/li&gt;
&lt;li&gt;Source attribution: &lt;a href="https://github.com/puppetlabs/puppetlabs-pe_event_forwarding/pull/137" rel="noopener noreferrer"&gt;(PIE-1683) Add support for collecting plan data #137&lt;/a&gt; (&lt;a href="https://github.com/coreymbe" rel="noopener noreferrer"&gt;coreymbe&lt;/a&gt;).&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  peadm 3.37.0
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-04-01 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/peadm" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release is a small change to add support for Puppet Enterprise 2025.10.0.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;(PE-43654) Add support for PE 2025.10.0 &lt;a href="https://github.com/puppetlabs/puppetlabs-peadm/pull/661" rel="noopener noreferrer"&gt;#661&lt;/a&gt; (&lt;a href="https://github.com/davidmalloncares" rel="noopener noreferrer"&gt;davidmalloncares&lt;/a&gt;)&lt;/li&gt;
&lt;/ul&gt;




&lt;h3&gt;
  
  
  sce_linux 2.6.1
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-04-06 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/sce_linux" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;A few highlights from this release:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Error message:&lt;/strong&gt; comparison of NilClass with  failed. Some users saw this error message and experienced catalog compilation failures when attempting to manage mounted file system options with SCE for Linux. The issue was caused by /etc/fstab files that did not have at least one comment line or blank line. The internal parser was updated to avoid the issue and help prevent compilation errors.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;sce_mount_info fact does not resolve to a value.&lt;/strong&gt; This issue is related to the Linux findmnt command, which is used to list all mounted file systems. The command, which supports different options depending on operating system, was failing on specific systems, resulting in a failure of the sce_mount_info fact. Now, if the findmnt command fails, warnings will be logged, and the /etc/fstab file will be parsed directly for mount information.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Error messages related to rsyslog.&lt;/strong&gt; Because the version of the rsyslog logging service used on Red Hat Enterprise Linux (RHEL) 9 differs from earlier versions, users of RHEL 9 sometimes see error messages like this: imjournal: open() failed for path: '/var/lib/rsyslog/imjournal.state.tmp': Operation not permitted These messages indicate a failure to load the rsyslog imjournal module. To accommodate the changes in rsyslog and avoid this error, the module loading syntax was updated in SCE for Linux.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;Check the official &lt;a href="https://help.puppet.com/sce/current/linux/scel_relnotes_261.htm" rel="noopener noreferrer"&gt;release notes for sce_linux 2.6.1&lt;/a&gt; for the full details.&lt;/p&gt;




&lt;h3&gt;
  
  
  splunk_hec 2.2.0
&lt;/h3&gt;

&lt;p&gt;📅 Latest release: 2026-04-09 (🌐 &lt;a href="https://forge.puppet.com/modules/puppetlabs/splunk_hec" rel="noopener noreferrer"&gt;View on the Forge&lt;/a&gt;)&lt;/p&gt;

&lt;p&gt;This release focuses on support for the &lt;code&gt;orchestrator_plan&lt;/code&gt; event type from the &lt;strong&gt;puppetlabs-pe_event_forwarding&lt;/strong&gt; module while also addressing &lt;code&gt;orchestrator_plan&lt;/code&gt; to index mappings in util_splunk_hec template.&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Added support for the &lt;code&gt;orchestrator_plan&lt;/code&gt; event type from the &lt;strong&gt;puppetlabs-pe_event_forwarding&lt;/strong&gt; module.&lt;/li&gt;
&lt;li&gt;Added &lt;code&gt;orchestrator_plan&lt;/code&gt; to index mappings in util_splunk_hec template.&lt;/li&gt;
&lt;li&gt;New PE Event Forwarding filter &lt;code&gt;orchestrator_plan_data_filter&lt;/code&gt; to allow filtering orchestrator plan event payloads.&lt;/li&gt;
&lt;li&gt;Module dependency updated to ensure &lt;code&gt;pe_event_forwarding&lt;/code&gt; v2.3.0+ is installed.&lt;/li&gt;
&lt;li&gt;Removed support for Debian platform and EOL operating system versions.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Until Next Time!
&lt;/h2&gt;

&lt;p&gt;That’s the full pass through the 8 Puppetlabs module releases from April 2026. The Forge links above are the quickest path to the underlying release details.&lt;/p&gt;

&lt;p&gt;If there is a part of the Puppetlabs ecosystem that would benefit from more context in future roundups, that feedback is worth sending along.&lt;/p&gt;

&lt;p&gt;Catch you in the next roundup for May 2026.&lt;/p&gt;

</description>
      <category>puppet</category>
    </item>
    <item>
      <title>Why GRC Should Matter to Every Developer, Not Just Compliance Teams</title>
      <dc:creator>neviarrawlinson</dc:creator>
      <pubDate>Wed, 06 May 2026 13:53:13 +0000</pubDate>
      <link>https://dev.to/neviarrawl_44fe25f50/why-grc-should-matter-to-every-developer-not-just-compliance-teams-24am</link>
      <guid>https://dev.to/neviarrawl_44fe25f50/why-grc-should-matter-to-every-developer-not-just-compliance-teams-24am</guid>
      <description>&lt;h2&gt;
  
  
  Why GRC Should Matter to Every Developer, Not Just Compliance Teams
&lt;/h2&gt;

&lt;p&gt;When most people hear "GRC" — &lt;a href="https://www.scrut.io/post/how-governance-aces-compliance-and-risk-management-in-the-grc-program" rel="noopener noreferrer"&gt;governance, risk management, and compliance&lt;/a&gt; — they think of legal teams, auditors, or cybersecurity experts. Rarely do they think of developers.&lt;/p&gt;

&lt;p&gt;But the truth is, GRC affects everyone who builds, ships, and maintains technology.&lt;/p&gt;

&lt;p&gt;Whether you realize it or not, the choices you make in your code, architecture, or workflows impact your organization's ability to stay secure, compliant, and trusted.&lt;/p&gt;

&lt;h2&gt;
  
  
  What is GRC Anyway?
&lt;/h2&gt;

&lt;p&gt;GRC stands for:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Governance:&lt;/strong&gt; Making sure decisions align with company goals and policies.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Risk Management:&lt;/strong&gt; Identifying and reducing potential threats to systems, data, and users.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Compliance:&lt;/strong&gt; Following the laws, regulations, and industry standards that apply to your work.&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;At its core, GRC is about &lt;strong&gt;protecting the business and its customers while enabling growth.&lt;/strong&gt;&lt;/p&gt;

&lt;p&gt;And guess who sits at the heart of building that growth? Developers and tech teams.&lt;/p&gt;

&lt;h2&gt;
  
  
  Why Developers Should Care
&lt;/h2&gt;

&lt;p&gt;Here’s why GRC should be part of every developer’s mindset:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Security starts in the code:&lt;/strong&gt; Secure coding practices directly affect risk management.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Documentation matters:&lt;/strong&gt; Process documentation makes audits and compliance checks smoother — and helps your team scale faster.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Tech debt can become risk debt:&lt;/strong&gt; Skipping best practices today can create serious governance and compliance issues tomorrow.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Customers expect trust:&lt;/strong&gt; Data breaches and compliance failures destroy trust. Good GRC practices protect it.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  How Developers Can Contribute to GRC
&lt;/h2&gt;

&lt;p&gt;You don't need to become a compliance officer overnight.&lt;/p&gt;

&lt;p&gt;Simple steps make a big difference:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Follow secure coding guidelines (like OWASP Top 10).&lt;/li&gt;
&lt;li&gt;Document your APIs, services, and system behaviors clearly.&lt;/li&gt;
&lt;li&gt;Keep dependencies up-to-date and monitor for vulnerabilities.&lt;/li&gt;
&lt;li&gt;Understand the compliance requirements that apply to your industry (HIPAA, GDPR, SOC 2, etc.).&lt;/li&gt;
&lt;li&gt;Speak up if you see a potential risk or issue — risk management is everyone's job.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Final Thoughts
&lt;/h2&gt;

&lt;p&gt;GRC is not just a checkbox for the legal team.&lt;/p&gt;

&lt;p&gt;It’s a shared responsibility — and one that smart developers embrace.&lt;/p&gt;

&lt;p&gt;When you understand governance, risk, and compliance, you become a more valuable teammate, a better builder, and a stronger protector of your organization’s future.&lt;/p&gt;

&lt;p&gt;Tech doesn’t exist in a vacuum. Neither does trust.&lt;/p&gt;

&lt;p&gt;Let’s build better, safer, more resilient systems — together.&lt;/p&gt;

</description>
      <category>grc</category>
      <category>infosec</category>
      <category>security</category>
      <category>puppet</category>
    </item>
    <item>
      <title>Puppet for System Administrators: A Comprehensive Guide</title>
      <dc:creator>Pranay Trivedi</dc:creator>
      <pubDate>Sat, 02 May 2026 06:30:14 +0000</pubDate>
      <link>https://dev.to/koenig_solutions/puppet-for-system-administrators-a-comprehensive-guide-58f2</link>
      <guid>https://dev.to/koenig_solutions/puppet-for-system-administrators-a-comprehensive-guide-58f2</guid>
      <description>&lt;h2&gt;
  
  
  Introduction
&lt;/h2&gt;

&lt;p&gt;In the ever-evolving world of IT, effective configuration management is vital for maintaining system integrity and optimizing workflow. &lt;strong&gt;Puppet&lt;/strong&gt; stands as a powerful automation tool designed specifically for system administrators. It empowers them to manage infrastructure as code, ensuring consistency, reliability, and reduced manual effort.&lt;/p&gt;

&lt;h2&gt;
  
  
  What is Puppet?
&lt;/h2&gt;

&lt;p&gt;Puppet is an open-source software configuration management tool that allows you to define the state of your infrastructure. It enables automation of system administration tasks across a network of devices. Here are key features of Puppet:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Declarative Language:&lt;/strong&gt; Puppet uses a simple, declarative language to define system states.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Resource Abstraction:&lt;/strong&gt; You can manage multiple resources like files, packages, and services without worrying about underlying system specifics.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Idempotency:&lt;/strong&gt; Puppet ensures that no matter how many times you apply a configuration, the final state remains unchanged.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Why Use Puppet?
&lt;/h2&gt;

&lt;p&gt;For system administrators, Puppet offers several compelling benefits:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Efficiency:&lt;/strong&gt; Automate repetitive tasks to save time.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Consistency:&lt;/strong&gt; Ensure uniform configurations across different systems.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Scalability:&lt;/strong&gt; Easily manage large numbers of servers across diverse environments.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Version Control:&lt;/strong&gt; Puppet allows tracking of changes, akin to software development, facilitating better rollback capabilities.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Getting Started with Puppet
&lt;/h2&gt;

&lt;p&gt;Before diving deeper into Puppet's capabilities, here are some &lt;strong&gt;practical tips&lt;/strong&gt; for getting started:&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Install Puppet:&lt;/strong&gt; Start by installing Puppet on your primary server, known as the Puppet Master. Nodes that will be managed can connect to this Master.

&lt;ul&gt;
&lt;li&gt;Use package managers like APT or YUM for Linux distributions.&lt;/li&gt;
&lt;li&gt;For Windows, enable using the MSI installer.&lt;/li&gt;
&lt;/ul&gt;
&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Create a Manifest:&lt;/strong&gt; A manifest is a file where you define the resources. Begin with a simple one:&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;file { '/tmp/example.txt':&lt;br&gt;
     ensure =&amp;gt; present,&lt;br&gt;
     content =&amp;gt; 'Hello, Puppet!',&lt;br&gt;
   }&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Test Your Configuration:&lt;/strong&gt; Use the agent on the client machines to test your Manifest to ensure it works as expected.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Utilize the Puppet Forge:&lt;/strong&gt; Access a repository of reusable modules and codes that can help you implement solutions faster.&lt;/li&gt;
&lt;/ol&gt;

&lt;h2&gt;
  
  
  Puppet Architecture
&lt;/h2&gt;

&lt;p&gt;Understanding Puppet’s architecture is crucial for effective management:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Puppet Master:&lt;/strong&gt; The central server that compiles configurations.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Puppet Agent:&lt;/strong&gt; The client installed on nodes that communicates with the Puppet Master to fetch configurations.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Catalog:&lt;/strong&gt; A document containing the desired state of a node, compiled by the Puppet Master.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Resource Types:&lt;/strong&gt; Define various system elements such as packages, services, or files.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Essential Puppet Commands
&lt;/h2&gt;

&lt;p&gt;Here are several commands every System Administrator should know:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;puppet agent -t&lt;/code&gt;:&lt;/strong&gt; To test the agent and apply configurations.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;puppet status&lt;/code&gt;:&lt;/strong&gt; To check the current status of agents.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;&lt;code&gt;puppet resource&lt;/code&gt;:&lt;/strong&gt; To query the current state of a resource.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Advanced Puppet Features
&lt;/h2&gt;

&lt;p&gt;Once you grasp the basics, consider exploring advanced features to enhance your Puppet skills:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;
&lt;strong&gt;Hiera Configuration:&lt;/strong&gt; Use Hiera for data separation and to manage environment-specific variables effectively.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Puppet Modules:&lt;/strong&gt; Organize shared code in modules to make your manifests cleaner and reusable.&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Environment Management:&lt;/strong&gt; Separate code into different environments to safely test changes before applying them in production.&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Continuous Learning: Puppet Certifications
&lt;/h2&gt;

&lt;p&gt;To deepen your Puppet knowledge, consider investing in training or certification. Platforms such as &lt;a href="https://www.koenig-solutions.com/puppet-certification-training?utm_source=devto&amp;amp;utm_medium=backlink&amp;amp;utm_campaign=web20-publisher" rel="noopener noreferrer"&gt;Puppet for System Administrators&lt;/a&gt; provide structured learning paths and official recognition of your skills. This can enhance your career prospects and solidify your expertise.&lt;/p&gt;

&lt;h2&gt;
  
  
  Conclusion
&lt;/h2&gt;

&lt;p&gt;Puppet is an invaluable tool for system administrators looking to streamline their workflows and improve system management. By harnessing Puppet’s capabilities, you can automate tasks, enforce configurations, and ultimately provide a more reliable IT service. Start small, practice regularly, and continuously seek to learn more as technology evolves.&lt;/p&gt;

</description>
      <category>puppet</category>
      <category>systemadministration</category>
      <category>configurationmanagement</category>
    </item>
  </channel>
</rss>
