+
+ Read the Docs
+ v: ${config.versions.current.slug}
+
+
+
+
+ ${renderLanguages(config)}
+ ${renderVersions(config)}
+ ${renderDownloads(config)}
+
+ On Read the Docs
+
+ Project Home
+
+
+ Builds
+
+
+ Downloads
+
+
+
+ Search
+
+
+
+
+
+
+ Hosted by Read the Docs
+
+
+
+ `;
+
+ // Inject the generated flyout into the body HTML element.
+ document.body.insertAdjacentHTML("beforeend", flyout);
+
+ // Trigger the Read the Docs Addons Search modal when clicking on the "Search docs" input from inside the flyout.
+ document
+ .querySelector("#flyout-search-form")
+ .addEventListener("focusin", () => {
+ const event = new CustomEvent("readthedocs-search-show");
+ document.dispatchEvent(event);
+ });
+ })
+}
+
+if (themeLanguageSelector || themeVersionSelector) {
+ function onSelectorSwitch(event) {
+ const option = event.target.selectedIndex;
+ const item = event.target.options[option];
+ window.location.href = item.dataset.url;
+ }
+
+ document.addEventListener("readthedocs-addons-data-ready", function (event) {
+ const config = event.detail.data();
+
+ const versionSwitch = document.querySelector(
+ "div.switch-menus > div.version-switch",
+ );
+ if (themeVersionSelector) {
+ let versions = config.versions.active;
+ if (config.versions.current.hidden || config.versions.current.type === "external") {
+ versions.unshift(config.versions.current);
+ }
+ const versionSelect = `
+
+ ${versions
+ .map(
+ (version) => `
+
+ ${version.slug}
+ `,
+ )
+ .join("\n")}
+
+ `;
+
+ versionSwitch.innerHTML = versionSelect;
+ versionSwitch.firstElementChild.addEventListener("change", onSelectorSwitch);
+ }
+
+ const languageSwitch = document.querySelector(
+ "div.switch-menus > div.language-switch",
+ );
+
+ if (themeLanguageSelector) {
+ if (config.projects.translations.length) {
+ // Add the current language to the options on the selector
+ let languages = config.projects.translations.concat(
+ config.projects.current,
+ );
+ languages = languages.sort((a, b) =>
+ a.language.name.localeCompare(b.language.name),
+ );
+
+ const languageSelect = `
+
+ ${languages
+ .map(
+ (language) => `
+
+ ${language.language.name}
+ `,
+ )
+ .join("\n")}
+
+ `;
+
+ languageSwitch.innerHTML = languageSelect;
+ languageSwitch.firstElementChild.addEventListener("change", onSelectorSwitch);
+ }
+ else {
+ languageSwitch.remove();
+ }
+ }
+ });
+}
+
+document.addEventListener("readthedocs-addons-data-ready", function (event) {
+ // Trigger the Read the Docs Addons Search modal when clicking on "Search docs" input from the topnav.
+ document
+ .querySelector("[role='search'] input")
+ .addEventListener("focusin", () => {
+ const event = new CustomEvent("readthedocs-search-show");
+ document.dispatchEvent(event);
+ });
+});
\ No newline at end of file
diff --git a/_static/language_data.js b/_static/language_data.js
new file mode 100644
index 00000000..367b8ed8
--- /dev/null
+++ b/_static/language_data.js
@@ -0,0 +1,199 @@
+/*
+ * language_data.js
+ * ~~~~~~~~~~~~~~~~
+ *
+ * This script contains the language-specific data used by searchtools.js,
+ * namely the list of stopwords, stemmer, scorer and splitter.
+ *
+ * :copyright: Copyright 2007-2024 by the Sphinx team, see AUTHORS.
+ * :license: BSD, see LICENSE for details.
+ *
+ */
+
+var stopwords = ["a", "and", "are", "as", "at", "be", "but", "by", "for", "if", "in", "into", "is", "it", "near", "no", "not", "of", "on", "or", "such", "that", "the", "their", "then", "there", "these", "they", "this", "to", "was", "will", "with"];
+
+
+/* Non-minified version is copied as a separate JS file, if available */
+
+/**
+ * Porter Stemmer
+ */
+var Stemmer = function() {
+
+ var step2list = {
+ ational: 'ate',
+ tional: 'tion',
+ enci: 'ence',
+ anci: 'ance',
+ izer: 'ize',
+ bli: 'ble',
+ alli: 'al',
+ entli: 'ent',
+ eli: 'e',
+ ousli: 'ous',
+ ization: 'ize',
+ ation: 'ate',
+ ator: 'ate',
+ alism: 'al',
+ iveness: 'ive',
+ fulness: 'ful',
+ ousness: 'ous',
+ aliti: 'al',
+ iviti: 'ive',
+ biliti: 'ble',
+ logi: 'log'
+ };
+
+ var step3list = {
+ icate: 'ic',
+ ative: '',
+ alize: 'al',
+ iciti: 'ic',
+ ical: 'ic',
+ ful: '',
+ ness: ''
+ };
+
+ var c = "[^aeiou]"; // consonant
+ var v = "[aeiouy]"; // vowel
+ var C = c + "[^aeiouy]*"; // consonant sequence
+ var V = v + "[aeiou]*"; // vowel sequence
+
+ var mgr0 = "^(" + C + ")?" + V + C; // [C]VC... is m>0
+ var meq1 = "^(" + C + ")?" + V + C + "(" + V + ")?$"; // [C]VC[V] is m=1
+ var mgr1 = "^(" + C + ")?" + V + C + V + C; // [C]VCVC... is m>1
+ var s_v = "^(" + C + ")?" + v; // vowel in stem
+
+ this.stemWord = function (w) {
+ var stem;
+ var suffix;
+ var firstch;
+ var origword = w;
+
+ if (w.length < 3)
+ return w;
+
+ var re;
+ var re2;
+ var re3;
+ var re4;
+
+ firstch = w.substr(0,1);
+ if (firstch == "y")
+ w = firstch.toUpperCase() + w.substr(1);
+
+ // Step 1a
+ re = /^(.+?)(ss|i)es$/;
+ re2 = /^(.+?)([^s])s$/;
+
+ if (re.test(w))
+ w = w.replace(re,"$1$2");
+ else if (re2.test(w))
+ w = w.replace(re2,"$1$2");
+
+ // Step 1b
+ re = /^(.+?)eed$/;
+ re2 = /^(.+?)(ed|ing)$/;
+ if (re.test(w)) {
+ var fp = re.exec(w);
+ re = new RegExp(mgr0);
+ if (re.test(fp[1])) {
+ re = /.$/;
+ w = w.replace(re,"");
+ }
+ }
+ else if (re2.test(w)) {
+ var fp = re2.exec(w);
+ stem = fp[1];
+ re2 = new RegExp(s_v);
+ if (re2.test(stem)) {
+ w = stem;
+ re2 = /(at|bl|iz)$/;
+ re3 = new RegExp("([^aeiouylsz])\\1$");
+ re4 = new RegExp("^" + C + v + "[^aeiouwxy]$");
+ if (re2.test(w))
+ w = w + "e";
+ else if (re3.test(w)) {
+ re = /.$/;
+ w = w.replace(re,"");
+ }
+ else if (re4.test(w))
+ w = w + "e";
+ }
+ }
+
+ // Step 1c
+ re = /^(.+?)y$/;
+ if (re.test(w)) {
+ var fp = re.exec(w);
+ stem = fp[1];
+ re = new RegExp(s_v);
+ if (re.test(stem))
+ w = stem + "i";
+ }
+
+ // Step 2
+ re = /^(.+?)(ational|tional|enci|anci|izer|bli|alli|entli|eli|ousli|ization|ation|ator|alism|iveness|fulness|ousness|aliti|iviti|biliti|logi)$/;
+ if (re.test(w)) {
+ var fp = re.exec(w);
+ stem = fp[1];
+ suffix = fp[2];
+ re = new RegExp(mgr0);
+ if (re.test(stem))
+ w = stem + step2list[suffix];
+ }
+
+ // Step 3
+ re = /^(.+?)(icate|ative|alize|iciti|ical|ful|ness)$/;
+ if (re.test(w)) {
+ var fp = re.exec(w);
+ stem = fp[1];
+ suffix = fp[2];
+ re = new RegExp(mgr0);
+ if (re.test(stem))
+ w = stem + step3list[suffix];
+ }
+
+ // Step 4
+ re = /^(.+?)(al|ance|ence|er|ic|able|ible|ant|ement|ment|ent|ou|ism|ate|iti|ous|ive|ize)$/;
+ re2 = /^(.+?)(s|t)(ion)$/;
+ if (re.test(w)) {
+ var fp = re.exec(w);
+ stem = fp[1];
+ re = new RegExp(mgr1);
+ if (re.test(stem))
+ w = stem;
+ }
+ else if (re2.test(w)) {
+ var fp = re2.exec(w);
+ stem = fp[1] + fp[2];
+ re2 = new RegExp(mgr1);
+ if (re2.test(stem))
+ w = stem;
+ }
+
+ // Step 5
+ re = /^(.+?)e$/;
+ if (re.test(w)) {
+ var fp = re.exec(w);
+ stem = fp[1];
+ re = new RegExp(mgr1);
+ re2 = new RegExp(meq1);
+ re3 = new RegExp("^" + C + v + "[^aeiouwxy]$");
+ if (re.test(stem) || (re2.test(stem) && !(re3.test(stem))))
+ w = stem;
+ }
+ re = /ll$/;
+ re2 = new RegExp(mgr1);
+ if (re.test(w) && re2.test(w)) {
+ re = /.$/;
+ w = w.replace(re,"");
+ }
+
+ // and turn initial Y back to y
+ if (firstch == "y")
+ w = firstch.toLowerCase() + w.substr(1);
+ return w;
+ }
+}
+
diff --git a/_static/minus.png b/_static/minus.png
new file mode 100644
index 00000000..d96755fd
Binary files /dev/null and b/_static/minus.png differ
diff --git a/_static/plus.png b/_static/plus.png
new file mode 100644
index 00000000..7107cec9
Binary files /dev/null and b/_static/plus.png differ
diff --git a/_static/pygments.css b/_static/pygments.css
new file mode 100644
index 00000000..84ab3030
--- /dev/null
+++ b/_static/pygments.css
@@ -0,0 +1,75 @@
+pre { line-height: 125%; }
+td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; }
+span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; }
+td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; }
+span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; }
+.highlight .hll { background-color: #ffffcc }
+.highlight { background: #f8f8f8; }
+.highlight .c { color: #3D7B7B; font-style: italic } /* Comment */
+.highlight .err { border: 1px solid #FF0000 } /* Error */
+.highlight .k { color: #008000; font-weight: bold } /* Keyword */
+.highlight .o { color: #666666 } /* Operator */
+.highlight .ch { color: #3D7B7B; font-style: italic } /* Comment.Hashbang */
+.highlight .cm { color: #3D7B7B; font-style: italic } /* Comment.Multiline */
+.highlight .cp { color: #9C6500 } /* Comment.Preproc */
+.highlight .cpf { color: #3D7B7B; font-style: italic } /* Comment.PreprocFile */
+.highlight .c1 { color: #3D7B7B; font-style: italic } /* Comment.Single */
+.highlight .cs { color: #3D7B7B; font-style: italic } /* Comment.Special */
+.highlight .gd { color: #A00000 } /* Generic.Deleted */
+.highlight .ge { font-style: italic } /* Generic.Emph */
+.highlight .ges { font-weight: bold; font-style: italic } /* Generic.EmphStrong */
+.highlight .gr { color: #E40000 } /* Generic.Error */
+.highlight .gh { color: #000080; font-weight: bold } /* Generic.Heading */
+.highlight .gi { color: #008400 } /* Generic.Inserted */
+.highlight .go { color: #717171 } /* Generic.Output */
+.highlight .gp { color: #000080; font-weight: bold } /* Generic.Prompt */
+.highlight .gs { font-weight: bold } /* Generic.Strong */
+.highlight .gu { color: #800080; font-weight: bold } /* Generic.Subheading */
+.highlight .gt { color: #0044DD } /* Generic.Traceback */
+.highlight .kc { color: #008000; font-weight: bold } /* Keyword.Constant */
+.highlight .kd { color: #008000; font-weight: bold } /* Keyword.Declaration */
+.highlight .kn { color: #008000; font-weight: bold } /* Keyword.Namespace */
+.highlight .kp { color: #008000 } /* Keyword.Pseudo */
+.highlight .kr { color: #008000; font-weight: bold } /* Keyword.Reserved */
+.highlight .kt { color: #B00040 } /* Keyword.Type */
+.highlight .m { color: #666666 } /* Literal.Number */
+.highlight .s { color: #BA2121 } /* Literal.String */
+.highlight .na { color: #687822 } /* Name.Attribute */
+.highlight .nb { color: #008000 } /* Name.Builtin */
+.highlight .nc { color: #0000FF; font-weight: bold } /* Name.Class */
+.highlight .no { color: #880000 } /* Name.Constant */
+.highlight .nd { color: #AA22FF } /* Name.Decorator */
+.highlight .ni { color: #717171; font-weight: bold } /* Name.Entity */
+.highlight .ne { color: #CB3F38; font-weight: bold } /* Name.Exception */
+.highlight .nf { color: #0000FF } /* Name.Function */
+.highlight .nl { color: #767600 } /* Name.Label */
+.highlight .nn { color: #0000FF; font-weight: bold } /* Name.Namespace */
+.highlight .nt { color: #008000; font-weight: bold } /* Name.Tag */
+.highlight .nv { color: #19177C } /* Name.Variable */
+.highlight .ow { color: #AA22FF; font-weight: bold } /* Operator.Word */
+.highlight .w { color: #bbbbbb } /* Text.Whitespace */
+.highlight .mb { color: #666666 } /* Literal.Number.Bin */
+.highlight .mf { color: #666666 } /* Literal.Number.Float */
+.highlight .mh { color: #666666 } /* Literal.Number.Hex */
+.highlight .mi { color: #666666 } /* Literal.Number.Integer */
+.highlight .mo { color: #666666 } /* Literal.Number.Oct */
+.highlight .sa { color: #BA2121 } /* Literal.String.Affix */
+.highlight .sb { color: #BA2121 } /* Literal.String.Backtick */
+.highlight .sc { color: #BA2121 } /* Literal.String.Char */
+.highlight .dl { color: #BA2121 } /* Literal.String.Delimiter */
+.highlight .sd { color: #BA2121; font-style: italic } /* Literal.String.Doc */
+.highlight .s2 { color: #BA2121 } /* Literal.String.Double */
+.highlight .se { color: #AA5D1F; font-weight: bold } /* Literal.String.Escape */
+.highlight .sh { color: #BA2121 } /* Literal.String.Heredoc */
+.highlight .si { color: #A45A77; font-weight: bold } /* Literal.String.Interpol */
+.highlight .sx { color: #008000 } /* Literal.String.Other */
+.highlight .sr { color: #A45A77 } /* Literal.String.Regex */
+.highlight .s1 { color: #BA2121 } /* Literal.String.Single */
+.highlight .ss { color: #19177C } /* Literal.String.Symbol */
+.highlight .bp { color: #008000 } /* Name.Builtin.Pseudo */
+.highlight .fm { color: #0000FF } /* Name.Function.Magic */
+.highlight .vc { color: #19177C } /* Name.Variable.Class */
+.highlight .vg { color: #19177C } /* Name.Variable.Global */
+.highlight .vi { color: #19177C } /* Name.Variable.Instance */
+.highlight .vm { color: #19177C } /* Name.Variable.Magic */
+.highlight .il { color: #666666 } /* Literal.Number.Integer.Long */
\ No newline at end of file
diff --git a/_static/searchtools.js b/_static/searchtools.js
new file mode 100644
index 00000000..b08d58c9
--- /dev/null
+++ b/_static/searchtools.js
@@ -0,0 +1,620 @@
+/*
+ * searchtools.js
+ * ~~~~~~~~~~~~~~~~
+ *
+ * Sphinx JavaScript utilities for the full-text search.
+ *
+ * :copyright: Copyright 2007-2024 by the Sphinx team, see AUTHORS.
+ * :license: BSD, see LICENSE for details.
+ *
+ */
+"use strict";
+
+/**
+ * Simple result scoring code.
+ */
+if (typeof Scorer === "undefined") {
+ var Scorer = {
+ // Implement the following function to further tweak the score for each result
+ // The function takes a result array [docname, title, anchor, descr, score, filename]
+ // and returns the new score.
+ /*
+ score: result => {
+ const [docname, title, anchor, descr, score, filename] = result
+ return score
+ },
+ */
+
+ // query matches the full name of an object
+ objNameMatch: 11,
+ // or matches in the last dotted part of the object name
+ objPartialMatch: 6,
+ // Additive scores depending on the priority of the object
+ objPrio: {
+ 0: 15, // used to be importantResults
+ 1: 5, // used to be objectResults
+ 2: -5, // used to be unimportantResults
+ },
+ // Used when the priority is not in the mapping.
+ objPrioDefault: 0,
+
+ // query found in title
+ title: 15,
+ partialTitle: 7,
+ // query found in terms
+ term: 5,
+ partialTerm: 2,
+ };
+}
+
+const _removeChildren = (element) => {
+ while (element && element.lastChild) element.removeChild(element.lastChild);
+};
+
+/**
+ * See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_Expressions#escaping
+ */
+const _escapeRegExp = (string) =>
+ string.replace(/[.*+\-?^${}()|[\]\\]/g, "\\$&"); // $& means the whole matched string
+
+const _displayItem = (item, searchTerms, highlightTerms) => {
+ const docBuilder = DOCUMENTATION_OPTIONS.BUILDER;
+ const docFileSuffix = DOCUMENTATION_OPTIONS.FILE_SUFFIX;
+ const docLinkSuffix = DOCUMENTATION_OPTIONS.LINK_SUFFIX;
+ const showSearchSummary = DOCUMENTATION_OPTIONS.SHOW_SEARCH_SUMMARY;
+ const contentRoot = document.documentElement.dataset.content_root;
+
+ const [docName, title, anchor, descr, score, _filename] = item;
+
+ let listItem = document.createElement("li");
+ let requestUrl;
+ let linkUrl;
+ if (docBuilder === "dirhtml") {
+ // dirhtml builder
+ let dirname = docName + "/";
+ if (dirname.match(/\/index\/$/))
+ dirname = dirname.substring(0, dirname.length - 6);
+ else if (dirname === "index/") dirname = "";
+ requestUrl = contentRoot + dirname;
+ linkUrl = requestUrl;
+ } else {
+ // normal html builders
+ requestUrl = contentRoot + docName + docFileSuffix;
+ linkUrl = docName + docLinkSuffix;
+ }
+ let linkEl = listItem.appendChild(document.createElement("a"));
+ linkEl.href = linkUrl + anchor;
+ linkEl.dataset.score = score;
+ linkEl.innerHTML = title;
+ if (descr) {
+ listItem.appendChild(document.createElement("span")).innerHTML =
+ " (" + descr + ")";
+ // highlight search terms in the description
+ if (SPHINX_HIGHLIGHT_ENABLED) // set in sphinx_highlight.js
+ highlightTerms.forEach((term) => _highlightText(listItem, term, "highlighted"));
+ }
+ else if (showSearchSummary)
+ fetch(requestUrl)
+ .then((responseData) => responseData.text())
+ .then((data) => {
+ if (data)
+ listItem.appendChild(
+ Search.makeSearchSummary(data, searchTerms, anchor)
+ );
+ // highlight search terms in the summary
+ if (SPHINX_HIGHLIGHT_ENABLED) // set in sphinx_highlight.js
+ highlightTerms.forEach((term) => _highlightText(listItem, term, "highlighted"));
+ });
+ Search.output.appendChild(listItem);
+};
+const _finishSearch = (resultCount) => {
+ Search.stopPulse();
+ Search.title.innerText = _("Search Results");
+ if (!resultCount)
+ Search.status.innerText = Documentation.gettext(
+ "Your search did not match any documents. Please make sure that all words are spelled correctly and that you've selected enough categories."
+ );
+ else
+ Search.status.innerText = _(
+ "Search finished, found ${resultCount} page(s) matching the search query."
+ ).replace('${resultCount}', resultCount);
+};
+const _displayNextItem = (
+ results,
+ resultCount,
+ searchTerms,
+ highlightTerms,
+) => {
+ // results left, load the summary and display it
+ // this is intended to be dynamic (don't sub resultsCount)
+ if (results.length) {
+ _displayItem(results.pop(), searchTerms, highlightTerms);
+ setTimeout(
+ () => _displayNextItem(results, resultCount, searchTerms, highlightTerms),
+ 5
+ );
+ }
+ // search finished, update title and status message
+ else _finishSearch(resultCount);
+};
+// Helper function used by query() to order search results.
+// Each input is an array of [docname, title, anchor, descr, score, filename].
+// Order the results by score (in opposite order of appearance, since the
+// `_displayNextItem` function uses pop() to retrieve items) and then alphabetically.
+const _orderResultsByScoreThenName = (a, b) => {
+ const leftScore = a[4];
+ const rightScore = b[4];
+ if (leftScore === rightScore) {
+ // same score: sort alphabetically
+ const leftTitle = a[1].toLowerCase();
+ const rightTitle = b[1].toLowerCase();
+ if (leftTitle === rightTitle) return 0;
+ return leftTitle > rightTitle ? -1 : 1; // inverted is intentional
+ }
+ return leftScore > rightScore ? 1 : -1;
+};
+
+/**
+ * Default splitQuery function. Can be overridden in ``sphinx.search`` with a
+ * custom function per language.
+ *
+ * The regular expression works by splitting the string on consecutive characters
+ * that are not Unicode letters, numbers, underscores, or emoji characters.
+ * This is the same as ``\W+`` in Python, preserving the surrogate pair area.
+ */
+if (typeof splitQuery === "undefined") {
+ var splitQuery = (query) => query
+ .split(/[^\p{Letter}\p{Number}_\p{Emoji_Presentation}]+/gu)
+ .filter(term => term) // remove remaining empty strings
+}
+
+/**
+ * Search Module
+ */
+const Search = {
+ _index: null,
+ _queued_query: null,
+ _pulse_status: -1,
+
+ htmlToText: (htmlString, anchor) => {
+ const htmlElement = new DOMParser().parseFromString(htmlString, 'text/html');
+ for (const removalQuery of [".headerlink", "script", "style"]) {
+ htmlElement.querySelectorAll(removalQuery).forEach((el) => { el.remove() });
+ }
+ if (anchor) {
+ const anchorContent = htmlElement.querySelector(`[role="main"] ${anchor}`);
+ if (anchorContent) return anchorContent.textContent;
+
+ console.warn(
+ `Anchored content block not found. Sphinx search tries to obtain it via DOM query '[role=main] ${anchor}'. Check your theme or template.`
+ );
+ }
+
+ // if anchor not specified or not found, fall back to main content
+ const docContent = htmlElement.querySelector('[role="main"]');
+ if (docContent) return docContent.textContent;
+
+ console.warn(
+ "Content block not found. Sphinx search tries to obtain it via DOM query '[role=main]'. Check your theme or template."
+ );
+ return "";
+ },
+
+ init: () => {
+ const query = new URLSearchParams(window.location.search).get("q");
+ document
+ .querySelectorAll('input[name="q"]')
+ .forEach((el) => (el.value = query));
+ if (query) Search.performSearch(query);
+ },
+
+ loadIndex: (url) =>
+ (document.body.appendChild(document.createElement("script")).src = url),
+
+ setIndex: (index) => {
+ Search._index = index;
+ if (Search._queued_query !== null) {
+ const query = Search._queued_query;
+ Search._queued_query = null;
+ Search.query(query);
+ }
+ },
+
+ hasIndex: () => Search._index !== null,
+
+ deferQuery: (query) => (Search._queued_query = query),
+
+ stopPulse: () => (Search._pulse_status = -1),
+
+ startPulse: () => {
+ if (Search._pulse_status >= 0) return;
+
+ const pulse = () => {
+ Search._pulse_status = (Search._pulse_status + 1) % 4;
+ Search.dots.innerText = ".".repeat(Search._pulse_status);
+ if (Search._pulse_status >= 0) window.setTimeout(pulse, 500);
+ };
+ pulse();
+ },
+
+ /**
+ * perform a search for something (or wait until index is loaded)
+ */
+ performSearch: (query) => {
+ // create the required interface elements
+ const searchText = document.createElement("h2");
+ searchText.textContent = _("Searching");
+ const searchSummary = document.createElement("p");
+ searchSummary.classList.add("search-summary");
+ searchSummary.innerText = "";
+ const searchList = document.createElement("ul");
+ searchList.classList.add("search");
+
+ const out = document.getElementById("search-results");
+ Search.title = out.appendChild(searchText);
+ Search.dots = Search.title.appendChild(document.createElement("span"));
+ Search.status = out.appendChild(searchSummary);
+ Search.output = out.appendChild(searchList);
+
+ const searchProgress = document.getElementById("search-progress");
+ // Some themes don't use the search progress node
+ if (searchProgress) {
+ searchProgress.innerText = _("Preparing search...");
+ }
+ Search.startPulse();
+
+ // index already loaded, the browser was quick!
+ if (Search.hasIndex()) Search.query(query);
+ else Search.deferQuery(query);
+ },
+
+ _parseQuery: (query) => {
+ // stem the search terms and add them to the correct list
+ const stemmer = new Stemmer();
+ const searchTerms = new Set();
+ const excludedTerms = new Set();
+ const highlightTerms = new Set();
+ const objectTerms = new Set(splitQuery(query.toLowerCase().trim()));
+ splitQuery(query.trim()).forEach((queryTerm) => {
+ const queryTermLower = queryTerm.toLowerCase();
+
+ // maybe skip this "word"
+ // stopwords array is from language_data.js
+ if (
+ stopwords.indexOf(queryTermLower) !== -1 ||
+ queryTerm.match(/^\d+$/)
+ )
+ return;
+
+ // stem the word
+ let word = stemmer.stemWord(queryTermLower);
+ // select the correct list
+ if (word[0] === "-") excludedTerms.add(word.substr(1));
+ else {
+ searchTerms.add(word);
+ highlightTerms.add(queryTermLower);
+ }
+ });
+
+ if (SPHINX_HIGHLIGHT_ENABLED) { // set in sphinx_highlight.js
+ localStorage.setItem("sphinx_highlight_terms", [...highlightTerms].join(" "))
+ }
+
+ // console.debug("SEARCH: searching for:");
+ // console.info("required: ", [...searchTerms]);
+ // console.info("excluded: ", [...excludedTerms]);
+
+ return [query, searchTerms, excludedTerms, highlightTerms, objectTerms];
+ },
+
+ /**
+ * execute search (requires search index to be loaded)
+ */
+ _performSearch: (query, searchTerms, excludedTerms, highlightTerms, objectTerms) => {
+ const filenames = Search._index.filenames;
+ const docNames = Search._index.docnames;
+ const titles = Search._index.titles;
+ const allTitles = Search._index.alltitles;
+ const indexEntries = Search._index.indexentries;
+
+ // Collect multiple result groups to be sorted separately and then ordered.
+ // Each is an array of [docname, title, anchor, descr, score, filename].
+ const normalResults = [];
+ const nonMainIndexResults = [];
+
+ _removeChildren(document.getElementById("search-progress"));
+
+ const queryLower = query.toLowerCase().trim();
+ for (const [title, foundTitles] of Object.entries(allTitles)) {
+ if (title.toLowerCase().trim().includes(queryLower) && (queryLower.length >= title.length/2)) {
+ for (const [file, id] of foundTitles) {
+ const score = Math.round(Scorer.title * queryLower.length / title.length);
+ const boost = titles[file] === title ? 1 : 0; // add a boost for document titles
+ normalResults.push([
+ docNames[file],
+ titles[file] !== title ? `${titles[file]} > ${title}` : title,
+ id !== null ? "#" + id : "",
+ null,
+ score + boost,
+ filenames[file],
+ ]);
+ }
+ }
+ }
+
+ // search for explicit entries in index directives
+ for (const [entry, foundEntries] of Object.entries(indexEntries)) {
+ if (entry.includes(queryLower) && (queryLower.length >= entry.length/2)) {
+ for (const [file, id, isMain] of foundEntries) {
+ const score = Math.round(100 * queryLower.length / entry.length);
+ const result = [
+ docNames[file],
+ titles[file],
+ id ? "#" + id : "",
+ null,
+ score,
+ filenames[file],
+ ];
+ if (isMain) {
+ normalResults.push(result);
+ } else {
+ nonMainIndexResults.push(result);
+ }
+ }
+ }
+ }
+
+ // lookup as object
+ objectTerms.forEach((term) =>
+ normalResults.push(...Search.performObjectSearch(term, objectTerms))
+ );
+
+ // lookup as search terms in fulltext
+ normalResults.push(...Search.performTermsSearch(searchTerms, excludedTerms));
+
+ // let the scorer override scores with a custom scoring function
+ if (Scorer.score) {
+ normalResults.forEach((item) => (item[4] = Scorer.score(item)));
+ nonMainIndexResults.forEach((item) => (item[4] = Scorer.score(item)));
+ }
+
+ // Sort each group of results by score and then alphabetically by name.
+ normalResults.sort(_orderResultsByScoreThenName);
+ nonMainIndexResults.sort(_orderResultsByScoreThenName);
+
+ // Combine the result groups in (reverse) order.
+ // Non-main index entries are typically arbitrary cross-references,
+ // so display them after other results.
+ let results = [...nonMainIndexResults, ...normalResults];
+
+ // remove duplicate search results
+ // note the reversing of results, so that in the case of duplicates, the highest-scoring entry is kept
+ let seen = new Set();
+ results = results.reverse().reduce((acc, result) => {
+ let resultStr = result.slice(0, 4).concat([result[5]]).map(v => String(v)).join(',');
+ if (!seen.has(resultStr)) {
+ acc.push(result);
+ seen.add(resultStr);
+ }
+ return acc;
+ }, []);
+
+ return results.reverse();
+ },
+
+ query: (query) => {
+ const [searchQuery, searchTerms, excludedTerms, highlightTerms, objectTerms] = Search._parseQuery(query);
+ const results = Search._performSearch(searchQuery, searchTerms, excludedTerms, highlightTerms, objectTerms);
+
+ // for debugging
+ //Search.lastresults = results.slice(); // a copy
+ // console.info("search results:", Search.lastresults);
+
+ // print the results
+ _displayNextItem(results, results.length, searchTerms, highlightTerms);
+ },
+
+ /**
+ * search for object names
+ */
+ performObjectSearch: (object, objectTerms) => {
+ const filenames = Search._index.filenames;
+ const docNames = Search._index.docnames;
+ const objects = Search._index.objects;
+ const objNames = Search._index.objnames;
+ const titles = Search._index.titles;
+
+ const results = [];
+
+ const objectSearchCallback = (prefix, match) => {
+ const name = match[4]
+ const fullname = (prefix ? prefix + "." : "") + name;
+ const fullnameLower = fullname.toLowerCase();
+ if (fullnameLower.indexOf(object) < 0) return;
+
+ let score = 0;
+ const parts = fullnameLower.split(".");
+
+ // check for different match types: exact matches of full name or
+ // "last name" (i.e. last dotted part)
+ if (fullnameLower === object || parts.slice(-1)[0] === object)
+ score += Scorer.objNameMatch;
+ else if (parts.slice(-1)[0].indexOf(object) > -1)
+ score += Scorer.objPartialMatch; // matches in last name
+
+ const objName = objNames[match[1]][2];
+ const title = titles[match[0]];
+
+ // If more than one term searched for, we require other words to be
+ // found in the name/title/description
+ const otherTerms = new Set(objectTerms);
+ otherTerms.delete(object);
+ if (otherTerms.size > 0) {
+ const haystack = `${prefix} ${name} ${objName} ${title}`.toLowerCase();
+ if (
+ [...otherTerms].some((otherTerm) => haystack.indexOf(otherTerm) < 0)
+ )
+ return;
+ }
+
+ let anchor = match[3];
+ if (anchor === "") anchor = fullname;
+ else if (anchor === "-") anchor = objNames[match[1]][1] + "-" + fullname;
+
+ const descr = objName + _(", in ") + title;
+
+ // add custom score for some objects according to scorer
+ if (Scorer.objPrio.hasOwnProperty(match[2]))
+ score += Scorer.objPrio[match[2]];
+ else score += Scorer.objPrioDefault;
+
+ results.push([
+ docNames[match[0]],
+ fullname,
+ "#" + anchor,
+ descr,
+ score,
+ filenames[match[0]],
+ ]);
+ };
+ Object.keys(objects).forEach((prefix) =>
+ objects[prefix].forEach((array) =>
+ objectSearchCallback(prefix, array)
+ )
+ );
+ return results;
+ },
+
+ /**
+ * search for full-text terms in the index
+ */
+ performTermsSearch: (searchTerms, excludedTerms) => {
+ // prepare search
+ const terms = Search._index.terms;
+ const titleTerms = Search._index.titleterms;
+ const filenames = Search._index.filenames;
+ const docNames = Search._index.docnames;
+ const titles = Search._index.titles;
+
+ const scoreMap = new Map();
+ const fileMap = new Map();
+
+ // perform the search on the required terms
+ searchTerms.forEach((word) => {
+ const files = [];
+ const arr = [
+ { files: terms[word], score: Scorer.term },
+ { files: titleTerms[word], score: Scorer.title },
+ ];
+ // add support for partial matches
+ if (word.length > 2) {
+ const escapedWord = _escapeRegExp(word);
+ if (!terms.hasOwnProperty(word)) {
+ Object.keys(terms).forEach((term) => {
+ if (term.match(escapedWord))
+ arr.push({ files: terms[term], score: Scorer.partialTerm });
+ });
+ }
+ if (!titleTerms.hasOwnProperty(word)) {
+ Object.keys(titleTerms).forEach((term) => {
+ if (term.match(escapedWord))
+ arr.push({ files: titleTerms[term], score: Scorer.partialTitle });
+ });
+ }
+ }
+
+ // no match but word was a required one
+ if (arr.every((record) => record.files === undefined)) return;
+
+ // found search word in contents
+ arr.forEach((record) => {
+ if (record.files === undefined) return;
+
+ let recordFiles = record.files;
+ if (recordFiles.length === undefined) recordFiles = [recordFiles];
+ files.push(...recordFiles);
+
+ // set score for the word in each file
+ recordFiles.forEach((file) => {
+ if (!scoreMap.has(file)) scoreMap.set(file, {});
+ scoreMap.get(file)[word] = record.score;
+ });
+ });
+
+ // create the mapping
+ files.forEach((file) => {
+ if (!fileMap.has(file)) fileMap.set(file, [word]);
+ else if (fileMap.get(file).indexOf(word) === -1) fileMap.get(file).push(word);
+ });
+ });
+
+ // now check if the files don't contain excluded terms
+ const results = [];
+ for (const [file, wordList] of fileMap) {
+ // check if all requirements are matched
+
+ // as search terms with length < 3 are discarded
+ const filteredTermCount = [...searchTerms].filter(
+ (term) => term.length > 2
+ ).length;
+ if (
+ wordList.length !== searchTerms.size &&
+ wordList.length !== filteredTermCount
+ )
+ continue;
+
+ // ensure that none of the excluded terms is in the search result
+ if (
+ [...excludedTerms].some(
+ (term) =>
+ terms[term] === file ||
+ titleTerms[term] === file ||
+ (terms[term] || []).includes(file) ||
+ (titleTerms[term] || []).includes(file)
+ )
+ )
+ break;
+
+ // select one (max) score for the file.
+ const score = Math.max(...wordList.map((w) => scoreMap.get(file)[w]));
+ // add result to the result list
+ results.push([
+ docNames[file],
+ titles[file],
+ "",
+ null,
+ score,
+ filenames[file],
+ ]);
+ }
+ return results;
+ },
+
+ /**
+ * helper function to return a node containing the
+ * search summary for a given text. keywords is a list
+ * of stemmed words.
+ */
+ makeSearchSummary: (htmlText, keywords, anchor) => {
+ const text = Search.htmlToText(htmlText, anchor);
+ if (text === "") return null;
+
+ const textLower = text.toLowerCase();
+ const actualStartPosition = [...keywords]
+ .map((k) => textLower.indexOf(k.toLowerCase()))
+ .filter((i) => i > -1)
+ .slice(-1)[0];
+ const startWithContext = Math.max(actualStartPosition - 120, 0);
+
+ const top = startWithContext === 0 ? "" : "...";
+ const tail = startWithContext + 240 < text.length ? "..." : "";
+
+ let summary = document.createElement("p");
+ summary.classList.add("context");
+ summary.textContent = top + text.substr(startWithContext, 240).trim() + tail;
+
+ return summary;
+ },
+};
+
+_ready(Search.init);
diff --git a/_static/sphinx_highlight.js b/_static/sphinx_highlight.js
new file mode 100644
index 00000000..8a96c69a
--- /dev/null
+++ b/_static/sphinx_highlight.js
@@ -0,0 +1,154 @@
+/* Highlighting utilities for Sphinx HTML documentation. */
+"use strict";
+
+const SPHINX_HIGHLIGHT_ENABLED = true
+
+/**
+ * highlight a given string on a node by wrapping it in
+ * span elements with the given class name.
+ */
+const _highlight = (node, addItems, text, className) => {
+ if (node.nodeType === Node.TEXT_NODE) {
+ const val = node.nodeValue;
+ const parent = node.parentNode;
+ const pos = val.toLowerCase().indexOf(text);
+ if (
+ pos >= 0 &&
+ !parent.classList.contains(className) &&
+ !parent.classList.contains("nohighlight")
+ ) {
+ let span;
+
+ const closestNode = parent.closest("body, svg, foreignObject");
+ const isInSVG = closestNode && closestNode.matches("svg");
+ if (isInSVG) {
+ span = document.createElementNS("http://www.w3.org/2000/svg", "tspan");
+ } else {
+ span = document.createElement("span");
+ span.classList.add(className);
+ }
+
+ span.appendChild(document.createTextNode(val.substr(pos, text.length)));
+ const rest = document.createTextNode(val.substr(pos + text.length));
+ parent.insertBefore(
+ span,
+ parent.insertBefore(
+ rest,
+ node.nextSibling
+ )
+ );
+ node.nodeValue = val.substr(0, pos);
+ /* There may be more occurrences of search term in this node. So call this
+ * function recursively on the remaining fragment.
+ */
+ _highlight(rest, addItems, text, className);
+
+ if (isInSVG) {
+ const rect = document.createElementNS(
+ "http://www.w3.org/2000/svg",
+ "rect"
+ );
+ const bbox = parent.getBBox();
+ rect.x.baseVal.value = bbox.x;
+ rect.y.baseVal.value = bbox.y;
+ rect.width.baseVal.value = bbox.width;
+ rect.height.baseVal.value = bbox.height;
+ rect.setAttribute("class", className);
+ addItems.push({ parent: parent, target: rect });
+ }
+ }
+ } else if (node.matches && !node.matches("button, select, textarea")) {
+ node.childNodes.forEach((el) => _highlight(el, addItems, text, className));
+ }
+};
+const _highlightText = (thisNode, text, className) => {
+ let addItems = [];
+ _highlight(thisNode, addItems, text, className);
+ addItems.forEach((obj) =>
+ obj.parent.insertAdjacentElement("beforebegin", obj.target)
+ );
+};
+
+/**
+ * Small JavaScript module for the documentation.
+ */
+const SphinxHighlight = {
+
+ /**
+ * highlight the search words provided in localstorage in the text
+ */
+ highlightSearchWords: () => {
+ if (!SPHINX_HIGHLIGHT_ENABLED) return; // bail if no highlight
+
+ // get and clear terms from localstorage
+ const url = new URL(window.location);
+ const highlight =
+ localStorage.getItem("sphinx_highlight_terms")
+ || url.searchParams.get("highlight")
+ || "";
+ localStorage.removeItem("sphinx_highlight_terms")
+ url.searchParams.delete("highlight");
+ window.history.replaceState({}, "", url);
+
+ // get individual terms from highlight string
+ const terms = highlight.toLowerCase().split(/\s+/).filter(x => x);
+ if (terms.length === 0) return; // nothing to do
+
+ // There should never be more than one element matching "div.body"
+ const divBody = document.querySelectorAll("div.body");
+ const body = divBody.length ? divBody[0] : document.querySelector("body");
+ window.setTimeout(() => {
+ terms.forEach((term) => _highlightText(body, term, "highlighted"));
+ }, 10);
+
+ const searchBox = document.getElementById("searchbox");
+ if (searchBox === null) return;
+ searchBox.appendChild(
+ document
+ .createRange()
+ .createContextualFragment(
+ '
' +
+ '' +
+ _("Hide Search Matches") +
+ "
"
+ )
+ );
+ },
+
+ /**
+ * helper function to hide the search marks again
+ */
+ hideSearchWords: () => {
+ document
+ .querySelectorAll("#searchbox .highlight-link")
+ .forEach((el) => el.remove());
+ document
+ .querySelectorAll("span.highlighted")
+ .forEach((el) => el.classList.remove("highlighted"));
+ localStorage.removeItem("sphinx_highlight_terms")
+ },
+
+ initEscapeListener: () => {
+ // only install a listener if it is really needed
+ if (!DOCUMENTATION_OPTIONS.ENABLE_SEARCH_SHORTCUTS) return;
+
+ document.addEventListener("keydown", (event) => {
+ // bail for input elements
+ if (BLACKLISTED_KEY_CONTROL_ELEMENTS.has(document.activeElement.tagName)) return;
+ // bail with special keys
+ if (event.shiftKey || event.altKey || event.ctrlKey || event.metaKey) return;
+ if (DOCUMENTATION_OPTIONS.ENABLE_SEARCH_SHORTCUTS && (event.key === "Escape")) {
+ SphinxHighlight.hideSearchWords();
+ event.preventDefault();
+ }
+ });
+ },
+};
+
+_ready(() => {
+ /* Do not call highlightSearchWords() when we are on the search page.
+ * It will highlight words from the *previous* search query.
+ */
+ if (typeof Search === "undefined") SphinxHighlight.highlightSearchWords();
+ SphinxHighlight.initEscapeListener();
+});
diff --git a/api.html b/api.html
new file mode 100644
index 00000000..3e80f8dc
--- /dev/null
+++ b/api.html
@@ -0,0 +1,117 @@
+
+
+
+
+
+
+
+
+
API — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+API
+Auto-harvested goodness, coming soon.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/badges.html b/badges.html
new file mode 100644
index 00000000..5d3d6245
--- /dev/null
+++ b/badges.html
@@ -0,0 +1,113 @@
+
+
+
+
+
+
+
+
+
Badges — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+Badges
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/contributing_to_doc.html b/contributing_to_doc.html
new file mode 100644
index 00000000..df8bccaf
--- /dev/null
+++ b/contributing_to_doc.html
@@ -0,0 +1,154 @@
+
+
+
+
+
+
+
+
+
Documentation-Documentation — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+Documentation-Documentation
+Welcome to fre-cli
’s Documentation-documentation- where we document how the documentation is
+documented
+
+How to Contribute to fre-cli
’s documentation
+
+fork and poke at the settings
+
+Fork fre-cli
on github
+On github, navigate to your fre-cli
fork, and click “settings”
+In “settings”, click “pages”
+In “pages”, under “build and deployment”, make sure “source” is set to “Deploy from a branch”
+Under that, find “Branch”, make sure the branch selected is gh-pages
+The branch gh-pages
is “automagic”- i.e. do not change anything about it nor create a new one,
+nor interact with anything in that branch directly
+
+
+
+enable workflows for your fork
+note: this step may depend on user-specific settings!
+
+Back on top where you found “settings”, find and click “actions” to the left
+Enable running the workflow actions assoc with the fre-cli
repo under .github/workflows
+
+
+
+run your fork’s first workflow
+
+The documentation builds as the last steps to create_test_conda_env.yml
when theres a push to main
+To get your first workflow run on your fork, comment out the github.ref == ‘refs/heads/main’
bit
+so that it runs when you push to any branch, and make a small, trivial, commit somewhere to your
+remote fork
+You should be able to find the deployed webpage from a successful workflow at
+https://your_username.github.io/fre-cli (if you did not change the fork’s name from fre-cli
, that is)
+If you’re only editing docs, you can make the turn-around time on your workflow ~3 min faster by
+commenting-out the pylint
and pytest
steps in create_test_conda_env.yml
, and disable running the
+build_conda.yml
workflow
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/for-developers.html b/for-developers.html
new file mode 100644
index 00000000..49459ef7
--- /dev/null
+++ b/for-developers.html
@@ -0,0 +1,231 @@
+
+
+
+
+
+
+
+
+
For developers — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+For developers
+Developers are free to use the user guide above to familiarize with the CLI and save time from
+having to install any dependencies, but development within a Conda environment is heavily
+recommended regardless.
+Gain access to the repository with git clone --recursive git@github.com:NOAA-GFDL/fre-cli.git
or your fork’s
+link (recommended) and an SSH RSA key. Once inside the repository, developers can test local changes
+by running a pip install .
inside of the root directory to install the fre-cli
package locally
+with the newest local changes. Test as a normal user would use the CLI.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.freapp.html b/fre.app.freapp.html
new file mode 100644
index 00000000..af7dc606
--- /dev/null
+++ b/fre.app.freapp.html
@@ -0,0 +1,112 @@
+
+
+
+
+
+
+
+
+
fre.app.freapp module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.app.freapp module
+fre app calls
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.generate_time_averages.cdoTimeAverager.html b/fre.app.generate_time_averages.cdoTimeAverager.html
new file mode 100644
index 00000000..6b0e016a
--- /dev/null
+++ b/fre.app.generate_time_averages.cdoTimeAverager.html
@@ -0,0 +1,126 @@
+
+
+
+
+
+
+
+
+
fre.app.generate_time_averages.cdoTimeAverager module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+ fre.app.generate_time_averages.cdoTimeAverager module
+
+ View page source
+
+
+
+
+
+
+
+
+fre.app.generate_time_averages.cdoTimeAverager module
+class using (mostly) cdo functions for time-averages
+
+
+class fre.app.generate_time_averages.cdoTimeAverager. cdoTimeAverager ( pkg , var , unwgt , avg_type , stddev_type )
+Bases: timeAverager
+class inheriting from abstract base class timeAverager
+generates time-averages using cdo (mostly, see weighted approach)
+
+
+generate_timavg ( infile = None , outfile = None )
+use cdo package routines via python bindings
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.generate_time_averages.frenctoolsTimeAverager.html b/fre.app.generate_time_averages.frenctoolsTimeAverager.html
new file mode 100644
index 00000000..afad989a
--- /dev/null
+++ b/fre.app.generate_time_averages.frenctoolsTimeAverager.html
@@ -0,0 +1,126 @@
+
+
+
+
+
+
+
+
+
fre.app.generate_time_averages.frenctoolsTimeAverager module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+ fre.app.generate_time_averages.frenctoolsTimeAverager module
+
+ View page source
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.generate_time_averages.frepytoolsTimeAverager.html b/fre.app.generate_time_averages.frepytoolsTimeAverager.html
new file mode 100644
index 00000000..b15a05d2
--- /dev/null
+++ b/fre.app.generate_time_averages.frepytoolsTimeAverager.html
@@ -0,0 +1,128 @@
+
+
+
+
+
+
+
+
+
fre.app.generate_time_averages.frepytoolsTimeAverager module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+ fre.app.generate_time_averages.frepytoolsTimeAverager module
+
+ View page source
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.generate_time_averages.generate_time_averages.html b/fre.app.generate_time_averages.generate_time_averages.html
new file mode 100644
index 00000000..ac9f2bf8
--- /dev/null
+++ b/fre.app.generate_time_averages.generate_time_averages.html
@@ -0,0 +1,118 @@
+
+
+
+
+
+
+
+
+
fre.app.generate_time_averages.generate_time_averages module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+ fre.app.generate_time_averages.generate_time_averages module
+
+ View page source
+
+
+
+
+
+
+
+
+fre.app.generate_time_averages.generate_time_averages module
+tools for generating time averages from various packages
+
+
+fre.app.generate_time_averages.generate_time_averages. generate_time_average ( infile = None , outfile = None , pkg = None , var = None , unwgt = False , avg_type = None , stddev_type = None )
+steering function to various averaging functions above
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.generate_time_averages.html b/fre.app.generate_time_averages.html
new file mode 100644
index 00000000..cfc4620b
--- /dev/null
+++ b/fre.app.generate_time_averages.html
@@ -0,0 +1,195 @@
+
+
+
+
+
+
+
+
+
fre.app.generate_time_averages package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.app.generate_time_averages package
+
+
+
+Module contents
+required for generate_time_averages module import functionality
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.generate_time_averages.tests.html b/fre.app.generate_time_averages.tests.html
new file mode 100644
index 00000000..9f89337f
--- /dev/null
+++ b/fre.app.generate_time_averages.tests.html
@@ -0,0 +1,142 @@
+
+
+
+
+
+
+
+
+
fre.app.generate_time_averages.tests package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.app.generate_time_averages.tests package
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.generate_time_averages.tests.test_generate_time_averages.html b/fre.app.generate_time_averages.tests.test_generate_time_averages.html
new file mode 100644
index 00000000..539fc487
--- /dev/null
+++ b/fre.app.generate_time_averages.tests.test_generate_time_averages.html
@@ -0,0 +1,220 @@
+
+
+
+
+
+
+
+
+
fre.app.generate_time_averages.tests.test_generate_time_averages module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+ fre.app.generate_time_averages.tests.test_generate_time_averages module
+
+ View page source
+
+
+
+
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages module
+for testing fre app generate-time-averages
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. run_avgtype_pkg_calculations ( infile = None , outfile = None , pkg = None , avg_type = None , unwgt = None , stddev_type = None )
+test-harness function, called by other test functions.
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_cdo_time_avgs ( )
+generates a weighted time averaged file using cdo
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_cdo_time_unwgt_avgs ( )
+generates an unweighted time averaged file using cdo
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_cdo_time_unwgt_stddevs ( )
+generates a time averaged file using cdo
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_compare_cdo_to_fre_nctools ( )
+compares cdo pkg answer to fre_nctools pkg answer
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_compare_fre_cli_to_cdo ( )
+compares fre_cli pkg answer to cdo pkg answer
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_compare_fre_cli_to_fre_nctools ( )
+compares fre_cli pkg answer to fre_nctools pkg answer
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_compare_unwgt_fre_cli_to_unwgt_cdo ( )
+compares fre_cli pkg answer to cdo pkg answer
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_fre_cli_time_avgs ( )
+generates a time averaged file using fre_cli’s version
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_fre_cli_time_avgs_stddevs ( )
+generates a time averaged file using fre_cli’s version
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_fre_cli_time_unwgt_avgs ( )
+generates a time averaged file using fre_cli’s version
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_fre_cli_time_unwgt_avgs_stddevs ( )
+generates a time averaged file using fre_cli’s version
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_monthly_cdo_time_unwgt_avgs ( )
+generates an unweighted monthly time averaged file using cdo
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_monthly_cdo_time_unwgt_stddevs ( )
+generates a monthly time averaged file using cdo
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_seasonal_cdo_time_unwgt_avgs ( )
+generates an unweighted seasonal time averaged file using cdo
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_seasonal_cdo_time_unwgt_stddevs ( )
+generates a seasonal time averaged file using cdo
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_time_avg_file_dir_exists ( )
+look for input test file directory
+
+
+
+
+fre.app.generate_time_averages.tests.test_generate_time_averages. test_time_avg_input_file_exists ( )
+look for input test file
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.generate_time_averages.timeAverager.html b/fre.app.generate_time_averages.timeAverager.html
new file mode 100644
index 00000000..623474d2
--- /dev/null
+++ b/fre.app.generate_time_averages.timeAverager.html
@@ -0,0 +1,158 @@
+
+
+
+
+
+
+
+
+
fre.app.generate_time_averages.timeAverager module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.app.generate_time_averages.timeAverager module
+core class structure for this module.
+
+
+class fre.app.generate_time_averages.timeAverager. timeAverager ( pkg , var , unwgt , avg_type , stddev_type )
+Bases: object
+abstract base class for generating time averages + related statistical quantities
+this class must be inherited by another for functionality.
+
+
+avg_type : str
+
+
+
+
+generate_timavg ( infile = None , outfile = None )
+# this is a hint: this is to be defined by classes inheriting from the abstract one
+this function is never to be fully defined here by design.
+
+
+
+
+pkg : str
+
+
+
+
+stddev_type : str
+
+
+
+
+unwgt : bool
+
+
+
+
+var : str
+
+
+
+
+var_has_time_units ( an_nc_var = None )
+checks if variable’s units are of time
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.html b/fre.app.html
new file mode 100644
index 00000000..067deb2c
--- /dev/null
+++ b/fre.app.html
@@ -0,0 +1,195 @@
+
+
+
+
+
+
+
+
+
fre.app package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.mask_atmos_plevel.html b/fre.app.mask_atmos_plevel.html
new file mode 100644
index 00000000..b153acee
--- /dev/null
+++ b/fre.app.mask_atmos_plevel.html
@@ -0,0 +1,154 @@
+
+
+
+
+
+
+
+
+
fre.app.mask_atmos_plevel module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.app.mask_atmos_plevel module
+This script contains the refineDiags that produce data at the same
+frequency as the input data (no reduction) such as surface albedo,
+masking fields,…
+It can accept any file and will only compute refineDiags in fields
+are present.
+
+
+fre.app.mask_atmos_plevel. mask_field_above_surface_pressure ( ds , var , ds_ps )
+mask data with pressure larger than surface pressure
+
+
+
+
+fre.app.mask_atmos_plevel. post_write ( filename , var_with_bounds , bounds_variables )
+fix a posteriori attributes that xarray.to_netcdf
+did not do properly using low level netcdf lib
+
+
+
+
+fre.app.mask_atmos_plevel. preprocess ( ds )
+add needs_atmos_masking attribute if var ends with _unmsk
+
+
+
+
+fre.app.mask_atmos_plevel. pressure_coordinate ( ds , varname )
+check if dataArray has pressure coordinate fitting requirements
+and return it
+
+
+
+
+fre.app.mask_atmos_plevel. set_netcdf_encoding ( ds , pressure_vars )
+set preferred options for netcdf encoding
+
+
+
+
+fre.app.mask_atmos_plevel. write_dataset ( ds , template , outfile )
+prepare the dataset and dump into netcdf file
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.regrid_xy.html b/fre.app.regrid_xy.html
new file mode 100644
index 00000000..f503538a
--- /dev/null
+++ b/fre.app.regrid_xy.html
@@ -0,0 +1,136 @@
+
+
+
+
+
+
+
+
+
fre.app.regrid_xy package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.app.regrid_xy package
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.app.regrid_xy.regrid_xy.html b/fre.app.regrid_xy.regrid_xy.html
new file mode 100644
index 00000000..e2500cc7
--- /dev/null
+++ b/fre.app.regrid_xy.regrid_xy.html
@@ -0,0 +1,189 @@
+
+
+
+
+
+
+
+
+
fre.app.regrid_xy.regrid_xy module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.app.regrid_xy.regrid_xy module
+remaps scalar and/or vector fields. It is capable of remapping
+from a spherical grid onto a different one, e.g. spherical,
+or tripolar. By default, it does so using a conservative scheme.
+most valid input args to fregrid are valid in this script
+
+
+fre.app.regrid_xy.regrid_xy. check_interp_method ( nc_variable , interp_method )
+print warning if optional interp_method clashes with nc file attribute field, if present
+
+
+
+
+fre.app.regrid_xy.regrid_xy. check_per_component_settings ( component_list , rose_app_cfg )
+for a source file ref’d by multiple components check per-component
+settings for uniqueness. output list of bools of same length to check
+in componenet loop
+
+
+
+
+fre.app.regrid_xy.regrid_xy. freq_to_date_format ( iso_freq )
+Print legacy Bronx-like date template format given a frequency (ISO 8601 duration)
+
+
+
+
+fre.app.regrid_xy.regrid_xy. get_mosaic_file_name ( grid_spec_file , mosaic_type )
+read string from a numpy masked array WHY
+
+
+
+
+fre.app.regrid_xy.regrid_xy. get_mosaic_grid_file_name ( input_mosaic )
+get mosaic grid file name from NESTED numpy masked array WHY
+
+
+
+
+fre.app.regrid_xy.regrid_xy. main ( )
+steering, local test/debug
+
+
+
+
+fre.app.regrid_xy.regrid_xy. make_component_list ( config , source )
+make list of relevant component names where source file appears in sources
+
+
+
+
+fre.app.regrid_xy.regrid_xy. make_regrid_var_list ( target_file , interp_method = None )
+create default list of variables to be regridded within target file.
+
+
+
+
+fre.app.regrid_xy.regrid_xy. regrid_xy ( input_dir = None , output_dir = None , begin = None , tmp_dir = None , remap_dir = None , source = None , grid_spec = None , def_xy_interp = None )
+calls fre-nctools’ fregrid to regrid net cdf files
+
+
+
+
+fre.app.regrid_xy.regrid_xy. safe_rose_config_get ( config , section , field )
+read optional variables from rose configuration, and don’t error on None value
+
+
+
+
+fre.app.regrid_xy.regrid_xy. test_import ( )
+for quickly testing import within pytest
+
+
+
+
+fre.app.regrid_xy.regrid_xy. truncate_date ( date , freq )
+truncates iso freq to iso date time
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.catalog.frecatalog.html b/fre.catalog.frecatalog.html
new file mode 100644
index 00000000..e83223eb
--- /dev/null
+++ b/fre.catalog.frecatalog.html
@@ -0,0 +1,112 @@
+
+
+
+
+
+
+
+
+
fre.catalog.frecatalog module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.catalog.frecatalog module
+entry point for fre catalog subcommands
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.catalog.html b/fre.catalog.html
new file mode 100644
index 00000000..5452de2c
--- /dev/null
+++ b/fre.catalog.html
@@ -0,0 +1,140 @@
+
+
+
+
+
+
+
+
+
fre.catalog package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.catalog package
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.catalog.tests.html b/fre.catalog.tests.html
new file mode 100644
index 00000000..cf3120ef
--- /dev/null
+++ b/fre.catalog.tests.html
@@ -0,0 +1,125 @@
+
+
+
+
+
+
+
+
+
fre.catalog.tests package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.catalog.tests package
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.catalog.tests.test_fre_catalog.html b/fre.catalog.tests.test_fre_catalog.html
new file mode 100644
index 00000000..d99e9888
--- /dev/null
+++ b/fre.catalog.tests.test_fre_catalog.html
@@ -0,0 +1,116 @@
+
+
+
+
+
+
+
+
+
fre.catalog.tests.test_fre_catalog module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.catalog.tests.test_fre_catalog module
+
+
+fre.catalog.tests.test_fre_catalog. test_fre_catalog_import ( )
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.check.frecheck.html b/fre.check.frecheck.html
new file mode 100644
index 00000000..b0739648
--- /dev/null
+++ b/fre.check.frecheck.html
@@ -0,0 +1,112 @@
+
+
+
+
+
+
+
+
+
fre.check.frecheck module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.check.frecheck module
+fre check
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.check.frecheckexample.html b/fre.check.frecheckexample.html
new file mode 100644
index 00000000..4c9a485b
--- /dev/null
+++ b/fre.check.frecheckexample.html
@@ -0,0 +1,114 @@
+
+
+
+
+
+
+
+
+
fre.check.frecheckexample module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.check.frecheckexample module
+experimentation file for integrating one file’s functions into main prototype fre file
+authored by Bennett. Chang@ noaa. gov | bcc2761
+NOAA | GFDL
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.check.html b/fre.check.html
new file mode 100644
index 00000000..7081513f
--- /dev/null
+++ b/fre.check.html
@@ -0,0 +1,123 @@
+
+
+
+
+
+
+
+
+
fre.check package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.cmor.cmor_mixer.html b/fre.cmor.cmor_mixer.html
new file mode 100644
index 00000000..0f9a0ef1
--- /dev/null
+++ b/fre.cmor.cmor_mixer.html
@@ -0,0 +1,270 @@
+
+
+
+
+
+
+
+
+
fre.cmor.cmor_mixer module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.cmor.cmor_mixer module
+python module housing the metadata processing routines utilizing the cmor module, in addition to
+click API entry points
+see README.md for additional information on fre cmor run (cmor_mixer.py) usage
+
+
+fre.cmor.cmor_mixer. check_dataset_for_ocean_grid ( ds )
+checks netCDF4.Dataset ds for ocean grid origin, and throws an error if it finds one. accepts
+one argument. this function has no return.
+
+ds: netCDF4.Dataset object containing variables with associated dimensional information.
+
+
+
+
+
+fre.cmor.cmor_mixer. cmor_run_subtool ( indir = None , json_var_list = None , json_table_config = None , json_exp_config = None , outdir = None , opt_var_name = None )
+
+primary steering function for the cmor_mixer tool, i.e essentially main. Accepts six args:
+indir: string, directory containing netCDF files. keys specified in json_var_list are local variable names used for targeting specific files
+
+json_var_list: string, path pointing to a json file containing directory of key/value pairs. the keys are the “local” names used in the filename, and the
+values pointed to by those keys are strings representing the name of the
+variable contained in targeted files. the key and value are often the same,
+but it is not required.
+
+json_table_config: json file containing CMIP-compliant per-variable/metadata for specific MIP table. The MIP table can generally be identified by the specific
+filename (e.g. “Omon”)
+
+json_exp_config: json file containing metadata dictionary for CMORization. this metadata is effectively appended to the final output file’s header
+
+outdir: string, directory root that will contain the full output and output directory structure generated by the cmor module upon request.
+
+opt_var_name: string, optional, specify a variable name to specifically process only filenames matching that variable name. I.e., this string help target local_vars, not target_vars.
+
+
+
+
+
+
+
+
+fre.cmor.cmor_mixer. cmorize_target_var_files ( indir = None , target_var = None , local_var = None , iso_datetime_arr = None , name_of_set = None , json_exp_config = None , outdir = None , proj_table_vars = None , json_table_config = None )
+processes a target directory/file
+this routine is almost entirely exposed data movement before/after calling
+rewrite_netcdf_file_var it is also the most hopelessly opaque routine in this entire dang macro.
+this badboy right here accepts… lord help us… !!!NINE!!! arguments, NINE.
+
+indir: string, path to target directories containing netcdf files to cmorize
+target_var: string, name of variable inside the netcdf file to cmorize
+local_var: string, value of the variable name in the filename, right before the .nc
+
+extension. often identical to target_var but not always.
+
+
+iso_datetime_arr: list of strings, each one a unique ISO datetime string found in targeted netcdf filenames
+
+name_of_set: string, representing the post-processing component (GFDL convention) of the targeted files.
+
+
+
json_exp_config: see cmor_run_subtool arg desc
+outdir: string, path to output directory root to move the cmor module output to, including
+
+the whole directory structure
+
+
proj_table_vars: an opened json file object, read from json_table_config
+json_table_config: see cmor_run_subtool arg desc
+
+
+
+
+
+fre.cmor.cmor_mixer. copy_nc ( in_nc , out_nc )
+copy target input netcdf file in_nc to target out_nc. I have to think this is not a trivial copy
+operation, as if it were, using shutil’s copy would be sufficient. accepts two arguments
+
+in_nc: string, path to an input netcdf file we wish to copy
+out_nc: string, an output path to copy the targeted input netcdf file to
+
+
+
+
+
+fre.cmor.cmor_mixer. create_tmp_dir ( outdir )
+creates a tmp_dir based on targeted output directory root. returns the name of the tmp dir.
+accepts one argument:
+
+
+outdir: string, representing the final output directory root for the cmor modules netcdf file output. tmp_dir will be slightly different depending on the output directory
+targeted
+
+
+
+
+
+
+
+fre.cmor.cmor_mixer. get_iso_datetimes ( var_filenames , iso_datetime_arr = None )
+
+appends iso datetime strings found amongst filenames to iso_datetime_arr.
+var_filenames: non-empty list of strings representing filenames. some of which presumably contain datetime strings
+
+iso_datetime_arr: list of strings, empty or non-empty, representing datetimes found in var_filenames entries. the objet pointed to by the reference
+iso_datetime_arr is manipulated, and so need-not be returned
+
+
+
+
+
+
+
+
+fre.cmor.cmor_mixer. get_var_filenames ( indir , var_filenames = None , local_var = None )
+
+appends files ending in .nc located within indir to list var_filenames accepts three arguments indir: string, representing a path to a directory containing files ending in .nc extension
+var_filenames: list of strings, empty or non-empty, to append discovered filenames to. the
+
+object pointed to by the reference var_filenames is manipulated, and so need
+not be returned.
+
+local_var: string, optional, if not None, will be used for ruling out filename targets
+
+
+
+
+
+
+fre.cmor.cmor_mixer. get_vertical_dimension ( ds , target_var )
+determines the vertical dimensionality of target_var within netCDF4 Dataset ds. accepts two
+arguments and returns an object represnting the vertical dimensions assoc with the target_var.
+
+ds: netCDF4.Dataset object containing variables with associated dimensional information.
+target_var: string, representating a variable contained within the netCDF4.Dataset ds
+
+
+
+
+
+fre.cmor.cmor_mixer. rewrite_netcdf_file_var ( proj_table_vars = None , local_var = None , netcdf_file = None , target_var = None , json_exp_config = None , json_table_config = None )
+rewrite the input netcdf file nc_fl containing target_var in a CMIP-compliant manner.
+accepts six arguments, all required:
+
+proj_table_vars: json dictionary object, variable table read from json_table_config.
+local_var: string, variable name used for finding files locally containing target_var,
+
+this argument is often equal to target_var.
+
+
netcdf_file: string, representing path to intput netcdf file.
+target_var: string, representing the variable name attached to the data object in the netcdf file.
+json_exp_config: string, representing path to json configuration file holding metadata for appending to output
+
+this argument is most used for making sure the right grid label is getting attached to the right output
+
+
+json_table_config: string, representing path to json configuration file holding variable names for a given table. proj_table_vars is read from this file, but both are passed anyways.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.cmor.frecmor.html b/fre.cmor.frecmor.html
new file mode 100644
index 00000000..245e0aa9
--- /dev/null
+++ b/fre.cmor.frecmor.html
@@ -0,0 +1,112 @@
+
+
+
+
+
+
+
+
+
fre.cmor.frecmor module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.cmor.frecmor module
+fre cmor
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.cmor.html b/fre.cmor.html
new file mode 100644
index 00000000..af118a49
--- /dev/null
+++ b/fre.cmor.html
@@ -0,0 +1,161 @@
+
+
+
+
+
+
+
+
+
fre.cmor package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.cmor package
+
+
+
+Module contents
+for fre.cmor imports
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.cmor.tests.html b/fre.cmor.tests.html
new file mode 100644
index 00000000..76134ee1
--- /dev/null
+++ b/fre.cmor.tests.html
@@ -0,0 +1,133 @@
+
+
+
+
+
+
+
+
+
fre.cmor.tests package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.cmor.tests package
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.cmor.tests.test_cmor_run_subtool.html b/fre.cmor.tests.test_cmor_run_subtool.html
new file mode 100644
index 00000000..676a2178
--- /dev/null
+++ b/fre.cmor.tests.test_cmor_run_subtool.html
@@ -0,0 +1,169 @@
+
+
+
+
+
+
+
+
+
fre.cmor.tests.test_cmor_run_subtool module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.fre.html b/fre.fre.html
new file mode 100644
index 00000000..1c91a709
--- /dev/null
+++ b/fre.fre.html
@@ -0,0 +1,117 @@
+
+
+
+
+
+
+
+
+
fre.fre module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.fre module
+Main host file for FRE-CLI program scripts
+authored by Bennett. Chang@ noaa. gov | bcc2761
+NOAA | GFDL
+2023-2024
+principal click group for main/fre allows for subgroup functions to
+be called via this script. I.e. ‘fre’ is the entry point
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.html b/fre.html
new file mode 100644
index 00000000..08067329
--- /dev/null
+++ b/fre.html
@@ -0,0 +1,489 @@
+
+
+
+
+
+
+
+
+
fre package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.lazy_group.html b/fre.lazy_group.html
new file mode 100644
index 00000000..92613987
--- /dev/null
+++ b/fre.lazy_group.html
@@ -0,0 +1,134 @@
+
+
+
+
+
+
+
+
+
fre.lazy_group module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.lazy_group module
+for lazy-style loading of commands and subcommands while using click
+see https://click.palletsprojects.com/en/8.1.x/complex/
+
+
+class fre.lazy_group. LazyGroup ( * args , lazy_subcommands = None , ** kwargs )
+Bases: Group
+class defining lazygroup command/subcommand loading
+
+
+get_command ( ctx , cmd_name )
+Given a context and a command name, this returns a
+Command
object if it exists or returns None .
+
+
+
+
+list_commands ( ctx )
+Returns a list of subcommand names in the order they should
+appear.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.list.frelist.html b/fre.list.frelist.html
new file mode 100644
index 00000000..7e9351cd
--- /dev/null
+++ b/fre.list.frelist.html
@@ -0,0 +1,112 @@
+
+
+
+
+
+
+
+
+
fre.list.frelist module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.list.frelist module
+fre list
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.list.frelistexample.html b/fre.list.frelistexample.html
new file mode 100644
index 00000000..a1b90fde
--- /dev/null
+++ b/fre.list.frelistexample.html
@@ -0,0 +1,114 @@
+
+
+
+
+
+
+
+
+
fre.list.frelistexample module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.list.frelistexample module
+experimentation file for integrating one file’s functions into main prototype fre file
+authored by Bennett. Chang@ noaa. gov | bcc2761
+NOAA | GFDL
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.list.html b/fre.list.html
new file mode 100644
index 00000000..4eeb7647
--- /dev/null
+++ b/fre.list.html
@@ -0,0 +1,123 @@
+
+
+
+
+
+
+
+
+
fre.list package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.make.createCheckout.html b/fre.make.createCheckout.html
new file mode 100644
index 00000000..b8d32c37
--- /dev/null
+++ b/fre.make.createCheckout.html
@@ -0,0 +1,116 @@
+
+
+
+
+
+
+
+
+
fre.make.createCheckout module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.make.createCheckout module
+
+
+fre.make.createCheckout. checkout_create ( yamlfile , platform , target , no_parallel_checkout , jobs , execute , verbose )
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.make.createCompile.html b/fre.make.createCompile.html
new file mode 100644
index 00000000..c14cec61
--- /dev/null
+++ b/fre.make.createCompile.html
@@ -0,0 +1,116 @@
+
+
+
+
+
+
+
+
+
fre.make.createCompile module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.make.createCompile module
+
+
+fre.make.createCompile. compile_create ( yamlfile , platform , target , jobs , parallel , execute , verbose )
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.make.createDocker.html b/fre.make.createDocker.html
new file mode 100644
index 00000000..751afcc6
--- /dev/null
+++ b/fre.make.createDocker.html
@@ -0,0 +1,116 @@
+
+
+
+
+
+
+
+
+
fre.make.createDocker module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.make.createDocker module
+
+
+fre.make.createDocker. dockerfile_create ( yamlfile , platform , target , execute )
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.make.createMakefile.html b/fre.make.createMakefile.html
new file mode 100644
index 00000000..7da0946c
--- /dev/null
+++ b/fre.make.createMakefile.html
@@ -0,0 +1,116 @@
+
+
+
+
+
+
+
+
+
fre.make.createMakefile module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.make.createMakefile module
+
+
+fre.make.createMakefile. makefile_create ( yamlfile , platform , target )
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.make.fremake.html b/fre.make.fremake.html
new file mode 100644
index 00000000..0c3d5c6b
--- /dev/null
+++ b/fre.make.fremake.html
@@ -0,0 +1,111 @@
+
+
+
+
+
+
+
+
+
fre.make.fremake module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.make.fremake module
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.make.html b/fre.make.html
new file mode 100644
index 00000000..22a21680
--- /dev/null
+++ b/fre.make.html
@@ -0,0 +1,164 @@
+
+
+
+
+
+
+
+
+
fre.make package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.make package
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.make.runFremake.html b/fre.make.runFremake.html
new file mode 100644
index 00000000..b04db054
--- /dev/null
+++ b/fre.make.runFremake.html
@@ -0,0 +1,120 @@
+
+
+
+
+
+
+
+
+
fre.make.runFremake module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.make.runFremake module
+date 2023
+author(s): Tom Robinson, Dana Singh, Bennett Chang
+fre make is used to create, run and checkout code, and compile a model.
+
+
+fre.make.runFremake. fremake_run ( yamlfile , platform , target , parallel , jobs , no_parallel_checkout , execute , verbose )
+run fremake via click
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.make.tests.html b/fre.make.tests.html
new file mode 100644
index 00000000..fa98f290
--- /dev/null
+++ b/fre.make.tests.html
@@ -0,0 +1,129 @@
+
+
+
+
+
+
+
+
+
fre.make.tests package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.make.tests package
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.make.tests.test_create_makefile.html b/fre.make.tests.test_create_makefile.html
new file mode 100644
index 00000000..b3e4e17c
--- /dev/null
+++ b/fre.make.tests.test_create_makefile.html
@@ -0,0 +1,142 @@
+
+
+
+
+
+
+
+
+
fre.make.tests.test_create_makefile module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.make.tests.test_create_makefile module
+Test fre make create-makefile
+
+
+fre.make.tests.test_create_makefile. test_bm_makefile_creation ( )
+Check the makefile is created when a bare-metal platform is used
+
+
+
+
+fre.make.tests.test_create_makefile. test_compileyaml_exists ( )
+Check the compile yaml exists
+
+
+
+
+fre.make.tests.test_create_makefile. test_container_makefile_creation ( )
+Check the makefile is created when the container platform is used
+
+
+
+
+fre.make.tests.test_create_makefile. test_modelyaml_exists ( )
+Check the model yaml exists
+
+
+
+
+fre.make.tests.test_create_makefile. test_platformyaml_exists ( )
+Check the platform yaml exists
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.checkout_script.html b/fre.pp.checkout_script.html
new file mode 100644
index 00000000..bdb26235
--- /dev/null
+++ b/fre.pp.checkout_script.html
@@ -0,0 +1,122 @@
+
+
+
+
+
+
+
+
+
fre.pp.checkout_script module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.pp.checkout_script module
+Description: Checkout script which accounts for 4 different scenarios:
+1. branch not given, folder does not exist,
+2. branch given, folder does not exist,
+3. branch not given, folder exists,
+4. branch given and folder exists
+
+
+fre.pp.checkout_script. checkout_template ( experiment = None , platform = None , target = None , branch = None )
+Checkout the workflow template files from the repo
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.configure_script_xml.html b/fre.pp.configure_script_xml.html
new file mode 100644
index 00000000..5d5c1d81
--- /dev/null
+++ b/fre.pp.configure_script_xml.html
@@ -0,0 +1,184 @@
+
+
+
+
+
+
+
+
+
fre.pp.configure_script_xml module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.configure_script_yaml.html b/fre.pp.configure_script_yaml.html
new file mode 100644
index 00000000..c8072a51
--- /dev/null
+++ b/fre.pp.configure_script_yaml.html
@@ -0,0 +1,159 @@
+
+
+
+
+
+
+
+
+
fre.pp.configure_script_yaml module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.frepp.html b/fre.pp.frepp.html
new file mode 100644
index 00000000..56d13ea9
--- /dev/null
+++ b/fre.pp.frepp.html
@@ -0,0 +1,112 @@
+
+
+
+
+
+
+
+
+
fre.pp.frepp module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.pp.frepp module
+fre pp
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.html b/fre.pp.html
new file mode 100644
index 00000000..c081553b
--- /dev/null
+++ b/fre.pp.html
@@ -0,0 +1,189 @@
+
+
+
+
+
+
+
+
+
fre.pp package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.install.html b/fre.pp.install.html
new file mode 100644
index 00000000..d49d9823
--- /dev/null
+++ b/fre.pp.install.html
@@ -0,0 +1,121 @@
+
+
+
+
+
+
+
+
+
fre.pp.install module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.pp.install module
+fre pp install
+
+
+fre.pp.install. install_subtool ( experiment , platform , target )
+Install the Cylc workflow definition located in
+~/cylc-src/<experiment>__<platform>__<target>
+to
+~/cylc-run/<experiment>__<platform>__<target>
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.run.html b/fre.pp.run.html
new file mode 100644
index 00000000..5cc2226c
--- /dev/null
+++ b/fre.pp.run.html
@@ -0,0 +1,119 @@
+
+
+
+
+
+
+
+
+
fre.pp.run module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.pp.run module
+fre pp run
+
+
+fre.pp.run. pp_run_subtool ( experiment , platform , target )
+Start or restart the Cylc workflow identified by:
+<experiment>__<platform>__<target>
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.status.html b/fre.pp.status.html
new file mode 100644
index 00000000..008ce77b
--- /dev/null
+++ b/fre.pp.status.html
@@ -0,0 +1,119 @@
+
+
+
+
+
+
+
+
+
fre.pp.status module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.pp.status module
+fre pp status
+
+
+fre.pp.status. status_subtool ( experiment , platform , target )
+Report workflow state for the Cylc workflow
+<experiment>__<platform>__<target>
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.tests.html b/fre.pp.tests.html
new file mode 100644
index 00000000..cb7a92e9
--- /dev/null
+++ b/fre.pp.tests.html
@@ -0,0 +1,131 @@
+
+
+
+
+
+
+
+
+
fre.pp.tests package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.pp.tests package
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.tests.test_configure_script_yaml.html b/fre.pp.tests.test_configure_script_yaml.html
new file mode 100644
index 00000000..e28c300d
--- /dev/null
+++ b/fre.pp.tests.test_configure_script_yaml.html
@@ -0,0 +1,126 @@
+
+
+
+
+
+
+
+
+
fre.pp.tests.test_configure_script_yaml module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.tests.test_rose_quoting.html b/fre.pp.tests.test_rose_quoting.html
new file mode 100644
index 00000000..c69157ab
--- /dev/null
+++ b/fre.pp.tests.test_rose_quoting.html
@@ -0,0 +1,121 @@
+
+
+
+
+
+
+
+
+
fre.pp.tests.test_rose_quoting module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.pp.tests.test_rose_quoting module
+
+
+fre.pp.tests.test_rose_quoting. test_boolean ( )
+
+
+
+
+fre.pp.tests.test_rose_quoting. test_string ( )
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.trigger.html b/fre.pp.trigger.html
new file mode 100644
index 00000000..1183837b
--- /dev/null
+++ b/fre.pp.trigger.html
@@ -0,0 +1,118 @@
+
+
+
+
+
+
+
+
+
fre.pp.trigger module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.pp.trigger module
+fre pp trigger
+
+
+fre.pp.trigger. trigger ( experiment , platform , target , time )
+Trigger the pp-starter task for the time indicated
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.validate.html b/fre.pp.validate.html
new file mode 100644
index 00000000..dd7ec1dd
--- /dev/null
+++ b/fre.pp.validate.html
@@ -0,0 +1,119 @@
+
+
+
+
+
+
+
+
+
fre.pp.validate module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.pp.validate module
+fre pp validate
+
+
+fre.pp.validate. validate_subtool ( experiment , platform , target )
+Validate the Cylc workflow definition located in
+~/cylc-src/<experiment>__<platform>__<target>
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.pp.wrapper.html b/fre.pp.wrapper.html
new file mode 100644
index 00000000..af535cd5
--- /dev/null
+++ b/fre.pp.wrapper.html
@@ -0,0 +1,114 @@
+
+
+
+
+
+
+
+
+
fre.pp.wrapper module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.run.frerun.html b/fre.run.frerun.html
new file mode 100644
index 00000000..a0bbb3fd
--- /dev/null
+++ b/fre.run.frerun.html
@@ -0,0 +1,112 @@
+
+
+
+
+
+
+
+
+
fre.run.frerun module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.run.frerun module
+entry point for fre run subcommands
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.run.frerunexample.html b/fre.run.frerunexample.html
new file mode 100644
index 00000000..2199c8e0
--- /dev/null
+++ b/fre.run.frerunexample.html
@@ -0,0 +1,114 @@
+
+
+
+
+
+
+
+
+
fre.run.frerunexample module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.run.frerunexample module
+experimentation file for integrating one file’s functions into main prototype fre file
+authored by Bennett. Chang@ noaa. gov | bcc2761
+NOAA | GFDL
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.run.html b/fre.run.html
new file mode 100644
index 00000000..95a8a40d
--- /dev/null
+++ b/fre.run.html
@@ -0,0 +1,123 @@
+
+
+
+
+
+
+
+
+
fre.run package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.test.fretest.html b/fre.test.fretest.html
new file mode 100644
index 00000000..f6a0310e
--- /dev/null
+++ b/fre.test.fretest.html
@@ -0,0 +1,112 @@
+
+
+
+
+
+
+
+
+
fre.test.fretest module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.test.fretest module
+entry point for fre test subcommands
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.test.fretestexample.html b/fre.test.fretestexample.html
new file mode 100644
index 00000000..1686dac3
--- /dev/null
+++ b/fre.test.fretestexample.html
@@ -0,0 +1,114 @@
+
+
+
+
+
+
+
+
+
fre.test.fretestexample module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.test.fretestexample module
+experimentation file for integrating one file’s functions into main prototype fre file
+authored by Bennett. Chang@ noaa. gov | bcc2761
+NOAA | GFDL
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.test.html b/fre.test.html
new file mode 100644
index 00000000..6cf133b5
--- /dev/null
+++ b/fre.test.html
@@ -0,0 +1,123 @@
+
+
+
+
+
+
+
+
+
fre.test package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.html b/fre.tests.html
new file mode 100644
index 00000000..cf8d2e9b
--- /dev/null
+++ b/fre.tests.html
@@ -0,0 +1,231 @@
+
+
+
+
+
+
+
+
+
fre.tests package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_app_cli.html b/fre.tests.test_fre_app_cli.html
new file mode 100644
index 00000000..a644cbee
--- /dev/null
+++ b/fre.tests.test_fre_app_cli.html
@@ -0,0 +1,172 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_app_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_app_cli module
+test “fre app” calls
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app ( capfd )
+fre app
+
+
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app_gen_time_averages ( capfd )
+fre cmor run
+
+
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app_gen_time_averages_help ( capfd )
+fre cmor run –help
+
+
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app_gen_time_averages_opt_dne ( capfd )
+fre cmor run optionDNE
+
+
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app_help ( capfd )
+fre app –help
+
+
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app_opt_dne ( capfd )
+fre app optionDNE
+
+
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app_regrid ( capfd )
+fre cmor run
+
+
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app_regrid_help ( capfd )
+fre cmor run –help
+
+
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app_regrid_opt_dne ( capfd )
+fre cmor run optionDNE
+
+
+
+
+fre.tests.test_fre_app_cli. test_cli_fre_app_regrid_test_case_1 ( capfd )
+fre cmor run –help
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_catalog_cli.html b/fre.tests.test_fre_catalog_cli.html
new file mode 100644
index 00000000..69fc46b0
--- /dev/null
+++ b/fre.tests.test_fre_catalog_cli.html
@@ -0,0 +1,152 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_catalog_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_catalog_cli module
+test “fre catalog” calls
+
+
+fre.tests.test_fre_catalog_cli. test_cli_fre_catalog ( )
+fre catalog
+
+
+
+
+fre.tests.test_fre_catalog_cli. test_cli_fre_catalog_builder ( )
+fre catalog builder
+
+
+
+
+fre.tests.test_fre_catalog_cli. test_cli_fre_catalog_builder_help ( )
+fre catalog builder –help
+
+
+
+
+fre.tests.test_fre_catalog_cli. test_cli_fre_catalog_help ( )
+fre catalog –help
+
+
+
+
+fre.tests.test_fre_catalog_cli. test_cli_fre_catalog_merge ( )
+
+
+
+
+fre.tests.test_fre_catalog_cli. test_cli_fre_catalog_merge_help ( )
+
+
+
+
+fre.tests.test_fre_catalog_cli. test_cli_fre_catalog_opt_dne ( )
+fre catalog optionDNE
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_check_cli.html b/fre.tests.test_fre_check_cli.html
new file mode 100644
index 00000000..137fce31
--- /dev/null
+++ b/fre.tests.test_fre_check_cli.html
@@ -0,0 +1,130 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_check_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_check_cli module
+test “fre check” calls
+
+
+fre.tests.test_fre_check_cli. test_cli_fre_check ( )
+fre check
+
+
+
+
+fre.tests.test_fre_check_cli. test_cli_fre_check_help ( )
+fre check –help
+
+
+
+
+fre.tests.test_fre_check_cli. test_cli_fre_check_opt_dne ( )
+fre check optionDNE
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_cli.html b/fre.tests.test_fre_cli.html
new file mode 100644
index 00000000..42f3eb15
--- /dev/null
+++ b/fre.tests.test_fre_cli.html
@@ -0,0 +1,130 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_cli module
+test “fre” calls
+
+
+fre.tests.test_fre_cli. test_cli_fre ( )
+fre
+
+
+
+
+fre.tests.test_fre_cli. test_cli_fre_help ( )
+fre –help
+
+
+
+
+fre.tests.test_fre_cli. test_cli_fre_option_dne ( )
+fre optionDNE
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_cmor_cli.html b/fre.tests.test_fre_cmor_cli.html
new file mode 100644
index 00000000..9a7df35d
--- /dev/null
+++ b/fre.tests.test_fre_cmor_cli.html
@@ -0,0 +1,160 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_cmor_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_cmor_cli module
+test “fre cmor” calls
+
+
+fre.tests.test_fre_cmor_cli. test_cli_fre_cmor ( )
+fre cmor
+
+
+
+
+fre.tests.test_fre_cmor_cli. test_cli_fre_cmor_help ( )
+fre cmor –help
+
+
+
+
+fre.tests.test_fre_cmor_cli. test_cli_fre_cmor_opt_dne ( )
+fre cmor optionDNE
+
+
+
+
+fre.tests.test_fre_cmor_cli. test_cli_fre_cmor_run ( )
+fre cmor run
+
+
+
+
+fre.tests.test_fre_cmor_cli. test_cli_fre_cmor_run_case1 ( )
+fre cmor run, test-use case
+
+
+
+
+fre.tests.test_fre_cmor_cli. test_cli_fre_cmor_run_case2 ( )
+fre cmor run, test-use case
+
+
+
+
+fre.tests.test_fre_cmor_cli. test_cli_fre_cmor_run_help ( )
+fre cmor run –help
+
+
+
+
+fre.tests.test_fre_cmor_cli. test_cli_fre_cmor_run_opt_dne ( )
+fre cmor run optionDNE
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_list_cli.html b/fre.tests.test_fre_list_cli.html
new file mode 100644
index 00000000..fe14cd00
--- /dev/null
+++ b/fre.tests.test_fre_list_cli.html
@@ -0,0 +1,130 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_list_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_list_cli module
+test “fre list” calls
+
+
+fre.tests.test_fre_list_cli. test_cli_fre_list ( )
+fre list
+
+
+
+
+fre.tests.test_fre_list_cli. test_cli_fre_list_help ( )
+fre list –help
+
+
+
+
+fre.tests.test_fre_list_cli. test_cli_fre_list_opt_dne ( )
+fre list optionDNE
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_make_cli.html b/fre.tests.test_fre_make_cli.html
new file mode 100644
index 00000000..edffda20
--- /dev/null
+++ b/fre.tests.test_fre_make_cli.html
@@ -0,0 +1,142 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_make_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_make_cli module
+test “fre make” calls
+
+
+fre.tests.test_fre_make_cli. test_cli_fre_make ( )
+fre make
+
+
+
+
+fre.tests.test_fre_make_cli. test_cli_fre_make_create_checkout_baremetal ( )
+fre make create-checkout -y am5.yaml -p ncrc5.intel23 -t debug
+
+
+
+
+fre.tests.test_fre_make_cli. test_cli_fre_make_create_checkout_container ( )
+fre make create-checkout -y am5.yaml -p hpcme.2023 -t debug
+
+
+
+
+fre.tests.test_fre_make_cli. test_cli_fre_make_help ( )
+fre make –help
+
+
+
+
+fre.tests.test_fre_make_cli. test_cli_fre_make_opt_dne ( )
+fre make optionDNE
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_pp_cli.html b/fre.tests.test_fre_pp_cli.html
new file mode 100644
index 00000000..c6ab739f
--- /dev/null
+++ b/fre.tests.test_fre_pp_cli.html
@@ -0,0 +1,286 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_pp_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_pp_cli module
+test “fre pp” calls
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp ( )
+fre pp
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_checkout ( )
+fre pp checkout
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_checkout_case ( )
+fre pp checkout -e FOO -p BAR -t BAZ
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_checkout_help ( )
+fre pp checkout –help
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_checkout_opt_dne ( )
+fre pp checkout optionDNE
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_configure_xml ( )
+fre pp configure-xml
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_configure_xml_help ( )
+fre pp configure-xml –help
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_configure_xml_opt_dne ( )
+fre pp configure-xml optionDNE
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_configure_yaml ( )
+fre pp configure-yaml
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_configure_yaml_fail1 ( )
+fre pp configure-yaml
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_configure_yaml_help ( )
+fre pp configure-yaml –help
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_configure_yaml_opt_dne ( )
+fre pp configure-yaml optionDNE
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_help ( )
+fre pp –help
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_install ( )
+fre pp install
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_install_help ( )
+fre pp install –help
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_install_opt_dne ( )
+fre pp install optionDNE
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_opt_dne ( )
+fre pp optionDNE
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_run ( )
+fre pp run
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_run_help ( )
+fre pp run –help
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_run_opt_dne ( )
+fre pp run optionDNE
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_status ( )
+fre pp status
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_status_help ( )
+fre pp status –help
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_status_opt_dne ( )
+fre pp status optionDNE
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_validate ( )
+fre pp validate
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_validate_help ( )
+fre pp validate –help
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_validate_opt_dne ( )
+fre pp validate optionDNE
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_wrapper ( )
+fre pp wrapper
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_wrapper_help ( )
+fre pp wrapper –help
+
+
+
+
+fre.tests.test_fre_pp_cli. test_cli_fre_pp_wrapper_opt_dne ( )
+fre pp wrapper optionDNE
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_run_cli.html b/fre.tests.test_fre_run_cli.html
new file mode 100644
index 00000000..cae2052d
--- /dev/null
+++ b/fre.tests.test_fre_run_cli.html
@@ -0,0 +1,130 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_run_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_run_cli module
+test “fre run” calls
+
+
+fre.tests.test_fre_run_cli. test_cli_fre_run ( )
+fre run
+
+
+
+
+fre.tests.test_fre_run_cli. test_cli_fre_run_help ( )
+fre run –help
+
+
+
+
+fre.tests.test_fre_run_cli. test_cli_fre_run_opt_dne ( )
+fre run optionDNE
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_test_cli.html b/fre.tests.test_fre_test_cli.html
new file mode 100644
index 00000000..d00e47cc
--- /dev/null
+++ b/fre.tests.test_fre_test_cli.html
@@ -0,0 +1,130 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_test_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+fre.tests.test_fre_test_cli module
+test “fre test” calls
+
+
+fre.tests.test_fre_test_cli. test_cli_fre_test ( )
+fre test
+
+
+
+
+fre.tests.test_fre_test_cli. test_cli_fre_test_help ( )
+fre test –help
+
+
+
+
+fre.tests.test_fre_test_cli. test_cli_fre_test_opt_dne ( )
+fre test optionDNE
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.tests.test_fre_yamltools_cli.html b/fre.tests.test_fre_yamltools_cli.html
new file mode 100644
index 00000000..343fb16c
--- /dev/null
+++ b/fre.tests.test_fre_yamltools_cli.html
@@ -0,0 +1,130 @@
+
+
+
+
+
+
+
+
+
fre.tests.test_fre_yamltools_cli module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.yamltools.combine_yamls.html b/fre.yamltools.combine_yamls.html
new file mode 100644
index 00000000..790f549e
--- /dev/null
+++ b/fre.yamltools.combine_yamls.html
@@ -0,0 +1,232 @@
+
+
+
+
+
+
+
+
+
fre.yamltools.combine_yamls module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.yamltools.freyamltools.html b/fre.yamltools.freyamltools.html
new file mode 100644
index 00000000..702323e9
--- /dev/null
+++ b/fre.yamltools.freyamltools.html
@@ -0,0 +1,112 @@
+
+
+
+
+
+
+
+
+
fre.yamltools.freyamltools module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.yamltools.html b/fre.yamltools.html
new file mode 100644
index 00000000..6f692f68
--- /dev/null
+++ b/fre.yamltools.html
@@ -0,0 +1,175 @@
+
+
+
+
+
+
+
+
+
fre.yamltools package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.yamltools.tests.html b/fre.yamltools.tests.html
new file mode 100644
index 00000000..3acdf727
--- /dev/null
+++ b/fre.yamltools.tests.html
@@ -0,0 +1,135 @@
+
+
+
+
+
+
+
+
+
fre.yamltools.tests package — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/fre.yamltools.tests.test_combine_yamls.html b/fre.yamltools.tests.test_combine_yamls.html
new file mode 100644
index 00000000..e02aae90
--- /dev/null
+++ b/fre.yamltools.tests.test_combine_yamls.html
@@ -0,0 +1,182 @@
+
+
+
+
+
+
+
+
+
fre.yamltools.tests.test_combine_yamls module — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/genindex.html b/genindex.html
new file mode 100644
index 00000000..449430c2
--- /dev/null
+++ b/genindex.html
@@ -0,0 +1,1405 @@
+
+
+
+
+
+
+
+
Index — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+
Index
+
+
+
A
+ |
C
+ |
D
+ |
E
+ |
F
+ |
G
+ |
I
+ |
J
+ |
L
+ |
M
+ |
P
+ |
Q
+ |
R
+ |
S
+ |
T
+ |
U
+ |
V
+ |
W
+ |
Y
+
+
+
A
+
+
+
C
+
+
+
D
+
+
+
E
+
+
+
F
+
+
+
+ fre
+
+
+
+ fre.app
+
+
+
+ fre.app.freapp
+
+
+
+ fre.app.generate_time_averages
+
+
+
+ fre.app.generate_time_averages.cdoTimeAverager
+
+
+
+ fre.app.generate_time_averages.frenctoolsTimeAverager
+
+
+
+ fre.app.generate_time_averages.frepytoolsTimeAverager
+
+
+
+ fre.app.generate_time_averages.generate_time_averages
+
+
+
+ fre.app.generate_time_averages.tests
+
+
+
+ fre.app.generate_time_averages.tests.test_generate_time_averages
+
+
+
+ fre.app.generate_time_averages.timeAverager
+
+
+
+ fre.app.mask_atmos_plevel
+
+
+
+ fre.app.regrid_xy
+
+
+
+ fre.app.regrid_xy.regrid_xy
+
+
+
+ fre.catalog
+
+
+
+ fre.catalog.frecatalog
+
+
+
+ fre.catalog.tests
+
+
+
+ fre.catalog.tests.test_fre_catalog
+
+
+
+ fre.check
+
+
+
+ fre.check.frecheck
+
+
+
+ fre.check.frecheckexample
+
+
+
+ fre.cmor
+
+
+
+ fre.cmor.cmor_mixer
+
+
+
+ fre.cmor.frecmor
+
+
+
+ fre.cmor.tests
+
+
+
+ fre.cmor.tests.test_cmor_run_subtool
+
+
+
+ fre.fre
+
+
+
+ fre.lazy_group
+
+
+
+ fre.list
+
+
+
+ fre.list.frelist
+
+
+
+ fre.list.frelistexample
+
+
+
+ fre.make
+
+
+
+ fre.make.createCheckout
+
+
+
+ fre.make.createCompile
+
+
+
+ fre.make.createDocker
+
+
+
+ fre.make.createMakefile
+
+
+
+ fre.make.fremake
+
+
+
+ fre.make.runFremake
+
+
+
+ fre.make.tests
+
+
+
+ fre.make.tests.test_create_makefile
+
+
+
+
+
+
+
G
+
+
+
I
+
+
+
J
+
+
+
L
+
+
+
M
+
+
+
P
+
+
+
Q
+
+
+
R
+
+
+
S
+
+
+
T
+
+
+
U
+
+
+
V
+
+
+
W
+
+
+
Y
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/index.html b/index.html
new file mode 100644
index 00000000..a2a1b2f1
--- /dev/null
+++ b/index.html
@@ -0,0 +1,157 @@
+
+
+
+
+
+
+
+
+
Welcome to fre-cli’s documentation! — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+Welcome to fre-cli
’s documentation!
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/lib/docs/conf.py b/lib/docs/conf.py
new file mode 100644
index 00000000..32399782
--- /dev/null
+++ b/lib/docs/conf.py
@@ -0,0 +1,29 @@
+# Configuration file for the Sphinx documentation builder.
+#
+# For the full list of built-in configuration values, see the documentation:
+# https://www.sphinx-doc.org/en/master/usage/configuration.html
+
+# -- Project information -----------------------------------------------------
+# https://www.sphinx-doc.org/en/master/usage/configuration.html#project-information
+
+project = 'Fre-Cli'
+copyright = '2024, Bennett Chang'
+author = 'Bennett Chang'
+release = '1.0'
+
+# -- General configuration ---------------------------------------------------
+# https://www.sphinx-doc.org/en/master/usage/configuration.html#general-configuration
+
+extensions = ['sphinx.ext.autodoc']
+
+templates_path = ['_templates']
+exclude_patterns = []
+
+
+
+# -- Options for HTML output -------------------------------------------------
+# https://www.sphinx-doc.org/en/master/usage/configuration.html#options-for-html-output
+
+html_theme = 'renku'
+#html_theme = 'sphinx_rtd_theme'
+html_static_path = ['_static']
diff --git a/lib/fre/README.md b/lib/fre/README.md
new file mode 100644
index 00000000..4570b317
--- /dev/null
+++ b/lib/fre/README.md
@@ -0,0 +1,43 @@
+
+* Refer to fre-cli [README.md](https://github.com/NOAA-GFDL/fre-cli/blob/main/README.md) for usage and tips.
+
+
diff --git a/lib/fre/__init__.py b/lib/fre/__init__.py
new file mode 100644
index 00000000..4524edb3
--- /dev/null
+++ b/lib/fre/__init__.py
@@ -0,0 +1,10 @@
+# turn xxxx.y into xxxx.0y
+import importlib.metadata
+version_unexpanded = importlib.metadata.version('fre-cli')
+version_unexpanded_split = version_unexpanded.split('.')
+if len(version_unexpanded_split[1]) == 1:
+ version_minor = "0" + version_unexpanded_split[1]
+else:
+ version_minor = version_unexpanded_split[1]
+version = version_unexpanded_split[0] + '.' + version_minor
+__version__ = version
diff --git a/lib/fre/app/__init__.py b/lib/fre/app/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/app/freapp.py b/lib/fre/app/freapp.py
new file mode 100644
index 00000000..545356e4
--- /dev/null
+++ b/lib/fre/app/freapp.py
@@ -0,0 +1,127 @@
+#!/usr/bin/env python3
+''' fre app calls '''
+
+import time
+
+import click
+
+from .mask_atmos_plevel import mask_atmos_plevel_subtool
+from .generate_time_averages.generate_time_averages import generate
+from .regrid_xy.regrid_xy import _regrid_xy
+
+@click.group(help=click.style(" - access fre app subcommands", fg=(250,154,90)))
+def app_cli():
+ ''' entry point to fre app click commands '''
+
+@app_cli.command()
+@click.option("-i", "--input_dir",
+ type = str,
+ help = "`inputDir` / `input_dir` (env var) specifies input directory to regrid, " + \
+ "typically an untarredv history file archive" ,
+ required = True)
+@click.option("-o", "--output_dir",
+ type = str,
+ help = "`outputDir` / `output_dir` (env var) specifies target location for output" + \
+ " regridded files",
+ required = True)
+@click.option("-b", "--begin",
+ type = str,
+ help = "`begin` / `begin` (env var) ISO8601 datetime format specification for" + \
+ " starting date of data, part of input target file name",
+ required = True)
+@click.option("-tmp", "--tmp_dir",
+ type = str,
+ help = "`TMPDIR` / `tmp_dir` (env var) temp directory for location of file " + \
+ "read/writes",
+ required = True)
+@click.option("-rd", "--remap_dir",
+ type = str,
+ help = "`fregridRemapDir` / `remap_dir` (env var) directory containing remap file" + \
+ " for regridding",
+ required = True)
+@click.option("-s", "--source",
+ type = str,
+ help = "`source` / `source` (env var) source name for input target file name " + \
+ "within input directory to target for regridding. the value for `source` " + \
+ "must be present in at least one component's configuration fields",
+ required = True)
+@click.option("-g", "--grid_spec",
+ type = str,
+ help = "`gridSpec` / `grid_spec` (env var) file containing mosaic for regridding",
+ required = True)
+@click.option("-xy", "--def_xy_interp",
+ type = str,
+ help = "`defaultxyInterp` / `def_xy_interp` (env var) default lat/lon resolution " + \
+ "for output regridding. (change me? TODO)",
+ required = True)
+@click.pass_context
+def regrid(context,
+ input_dir, output_dir, begin, tmp_dir,
+ remap_dir, source, grid_spec, def_xy_interp ):
+ ''' regrid target netcdf file '''
+ context.forward(_regrid_xy)
+
+@app_cli.command()
+@click.option("-i", "--infile",
+ type = str,
+ help = "Input NetCDF file containing pressure-level output to be masked",
+ required = True)
+@click.option("-o", "--outfile",
+ type = str,
+ help = "Output file",
+ required = True)
+@click.option("-p", "--psfile", # surface pressure... ps? TODO
+ help = "Input NetCDF file containing surface pressure (ps)",
+ required = True)
+@click.pass_context
+def mask_atmos_plevel(context, infile, outfile, psfile):
+ # pylint: disable = unused-argument
+ """Mask out pressure level diagnostic output below land surface"""
+ context.forward(mask_atmos_plevel_subtool)
+
+
+@app_cli.command()
+@click.option("-i", "--inf",
+ type = str,
+ required = True,
+ help = "Input file name")
+@click.option("-o", "--outf",
+ type = str,
+ required = True,
+ help = "Output file name")
+@click.option("-p", "--pkg",
+ type = click.Choice(["cdo","fre-nctools","fre-python-tools"]),
+ default = "cdo",
+ help = "Time average approach")
+@click.option("-v", "--var",
+ type = str,
+ default = None,
+ help = "Specify variable to average")
+@click.option("-u", "--unwgt",
+ is_flag = True,
+ default = False,
+ help = "Request unweighted statistics")
+@click.option("-a", "--avg_type",
+ type = click.Choice(["month","seas","all"]),
+ default = "all",
+ help = "Type of time average to generate. \n \
+ currently, fre-nctools and fre-python-tools pkg options\n \
+ do not support seasonal and monthly averaging.\n")
+@click.option("-s", "--stddev_type",
+ type = click.Choice(["samp","pop","samp_mean","pop_mean"]),
+ default = "samp",
+ help = "Compute standard deviations for time-averages as well")
+@click.pass_context
+def gen_time_averages(context, inf, outf, pkg, var, unwgt, avg_type, stddev_type):
+ # pylint: disable = unused-argument
+ """
+ generate time averages for specified set of netCDF files.
+ Example: generate-time-averages.py /path/to/your/files/
+ """
+ start_time = time.perf_counter()
+ context.forward(generate)
+ # need to change to a click.echo, not sure if echo supports f strings
+ print(f'Finished in total time {round(time.perf_counter() - start_time , 2)} second(s)')
+
+if __name__ == "__main__":
+ app_cli()
diff --git a/lib/fre/app/generate_time_averages/README.md b/lib/fre/app/generate_time_averages/README.md
new file mode 100644
index 00000000..49991e13
--- /dev/null
+++ b/lib/fre/app/generate_time_averages/README.md
@@ -0,0 +1,8 @@
+From an input netCDF file, output a time-averaged netCDF file.
+
+Will average all available information on a given variable across time,
+but can also handle monthly+seasonal averaging.
+
+To run time-averaging tests, return to root directory and call just those
+tests with `python -m pytest tests/test_generate_time_averages.py`, or run
+all tests with `python -m pytests tests/`
\ No newline at end of file
diff --git a/lib/fre/app/generate_time_averages/__init__.py b/lib/fre/app/generate_time_averages/__init__.py
new file mode 100644
index 00000000..37410fef
--- /dev/null
+++ b/lib/fre/app/generate_time_averages/__init__.py
@@ -0,0 +1,3 @@
+'''required for generate_time_averages module import functionality'''
+__all__ = ['generate_time_averages', 'timeAverager',
+ 'frenctoolsTimeAverager', 'cdoTimeAverager', 'frepytoolsTimeAverager']
diff --git a/lib/fre/app/generate_time_averages/cdoTimeAverager.py b/lib/fre/app/generate_time_averages/cdoTimeAverager.py
new file mode 100644
index 00000000..d288bebb
--- /dev/null
+++ b/lib/fre/app/generate_time_averages/cdoTimeAverager.py
@@ -0,0 +1,85 @@
+''' class using (mostly) cdo functions for time-averages '''
+from .timeAverager import timeAverager
+
+class cdoTimeAverager(timeAverager):
+ '''
+ class inheriting from abstract base class timeAverager
+ generates time-averages using cdo (mostly, see weighted approach)
+ '''
+
+ def generate_timavg(self, infile=None, outfile=None):
+ ''' use cdo package routines via python bindings '''
+ assert self.pkg=="cdo"
+ if __debug__:
+ print(locals()) #input argument details
+
+ if all([self.avg_type!='all',self.avg_type!='seas',self.avg_type!='month',
+ self.avg_type is not None]):
+ print('ERROR, avg_type requested unknown.')
+ return 1
+
+ if self.var is not None:
+ print(f'WARNING: variable specification (var={self.var})' + \
+ f' not currently supported for cdo time averaging. ignoring!')
+
+ import cdo
+ print(f'python-cdo version is {cdo.__version__}')
+ from cdo import Cdo
+
+ _cdo=Cdo()
+
+ wgts_sum=0
+ if not self.unwgt: #weighted case, cdo ops alone don't support a weighted time-average.
+ from netCDF4 import Dataset
+ import numpy
+
+ nc_fin = Dataset(infile, 'r')
+
+ time_bnds=nc_fin['time_bnds'][:].copy()
+ wgts = ( numpy.moveaxis(time_bnds,0,-1)[1][:].copy() - \
+ numpy.moveaxis(time_bnds,0,-1)[0][:].copy() )
+ wgts_sum=sum(wgts)
+ if __debug__:
+ print(f'wgts_sum={wgts_sum}')
+
+ if self.avg_type == 'all':
+ if self.stddev_type is None:
+ print('time average over all time requested.')
+ if self.unwgt:
+ _cdo.timmean(input=infile, output=outfile, returnCdf=True)
+ else:
+ _cdo.divc( str(wgts_sum), input="-timsum -muldpm "+infile, output=outfile)
+
+ print('done averaging over all time.')
+ else:
+ print('time standard-deviation (N-1) over all time requested.')
+ _cdo.timstd1(input=infile, output=outfile, returnCdf=True)
+ print('done computing standard-deviation over all time.')
+
+ elif self.avg_type == 'seas':
+ if self.stddev_type is None:
+ print('seasonal time-averages requested.')
+ _cdo.yseasmean(input=infile, output=outfile, returnCdf=True)
+ print('done averaging over seasons.')
+ else:
+ print('seasonal time standard-deviation (N-1) requested.')
+ _cdo.yseasstd1(input=infile, output=outfile, returnCdf=True)
+ print('done averaging over seasons.')
+
+ elif self.avg_type == 'month':
+
+ if self.stddev_type is None:
+ print('monthly time-averages requested.')
+ _cdo.ymonmean(input=infile, output=outfile, returnCdf=True)
+ print('done averaging over months.')
+ else:
+ print('monthly time standard-deviation (N-1) requested.')
+ _cdo.ymonstd1(input=infile, output=outfile, returnCdf=True)
+ print('done averaging over months.')
+
+ else:
+ print(f'problem: unknown avg_type={self.avg_type}')
+ return 1
+
+ print('done averaging')
+ return 0
diff --git a/lib/fre/app/generate_time_averages/frenctoolsTimeAverager.py b/lib/fre/app/generate_time_averages/frenctoolsTimeAverager.py
new file mode 100644
index 00000000..32b9c2bb
--- /dev/null
+++ b/lib/fre/app/generate_time_averages/frenctoolsTimeAverager.py
@@ -0,0 +1,55 @@
+''' class for utilizing timavg.csh (aka script to TAVG fortran exe) in frenc-tools '''
+from .timeAverager import timeAverager
+
+class frenctoolsTimeAverager(timeAverager):
+ '''
+ class inheriting from abstract base class timeAverager
+ generates time-averages using fre-nctools
+ '''
+
+ def generate_timavg(self, infile=None, outfile=None):
+ ''' use fre-nctool's CLI timavg.csh with subprocess call '''
+ assert self.pkg=="fre-nctools"
+ if __debug__:
+ print(locals()) #input argument details
+
+ exitstatus=1
+ if self.avg_type!='all':
+ print(f'ERROR: avg_type={self.avg_type} not supported by this class at this time.')
+ return exitstatus
+
+ if self.unwgt:
+ print('WARNING: unwgt=True unsupported by frenctoolsAverager. ignoring!!!')
+
+ if self.stddev_type is not None:
+ print('WARNING: stddev_type arg unsupported by frenctoolsTimeAverager. ignoring!!!')
+
+ if self.var is not None:
+ print(f'WARNING: variable specification (var={self.var})' + \
+ f' not currently supported for frenctols time averaging. ignoring!')
+
+ if infile is None:
+ print('ERROR: I need an input file, specify a value for the infile argument')
+ return exitstatus
+
+ if outfile is None:
+ outfile='frenctoolsTimeAverage_'+infile
+ print(f'WARNING: no output filename given, setting outfile={outfile}')
+
+ from subprocess import Popen, PIPE
+
+ precision='-r8'
+ timavgcsh_command=['timavg.csh', precision, '-mb','-o', outfile, infile]
+ exitstatus=1
+ with Popen(timavgcsh_command,
+ stdout=PIPE, stderr=PIPE, shell=False) as subp:
+ output=subp.communicate()[0]
+ print(f'output={output}')
+
+ if subp.returncode < 0:
+ print('error')
+ else:
+ print('success')
+ exitstatus=0
+
+ return exitstatus
diff --git a/lib/fre/app/generate_time_averages/frepytoolsTimeAverager.py b/lib/fre/app/generate_time_averages/frepytoolsTimeAverager.py
new file mode 100644
index 00000000..1e1e6638
--- /dev/null
+++ b/lib/fre/app/generate_time_averages/frepytoolsTimeAverager.py
@@ -0,0 +1,249 @@
+''' class for python-native routine usuing netCDF4 and numpy to crunch time-averages '''
+
+import math
+import numpy
+from netCDF4 import Dataset
+
+from .timeAverager import timeAverager
+
+class frepytoolsTimeAverager(timeAverager):
+ '''
+ class inheriting from abstract base class timeAverager
+ generates time-averages using a python-native approach
+ avoids using other third party statistics functions by design.
+ '''
+
+ def generate_timavg(self, infile=None, outfile=None):
+ ''' frepytools approach in a python-native manner.
+ deliberately avoids pre-packaged routines. '''
+ assert self.pkg=="fre-python-tools"
+ if __debug__:
+ print(locals()) #input argument details
+
+ if __debug__:
+ print('calling generate_frepythontools_timavg for file: ' + infile)
+
+ if self.avg_type != 'all':
+ print(f'ERROR: avg_type={self.avg_type} not supported at this time.')
+ return 1
+
+ # (TODO) file I/O should be a sep function, no? make tests, extend
+ nc_fin = Dataset(infile, 'r')
+ if nc_fin.file_format != 'NETCDF4':
+ print(f'INFO: input file is not netCDF4 format, is {nc_fin.file_format}')
+
+ # (TODO) make this a sep function, make tests, extend
+ # identifying the input variable, two approaches
+ # user inputs target variable OR
+ # attempt to determine target var w/ bronx convention
+ if self.var is not None:
+ targ_var = self.var
+ else: # this can be replaced w/ a regex search maybe
+ targ_var = infile.split('/').pop().split('.')[-2]
+
+ if __debug__:
+ print(f'targ_var={targ_var}')
+
+ # (TODO) make this a sep function, make tests, extend
+ # check for the variable we're hoping is in the file
+ time_bnds = None
+ nc_fin_vars = nc_fin.variables
+ for key in nc_fin_vars:
+ if str(key) == targ_var: # found our variable, grab bounds
+ time_bnds = nc_fin['time_bnds'][:].copy()
+ break
+ if time_bnds is None:
+ print('requested variable not found. exit.')
+ return 1
+
+
+ # (TODO) determine what we need to worry about with masks and account for it
+ # check for mask, adjust accordingly
+ #is_masked = ma.is_masked(val_array)
+
+ # (TODO) make this a sep function, make tests, extend
+ # read in sizes of specific axes / compute weights
+ # weights can be encoded as a class member, whose existence
+ # depends on the user specifying unwgt=True, if vect_wgts=None, set the avg
+ # and stddev gen functions to the appropriate behavior (TODO)
+ fin_dims = nc_fin.dimensions
+ num_time_bnds = fin_dims['time'].size
+ if not self.unwgt: #compute sum of weights
+ wgts = ( numpy.moveaxis( time_bnds,0,-1 )[1][:].copy() - \
+ numpy.moveaxis( time_bnds,0,-1 )[0][:].copy() )
+ wgts_sum=sum(wgts)
+ if __debug__:
+ print(f'wgts_sum={wgts_sum}')
+
+
+ # initialize arrays, is there better practice for reserving the memory necessary
+ # for holding the day? is something that does more write-on-demand possible like
+ # reading data on-demand? (TODO)
+ num_lat_bnds=fin_dims['lat'].size
+ print(f'num_lat_bnds={num_lat_bnds}')
+ num_lon_bnds=fin_dims['lon'].size
+ print(f'num_lon_bnds={num_lon_bnds}')
+ avgvals=numpy.zeros((1,num_lat_bnds,num_lon_bnds),dtype=float)
+ if self.stddev_type is not None:
+ print('computing std. deviations')
+ stddevs=numpy.zeros((1,num_lat_bnds,num_lon_bnds),dtype=float)
+
+ # this loop behavior 100% should be re-factored into generator functions.
+ # they should be slightly faster, and much more readable. (TODO)
+ # the average/stddev cpu settings should also be genfunctions, their behavior
+ # (e.g. stddev_pop v stddev_samp) should be set given user inputs. (TODO)
+ # the computations can lean on numpy.stat more- i imagine it's faster (TODO)
+ # parallelism via multiprocessing shouldn't be too bad- explore an alt [dask] too (TODO)
+ # compute average, for each lat/lon coordinate over time record in file
+ if not self.unwgt: #weighted case
+ print('computing weighted statistics')
+ for lat in range(num_lat_bnds):
+ lon_val_array=numpy.moveaxis( nc_fin[targ_var][:],0,-1)[lat].copy()
+
+ for lon in range(num_lon_bnds):
+ tim_val_array= lon_val_array[lon].copy()
+ avgvals[0][lat][lon]=sum( (tim_val_array[tim] * wgts[tim] )
+ for tim in range(num_time_bnds) ) / wgts_sum
+
+ if self.stddev_type is not None: # implement stddeviation types TO DO
+ stddevs[0][lat][lon]=math.sqrt(
+ sum( wgts[tim] *
+ (tim_val_array[tim]-avgvals[0][lat][lon]) ** 2
+ for tim in range(num_time_bnds) )
+ / wgts_sum )
+ del tim_val_array
+ del lon_val_array
+ else: #unweighted case
+ print('computing unweighted statistics')
+ for lat in range(num_lat_bnds):
+ lon_val_array=numpy.moveaxis( nc_fin[targ_var][:],0,-1)[lat].copy()
+
+ for lon in range(num_lon_bnds):
+ tim_val_array= lon_val_array[lon].copy()
+ avgvals[0][lat][lon]=sum( # no time sum needed here, b.c. unweighted, so sum
+ tim_val_array[tim] for tim in range(num_time_bnds)
+ ) / num_time_bnds
+
+ if self.stddev_type is not None:
+
+ stddevs[0][lat][lon]=math.sqrt(
+ sum(
+ (tim_val_array[tim]-avgvals[0][lat][lon]) ** 2
+ for tim in range(num_time_bnds)
+ ) / ( (num_time_bnds - 1. ) )
+ )
+ del tim_val_array
+ del lon_val_array
+
+
+
+ # write output file
+ # (TODO) make this a sep function, make tests, extend,
+ # (TODO) consider compression particular;y for NETCDF file writing
+ # consider this approach instead:
+ # with Dataset( outfile, 'w', format='NETCDF4', persist=True ) as nc_fout:
+ nc_fout= Dataset( outfile, 'w', format=nc_fin.file_format, persist=True )
+
+ # (TODO) make this a sep function, make tests, extend
+ # write file global attributes
+ print('------- writing output attributes. --------')
+ unwritten_ncattr_list=[]
+ try:
+ nc_fout.setncatts(nc_fin.__dict__) #this copies the global attributes exactly.
+ except: # if the first way doesn't work...
+ print('could not copy ncatts from input file. trying to copy one-by-one')
+ fin_ncattrs=nc_fin.ncattrs()
+ for ncattr in fin_ncattrs:
+ print(f'\n_________\nncattr={ncattr}')
+ try:
+ nc_fout.setncattr(ncattr, nc_fin.getncattr(ncattr))
+ except:
+ print(f'could not get nc file attribute: {ncattr}. moving on.')
+ unwritten_ncattr_list.append(ncattr)
+ if len(unwritten_ncattr_list)>0:
+ print(f'WARNING: Some global attributes ({unwritten_ncattr_list}) were not written.')
+ print('------- DONE writing output attributes. --------')
+ ##
+
+ # (TODO) make this a sep function, make tests, extend
+ # write file dimensions
+ print('\n ------ writing output dimensions. ------ ')
+ unwritten_dims_list=[]
+ for key in fin_dims:
+ try:
+ if key=='time':
+ # this strongly influences the final data structure shape of the averages.
+ # if set to None, and lets say you try to write
+ # e.g. the original 'time_bnds' (which has 60 time steps)
+ # the array holding the avg. value will suddently have 60 time steps
+ # even though only 1 is needed, 59 time steps will have no data
+ #nc_fout.createDimension( dimname=key, size=None )
+ nc_fout.createDimension( dimname=key, size=1)
+ else:
+ nc_fout.createDimension( dimname=key, size=fin_dims[key].size )
+ except:
+ print(f'problem. cannot read/write dimension {key}')
+ unwritten_dims_list.append(key)
+ if len(unwritten_dims_list)>0:
+ print(f'WARNING: Some dimensions ({unwritten_dims_list}) were not written.')
+ print('------ DONE writing output dimensions. ------- \n')
+ ##
+
+
+ # (TODO) make this a sep function, make tests, extend
+ # first write the data we care most about- those we computed
+ # copying metadata, not fully correct
+ # but not far from wrong according to CF
+ # cell_methods must be changed TO DO
+ print(f'\n------- writing data for data {targ_var} -------- ')
+ nc_fout.createVariable(targ_var, nc_fin[targ_var].dtype, nc_fin[targ_var].dimensions)
+ nc_fout.variables[targ_var].setncatts(nc_fin[targ_var].__dict__)
+
+
+ nc_fout.variables[targ_var][:]=avgvals
+ if self.stddev_type is not None:
+ stddev_varname=targ_var+'_'+self.stddev_type+'_stddev'
+ nc_fout.createVariable(
+ stddev_varname, nc_fin[targ_var].dtype, nc_fin[targ_var].dimensions )
+ nc_fout.variables[stddev_varname].setncatts(nc_fin[targ_var].__dict__)
+ nc_fout.variables[stddev_varname][:]=stddevs
+ print('---------- DONE writing output variables. ---------')
+ ##
+
+ # (TODO) make this a sep function, make tests, extend
+ # write OTHER output variables (aka data) #prev code.
+ print('\n------- writing other output variables. -------- ')
+ unwritten_var_list=[]
+ unwritten_var_ncattr_dict={}
+ for var in nc_fin_vars:
+ if var != targ_var:
+ print(f'\nattempting to create output variable: {var}')
+ #print(f'is it a time variable? {self.var_is_time(nc_fin.variables[var])}')
+ nc_fout.createVariable(var, nc_fin[var].dtype, nc_fin[var].dimensions)
+ nc_fout.variables[var].setncatts(nc_fin[var].__dict__)
+ try:
+ nc_fout.variables[var][:] = nc_fin[var][:]
+ except:
+ print(f'could not write var={var}. i bet its the shape!')
+ print(f'nc_fin[var].shape={nc_fin[var].shape}')
+ #print(f'len(nc_fout.variables[{var}])={len(nc_fout.variables[var])}')
+ nc_fout.variables[var][:] = [ nc_fin[var][0] ]
+ print(f'time variable? {self.var_has_time_units(nc_fin.variables[var])}')
+ else:
+ continue
+
+ if len(unwritten_var_list)>0:
+ print(f'WARNING: some variables\' data ({unwritten_var_list}) was not written.')
+ if len(unwritten_var_ncattr_dict)>0:
+ print('WARNING: some variables\' metadata was not successfully written.')
+ print(f'WARNING: relevant variable/attr pairs: \n{unwritten_var_ncattr_dict}')
+ print('---------- DONE writing output variables. ---------')
+ ##
+
+
+ nc_fout.close()
+ #close input file
+ nc_fin.close()
+ print(f'wrote ouput file: {outfile}')
+
+ return 0
diff --git a/lib/fre/app/generate_time_averages/generate_time_averages.py b/lib/fre/app/generate_time_averages/generate_time_averages.py
new file mode 100644
index 00000000..1c4bbf69
--- /dev/null
+++ b/lib/fre/app/generate_time_averages/generate_time_averages.py
@@ -0,0 +1,61 @@
+''' tools for generating time averages from various packages '''
+import click
+
+def generate_time_average(infile=None, outfile=None,
+ pkg=None, var=None,
+ unwgt=False, avg_type=None, stddev_type=None):
+ ''' steering function to various averaging functions above'''
+ if __debug__:
+ print(locals()) #input argument details
+ exitstatus=1
+
+ #needs a case statement. better yet, smarter way to do this? (TODO)
+ myavger=None
+ if pkg == 'cdo' :
+ from .cdoTimeAverager import cdoTimeAverager
+ myavger=cdoTimeAverager(pkg = pkg, var=var,
+ unwgt = unwgt,
+ avg_type = avg_type, stddev_type = stddev_type)
+
+ elif pkg == 'fre-nctools' :
+ from .frenctoolsTimeAverager import frenctoolsTimeAverager
+ myavger=frenctoolsTimeAverager(pkg = pkg, var=var,
+ unwgt = unwgt ,
+ avg_type = avg_type, stddev_type = stddev_type)
+
+ elif pkg == 'fre-python-tools':
+ from .frepytoolsTimeAverager import frepytoolsTimeAverager
+ myavger=frepytoolsTimeAverager(pkg = pkg, var=var,
+ unwgt = unwgt,
+ avg_type = avg_type, stddev_type = stddev_type)
+
+ else :
+ print('requested package unknown. exit.')
+ return exitstatus
+
+ if __debug__:
+ print(f'myavger.__repr__={myavger.__repr__}')
+ if myavger is not None:
+ exitstatus=myavger.generate_timavg(infile=infile, outfile=outfile)
+ else:
+ print('ERROR: averager is None, check generate_time_average in generate_time_averages.py!')
+
+ return exitstatus
+
+@click.command()
+def generate(inf, outf, pkg, var, unwgt, avg_type, stddev_type):
+ ''' click entrypoint to time averaging routine '''
+ exitstatus=generate_time_average( inf, outf,
+ pkg, var,
+ unwgt,
+ avg_type, stddev_type)
+ if exitstatus!=0:
+ print(f'WARNING: exitstatus={exitstatus} != 0. Something exited poorly!')
+ else:
+ print('time averaging finished successfully')
+
+if __name__ == '__main__':
+ import time
+ start_time=time.perf_counter()
+ generate(inf, outf, pkg, var, unwgt, avg_type, stddev_type)
+ print(f'Finished in total time {round(time.perf_counter() - start_time , 2)} second(s)')
diff --git a/lib/fre/app/generate_time_averages/tests/__init__.py b/lib/fre/app/generate_time_averages/tests/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/app/generate_time_averages/tests/test_generate_time_averages.py b/lib/fre/app/generate_time_averages/tests/test_generate_time_averages.py
new file mode 100644
index 00000000..7e1345d9
--- /dev/null
+++ b/lib/fre/app/generate_time_averages/tests/test_generate_time_averages.py
@@ -0,0 +1,308 @@
+''' for testing fre app generate-time-averages '''
+import pathlib as pl
+import pytest
+
+def run_avgtype_pkg_calculations(infile=None,outfile=None, pkg=None, avg_type=None, unwgt=None,
+ stddev_type=None):
+ ''' test-harness function, called by other test functions. '''
+ assert all( [infile is not None, outfile is not None,
+ pkg is not None, avg_type is not None,
+ unwgt is not None] )
+ if pl.Path(outfile).exists():
+ print('output test file exists. deleting before remaking.')
+ pl.Path(outfile).unlink() #delete file so we check that it can be recreated
+ from fre.app.generate_time_averages import generate_time_averages as gtas
+ gtas.generate_time_average(infile = infile, outfile = outfile,
+ pkg = pkg, unwgt = unwgt,
+ avg_type = avg_type, stddev_type = stddev_type)
+ return pl.Path(outfile).exists()
+
+### preamble tests. if these fail, none of the others will succeed. -----------------
+time_avg_file_dir=str(pl.Path.cwd())+'/fre/app/generate_time_averages/tests/time_avg_test_files/'
+test_file_name='atmos.197901-198312.LWP.nc'
+
+def test_time_avg_file_dir_exists():
+ ''' look for input test file directory '''
+ assert pl.Path(time_avg_file_dir).exists()
+
+def test_time_avg_input_file_exists():
+ ''' look for input test file '''
+ assert pl.Path( time_avg_file_dir + test_file_name ).exists()
+
+### cdo avgs, unweighted, all/seasonal/monthly ------------------------
+def test_monthly_cdo_time_unwgt_avgs():
+ ''' generates an unweighted monthly time averaged file using cdo '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'ymonmean_unwgt_'+test_file_name),
+ pkg='cdo',avg_type='month',unwgt=True )
+
+def test_seasonal_cdo_time_unwgt_avgs():
+ ''' generates an unweighted seasonal time averaged file using cdo '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'yseasmean_unwgt_'+test_file_name),
+ pkg='cdo',avg_type='seas',unwgt=True )
+
+def test_cdo_time_unwgt_avgs():
+ ''' generates an unweighted time averaged file using cdo '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'timmean_unwgt_'+test_file_name),
+ pkg='cdo',avg_type='all',unwgt=True )
+
+
+#### cdo avgs, weighted, all/seasonal/monthly ------------------------
+## (TODO) WRITE THESE VERSIONS FOR CDOTIMEAVERAGER CLASS THEN MAKE THESE TESTS
+##def test_monthly_cdo_time_avgs():
+##def test_seasonal_cdo_time_avgs():
+
+def test_cdo_time_avgs():
+ ''' generates a weighted time averaged file using cdo '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'timmean_'+test_file_name),
+ pkg='cdo',avg_type='all',unwgt=False )
+
+
+### cdo stddevs, unweighted, all/seasonal/monthly ------------------------
+def test_monthly_cdo_time_unwgt_stddevs():
+ ''' generates a monthly time averaged file using cdo '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'ymonstddev1_unwgt_'+test_file_name),
+ pkg='cdo',avg_type='month',stddev_type='samp', unwgt=True )
+
+def test_seasonal_cdo_time_unwgt_stddevs():
+ ''' generates a seasonal time averaged file using cdo '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'yseasstddev1_unwgt_'+test_file_name),
+ pkg='cdo',avg_type='seas',stddev_type='samp',unwgt=True )
+
+def test_cdo_time_unwgt_stddevs():
+ ''' generates a time averaged file using cdo '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'yseasmean_unwgt_'+test_file_name),
+ pkg='cdo',avg_type='all',stddev_type='samp', unwgt=True )
+
+
+#### cdo stddevs, weighted, all/seasonal/monthly -----------------------
+## (TODO) WRITE THESE VERSIONS FOR CDOTIMEAVERAGER CLASS THEN MAKE THESE TESTS
+#def test_monthly_cdo_time_stddevs():
+#def test_seasonal_cdo_time_stddevs():
+#def test_cdo_time_stddevs():
+
+## frepythontools avgs+stddevs, weighted+unweighted, all ------------------------
+def test_fre_cli_time_avgs():
+ ''' generates a time averaged file using fre_cli's version '''
+ ''' weighted average, no std deviation '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'frepytools_timavg_'+test_file_name),
+ pkg='fre-python-tools',avg_type='all', unwgt=False )
+
+def test_fre_cli_time_unwgt_avgs():
+ ''' generates a time averaged file using fre_cli's version '''
+ ''' weighted average, no std deviation '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'frepytools_unwgt_timavg_'+test_file_name),
+ pkg='fre-python-tools',avg_type='all', unwgt=True )
+
+def test_fre_cli_time_avgs_stddevs():
+ ''' generates a time averaged file using fre_cli's version '''
+ ''' weighted average, no std deviation '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'frepytools_stddev_'+test_file_name),
+ pkg='fre-python-tools',avg_type='all', stddev_type='samp', unwgt=False )
+
+def test_fre_cli_time_unwgt_avgs_stddevs():
+ ''' generates a time averaged file using fre_cli's version '''
+ ''' weighted average, no std deviation '''
+ assert run_avgtype_pkg_calculations(
+ infile = (time_avg_file_dir+test_file_name),
+ outfile = (time_avg_file_dir+'frepytools_unwgt_stddev_'+test_file_name),
+ pkg='fre-python-tools',avg_type='all', stddev_type='samp', unwgt=True )
+
+## (TODO) WRITE THESE VERSIONS FOR FREPYTOOLSTIMEAVERAGER CLASS THEN MAKE THESE TESTS
+#def test_monthly_fre_cli_time_avgs():
+#def test_monthly_fre_cli_time_unwgt_avgs():
+#def test_monthly_fre_cli_time_avgs_stddevs():
+#def test_monthly_fre_cli_time_unwgt_avgs_stddevs():
+#
+#def test_seasonal_fre_cli_time_avgs():
+#def test_seasonal_fre_cli_time_unwgt_avgs():
+#def test_seasonal_fre_cli_time_avgs_stddevs():
+#def test_seasonal_fre_cli_time_unwgt_avgs_stddevs(:)
+
+
+
+## this will only work at GFDL. dev convenience only.
+#alt_str_fre_nctools_inf= \
+# 'tests/time_avg_test_files/fre_nctools_timavg_CLI_test_r8_b_atmos_LWP_1979_5y.nc'
+#def test_fre_nctools_time_avgs():
+# ''' generates a time averaged file using fre_cli's version '''
+# ''' weighted average, no std deviation '''
+# infile =time_avg_file_dir+test_file_name
+# all_outfile=time_avg_file_dir+'frenctools_timavg_'+test_file_name
+#
+# if pl.Path(all_outfile).exists():
+# print('output test file exists. deleting before remaking.')
+# pl.Path(all_outfile).unlink() #delete file so we check that it can be recreated
+#
+# from fre_cli.generate_time_averages import generate_time_averages as gtas
+# gtas.generate_time_average(infile = infile, outfile = all_outfile,
+# pkg='fre-nctools', unwgt=False, avg_type='all')
+# assert pl.Path(all_outfile).exists()
+
+
+# Numerics-based tests. these have room for improvemnent for sure (TODO)
+# compare frepytools, cdo time-average output to fre-nctools where possible
+var='LWP'
+str_fre_nctools_inf=time_avg_file_dir+'frenctools_timavg_'+test_file_name # this is now in the repo
+str_fre_pytools_inf=time_avg_file_dir+'frepytools_timavg_'+test_file_name
+str_cdo_inf=time_avg_file_dir+'timmean_'+test_file_name
+str_unwgt_fre_pytools_inf=time_avg_file_dir+'frepytools_unwgt_timavg_'+test_file_name
+str_unwgt_cdo_inf=time_avg_file_dir+'timmean_unwgt_'+test_file_name
+
+
+def test_compare_fre_cli_to_fre_nctools():
+ ''' compares fre_cli pkg answer to fre_nctools pkg answer '''
+ import numpy as np
+ import netCDF4 as nc
+ fre_pytools_inf=nc.Dataset(str_fre_pytools_inf,'r')
+
+ try:
+ fre_nctools_inf=nc.Dataset(str_fre_nctools_inf,'r')
+ except:
+ print('fre-nctools input file not found. \
+ probably because you are not at GFDL! run the shell script \
+ example if you would like to see this pass. otherwise, \
+ i will error right after this message.')
+ try:
+ fre_nctools_inf=nc.Dataset(alt_str_fre_nctools_inf,'r')
+ except:
+ print('fre-nctools output does not exist. create it first!')
+ assert False
+
+ fre_pytools_timavg=fre_pytools_inf[var][:].copy()
+ fre_nctools_timavg=fre_nctools_inf[var][:].copy()
+ assert all([ len(fre_pytools_timavg)==len(fre_nctools_timavg),
+ len(fre_pytools_timavg[0])==len(fre_nctools_timavg[0]),
+ len(fre_pytools_timavg[0][0])==len(fre_nctools_timavg[0][0]) ])
+
+ diff_pytools_nctools_timavg=fre_pytools_timavg-fre_nctools_timavg
+ for lat in range(0,len(diff_pytools_nctools_timavg[0])):
+ for lon in range(0,len(diff_pytools_nctools_timavg[0][0])):
+ print(f'lat={lat},lon={lon}')
+ #diff_at_latlon=diff_pytools_nctools_timavg[0][lat][lon]
+ #print(f'diff_pytools_nctools_timavg[0][lat][lon]={diff_at_latlon}')
+ if lon>10: break
+ break
+
+ non_zero_count=np.count_nonzero(diff_pytools_nctools_timavg[:])
+ #assert (non_zero_count == 0.) # bad way to check for zero.
+ assert not( (non_zero_count > 0.) or (non_zero_count < 0.) )
+
+@pytest.mark.skip(reason='test fails b.c. cdo cannot bitwise-reproduce fre-nctools answer')
+def test_compare_fre_cli_to_cdo():
+ ''' compares fre_cli pkg answer to cdo pkg answer '''
+ import numpy as np
+ import netCDF4 as nc
+ fre_pytools_inf=nc.Dataset(str_fre_pytools_inf,'r')
+
+ try:
+ cdo_inf=nc.Dataset(str_cdo_inf,'r')
+ except:
+ print('cdo input file not found. run cdo tests first.')
+ assert False
+
+ fre_pytools_timavg=fre_pytools_inf[var][:].copy()
+ cdo_timavg=cdo_inf[var][:].copy()
+
+ assert all([ len(fre_pytools_timavg)==len(cdo_timavg),
+ len(fre_pytools_timavg[0])==len(cdo_timavg[0]),
+ len(fre_pytools_timavg[0][0])==len(cdo_timavg[0][0]) ])
+
+ diff_pytools_cdo_timavg=fre_pytools_timavg-cdo_timavg
+ for lat in range(0,len(diff_pytools_cdo_timavg[0])):
+ for lon in range(0,len(diff_pytools_cdo_timavg[0][0])):
+ print(f'lat={lat},lon={lon}')
+ print(f'diff_pytools_cdo_timavg[0][lat][lon]={diff_pytools_cdo_timavg[0][lat][lon]}')
+ if lon>10: break
+ break
+
+ non_zero_count=np.count_nonzero(diff_pytools_cdo_timavg[:])
+ assert not( (non_zero_count > 0.) or (non_zero_count < 0.) )
+
+
+def test_compare_unwgt_fre_cli_to_unwgt_cdo():
+ ''' compares fre_cli pkg answer to cdo pkg answer '''
+ import numpy as np
+ import netCDF4 as nc
+ fre_pytools_inf=nc.Dataset(str_unwgt_fre_pytools_inf,'r')
+
+ try:
+ cdo_inf=nc.Dataset(str_unwgt_cdo_inf,'r')
+ except:
+ print('cdo input file not found. run cdo tests first.')
+ assert False
+
+ fre_pytools_timavg=fre_pytools_inf[var][:].copy()
+ cdo_timavg=cdo_inf[var][:].copy()
+ assert all([ len(fre_pytools_timavg)==len(cdo_timavg),
+ len(fre_pytools_timavg[0])==len(cdo_timavg[0]),
+ len(fre_pytools_timavg[0][0])==len(cdo_timavg[0][0]) ])
+
+ diff_pytools_cdo_timavg=fre_pytools_timavg-cdo_timavg
+ for lat in range(0,len(diff_pytools_cdo_timavg[0])):
+ for lon in range(0,len(diff_pytools_cdo_timavg[0][0])):
+ print(f'lat={lat},lon={lon}')
+ print(f'diff_pytools_cdo_timavg[0][lat][lon]={diff_pytools_cdo_timavg[0][lat][lon]}')
+ if lon>10: break
+ break
+
+ non_zero_count=np.count_nonzero(diff_pytools_cdo_timavg[:])
+ assert not( (non_zero_count > 0.) or (non_zero_count < 0.) )
+
+@pytest.mark.skip(reason='test fails b.c. cdo cannot bitwise-reproduce fre-nctools answer')
+def test_compare_cdo_to_fre_nctools():
+ ''' compares cdo pkg answer to fre_nctools pkg answer '''
+ import numpy as np
+ import netCDF4 as nc
+ cdo_inf=nc.Dataset(str_cdo_inf,'r')
+
+ try:
+ fre_nctools_inf=nc.Dataset(str_fre_nctools_inf,'r')
+ except:
+ print('fre-nctools input file not found. \
+ probably because you are not at GFDL! run the shell script \
+ example if you would like to see this pass. otherwise, \
+ i will error right after this message.')
+ alt_str_fre_nctools_inf = \
+ 'tests/time_avg_test_files/fre_nctools_timavg_CLI_test_r8_b_atmos_LWP_1979_5y.nc'
+ try:
+ fre_nctools_inf=nc.Dataset(alt_str_fre_nctools_inf,'r')
+ except:
+ print('fre-nctools output does not exist. create it first!')
+ assert False
+
+ cdo_timavg=cdo_inf[var][:].copy()
+ fre_nctools_timavg=fre_nctools_inf[var][:].copy()
+ assert all([ len(cdo_timavg)==len(fre_nctools_timavg),
+ len(cdo_timavg[0])==len(fre_nctools_timavg[0]),
+ len(cdo_timavg[0][0])==len(fre_nctools_timavg[0][0]) ])
+
+ diff_cdo_nctools_timavg=cdo_timavg-fre_nctools_timavg
+ for lat in range(0,len(diff_cdo_nctools_timavg[0])):
+ for lon in range(0,len(diff_cdo_nctools_timavg[0][0])):
+ print(f'lat={lat},lon={lon}')
+ print(f'diff_cdo_nctools_timavg[0][lat][lon]={diff_cdo_nctools_timavg[0][lat][lon]}')
+ if lon>10: break
+ break
+
+ non_zero_count=np.count_nonzero(diff_cdo_nctools_timavg[:])
+ assert not( (non_zero_count > 0.) or (non_zero_count < 0.) )
diff --git a/lib/fre/app/generate_time_averages/tests/time_avg_test_files/README.md b/lib/fre/app/generate_time_averages/tests/time_avg_test_files/README.md
new file mode 100644
index 00000000..e25c76f4
--- /dev/null
+++ b/lib/fre/app/generate_time_averages/tests/time_avg_test_files/README.md
@@ -0,0 +1 @@
+this directory is for tests/examples/sample files, etc.
diff --git a/lib/fre/app/generate_time_averages/tests/time_avg_test_files/atmos.197901-198312.LWP.nc b/lib/fre/app/generate_time_averages/tests/time_avg_test_files/atmos.197901-198312.LWP.nc
new file mode 100644
index 00000000..e2aa57b2
Binary files /dev/null and b/lib/fre/app/generate_time_averages/tests/time_avg_test_files/atmos.197901-198312.LWP.nc differ
diff --git a/lib/fre/app/generate_time_averages/tests/time_avg_test_files/frenctools_timavg_atmos.197901-198312.LWP.nc b/lib/fre/app/generate_time_averages/tests/time_avg_test_files/frenctools_timavg_atmos.197901-198312.LWP.nc
new file mode 100644
index 00000000..049d668e
Binary files /dev/null and b/lib/fre/app/generate_time_averages/tests/time_avg_test_files/frenctools_timavg_atmos.197901-198312.LWP.nc differ
diff --git a/lib/fre/app/generate_time_averages/timeAverager.py b/lib/fre/app/generate_time_averages/timeAverager.py
new file mode 100644
index 00000000..803febfe
--- /dev/null
+++ b/lib/fre/app/generate_time_averages/timeAverager.py
@@ -0,0 +1,68 @@
+''' core class structure for this module.'''
+
+class timeAverager:
+ '''
+ abstract base class for generating time averages + related statistical quantities
+ this class must be inherited by another for functionality.
+ '''
+ pkg: str
+ var: str
+ unwgt: bool
+ avg_type: str
+ stddev_type: str
+
+ def __init__(self):
+ ''' init method 1, no inputs given '''
+ self.pkg = None
+ self.var = None
+ self.unwgt = False
+ self.avg_type = "all" #see argparser for options
+ self.stddev_type = None #see argparser for options
+
+ def __init__(self, pkg, var, unwgt,
+ avg_type, stddev_type):
+ ''' init method 2, all inputs specified '''
+ self.pkg = pkg
+ self.var = var
+ self.unwgt = unwgt
+ self.avg_type = avg_type
+ self.stddev_type = stddev_type
+
+ def __repr__(self):
+ ''' return text representation of object '''
+ return f'{type(self).__name__}( pkg={self.pkg}, \
+ unwgt={self.unwgt}, \
+ var={self.var}, \
+ avg_type={self.avg_type}, \
+ stddev_type={self.stddev_type})'
+
+ def var_has_time_units(self, an_nc_var=None):
+ ''' checks if variable's units are of time '''
+ try:
+ var_units=an_nc_var.units
+ units_is_time = False
+ units_is_time = any( [ var_units == 'seconds' , 'seconds since' in var_units ,
+ var_units == 'minutes' , 'minutes since' in var_units ,
+ var_units == 'hours' , 'hours since' in var_units ,
+ var_units == 'days' , 'days since' in var_units ,
+ var_units == 'months' , 'months since' in var_units ,
+ var_units == 'years' , 'years since' in var_units ] )
+ return units_is_time
+ except:
+ print('variable does not have units')
+ print('PROBABLY not time.')
+ return False
+
+ #def var_has_time_dims(self, an_nc_var=None):
+ #try:
+ # var_dims=an_nc_var.dimensions
+ #for dim in an_nc_var.dimensions:
+ # if dim ==
+
+
+
+
+ def generate_timavg(self, infile=None, outfile=None):
+ '''# this is a hint: this is to be defined by classes inheriting from the abstract one
+ this function is never to be fully defined here by design.'''
+ raise NotImplementedError()
diff --git a/lib/fre/app/mask_atmos_plevel.py b/lib/fre/app/mask_atmos_plevel.py
new file mode 100644
index 00000000..0e7b2c28
--- /dev/null
+++ b/lib/fre/app/mask_atmos_plevel.py
@@ -0,0 +1,146 @@
+#!/usr/bin/env python
+'''
+This script contains the refineDiags that produce data at the same
+frequency as the input data (no reduction) such as surface albedo,
+masking fields,...
+It can accept any file and will only compute refineDiags in fields
+are present.
+'''
+
+import os
+import netCDF4 as nc
+import xarray as xr
+import click
+
+@click.command()
+def mask_atmos_plevel_subtool(infile, outfile, psfile):
+ ''' click entry point to fre cmor mask-atmos-plevel'''
+
+ # Error if outfile exists
+ if os.path.exists(outfile):
+ raise FileExistsError(f"ERROR: Output file {outfile} already exists")
+
+ # Open ps dataset
+ if not os.path.exists(psfile):
+ raise FileNotFoundError(f"ERROR: Input surface pressure file {psfile} does not exist")
+ ds_ps = xr.open_dataset(psfile)
+
+ # Exit with message if "ps" not available
+ if "ps" not in list(ds_ps.variables):
+ raise ValueError(f"ERROR: File {infile} does not contain surface pressure. exit.")
+
+ # Open input dataset
+ if not os.path.exists(infile):
+ raise FileNotFoundError(f"ERROR: Input file {infile} does not exist")
+ ds_in = xr.open_dataset(infile)
+
+ # The trigger for atmos masking is a variable attribute "needs_atmos_masking = True".
+ # In the future this will be set within the model, but for now and testing,
+ # we'll add the attribute for variables that end with "_unmsk".
+ ds_in = preprocess(ds_in)
+
+ ds_out = xr.Dataset()
+
+ # Process all variables with attribute "needs_atmos_masking = True"
+ for var in list(ds_in.variables):
+ if 'needs_atmos_masking' in ds_in[var].attrs:
+ del ds_in[var].attrs['needs_atmos_masking']
+ ds_out[var] = mask_field_above_surface_pressure(ds_in, var, ds_ps)
+ else:
+ continue
+
+ # Write the output file if anything was done
+ if ds_out.variables:
+ print(f"Modifying variables: {list(ds_out.variables)}\n appending into new file {outfile}")
+ write_dataset(ds_out, ds_in, outfile)
+ else:
+ print(f"No variables modified, so not writing output file {outfile}")
+
+
+
+def preprocess(ds):
+ """add needs_atmos_masking attribute if var ends with _unmsk"""
+ for var in list(ds.variables):
+ if var.endswith('_unmsk'):
+ ds[var].attrs['needs_atmos_masking'] = True
+ return ds
+
+
+def mask_field_above_surface_pressure(ds, var, ds_ps):
+ """mask data with pressure larger than surface pressure"""
+ plev = pressure_coordinate(ds, var)
+
+ # broadcast pressure coordinate and surface pressure to
+ # the dimensions of the variable to mask
+ plev_extended, _ = xr.broadcast(plev, ds[var])
+ ps_extended, _ = xr.broadcast(ds_ps["ps"], ds[var])
+
+ # masking do not need looping
+ masked = xr.where(plev_extended > ps_extended, 1.0e20, ds[var])
+
+ # copy attributes, but it doesn't include the missing values
+ attrs = ds[var].attrs.copy()
+
+ # add the missing values back
+ attrs['missing_value'] = 1.0e20
+ attrs['_FillValue'] = 1.0e20
+ masked.attrs = attrs
+
+ # transpose dims like the original array
+ masked = masked.transpose(*ds[var].dims)
+ print(f"Processed {var}")
+
+ return masked
+
+
+def pressure_coordinate(ds, varname):#, verbose=False):
+ """check if dataArray has pressure coordinate fitting requirements
+ and return it"""
+
+ pressure_coord = None
+ for dim in list(ds[varname].dims):
+ if dim in list(ds.variables): # dim needs to have values in file
+ if ds[dim].attrs["long_name"] == "pressure":
+ pressure_coord = ds[dim]
+ elif ("coordinates" in ds.attrs) and (ds[dim].attrs["units"] == "Pa"):
+ pressure_coord = ds[dim]
+ return pressure_coord
+
+
+def write_dataset(ds, template, outfile):
+ """prepare the dataset and dump into netcdf file"""
+
+ # copy global attributes
+ ds.attrs = template.attrs.copy()
+
+ # copy all variables and their attributes
+ # except those already processed
+ for var in list(template.variables):
+ if var in list(ds.variables):
+ continue
+ ds[var] = template[var]
+ ds[var].attrs = template[var].attrs.copy()
+ # write to file
+ ds.to_netcdf(outfile, unlimited_dims="time")
+
+
+
+def set_netcdf_encoding(ds, pressure_vars):
+ """set preferred options for netcdf encoding"""
+ all_vars = list(ds.variables)
+ encoding = {}
+ #for var in do_not_encode_vars + pressure_vars: #what was here in first place
+ for var in pressure_vars: #remove unused variable
+ if var in all_vars:
+ encoding.update( { var :
+ { '_FillValue':None} } )
+ return encoding
+
+
+def post_write(filename, var_with_bounds, bounds_variables):
+ """fix a posteriori attributes that xarray.to_netcdf
+ did not do properly using low level netcdf lib"""
+ f = nc.Dataset(filename, "a")
+ for var, bndvar in zip(var_with_bounds, bounds_variables):
+ f.variables[var].setncattr("bounds", bndvar)
+ f.close()
diff --git a/lib/fre/app/regrid_xy/README.md b/lib/fre/app/regrid_xy/README.md
new file mode 100644
index 00000000..30bf9b22
--- /dev/null
+++ b/lib/fre/app/regrid_xy/README.md
@@ -0,0 +1,55 @@
+# ABOUT
+
+`regrid_xy.py` remaps scalar and/or vector fields from one kind of lat/lon grid to another. It can remap between different grids of the same type (e.g. spherical), and between grids of different types (e.g. spherical to tripolar). By default, it uses an O(1) conservative interpolation scheme to accomplish the regridding, except under certain conditions [defined within `fregrid`](https://github.com/NOAA-GFDL/FRE-NCtools/blob/master/tools/fregrid/fregrid.c#L915-L920) the underlying CLI tool which does the heavy lifting.
+
+requires `fre-nctools` and `fregrid` to be in one's `PATH` variable, and `python3` (tested/developed with python 3.9.16). there should be `netCDF4` and `metomi` python modules in one's python environment for imports. `pytest` and `nccmp` is required for tests. `pylint` recommended for future developers working on this tool.
+
+
+# INPUT PARAMETERS (mandatory, env vars)
+format here is:
+config field name / python variable name (type) explanation
+
+the following are required to be specified:
+_______________________________________
+`inputDir` / `input_dir` (env var) specifies input directory to regrid, typically an untarredv history file archive
+_______________________________________
+`source` / `source` (env var) source name for input target file name within input directory to target for regridding. the value for `source` must be present in at least one component's configuration fields
+_______________________________________
+`begin` / `begin` (env var) ISO8601 datetime format specification for starting date of data, part of input target file name
+_______________________________________
+`outputDir` / `output_dir` (env var) specifies target location for output regridded files
+_______________________________________
+`TMPDIR` / `tmp_dir` (env var) temp directory for location of file read/writes
+_______________________________________
+`fregridRemapDir` / `remap_dir` (env var) directory containing remap file for regridding
+_______________________________________
+`gridSpec` / `grid_spec` (env var) file containing mosaic for regridding
+_______________________________________
+`defaultxyInterp` / `def_xy_interp` (env var) default lat/lon resolution for output regridding. (change me? TODO)
+
+
+# INPUT PARAMETERS (configuration fields, mandatory)
+the following parameters are REQUIRED to be specified on a per-component basis wtihin `app/regrid-xy/rose-app.conf`. A component's input parameters are delineated with a `[component_name]`
+_______________________________________
+`inputRealm` / `input_realm` (config field) realm within model from which the input component/source files are derived
+_______________________________________
+`interpMethod` / `interp_method` (config field) interpolation method to use for regridding, it may be changed if it is specified in the target source file's attributes
+_______________________________________
+`inputGrid` / `input_grid` (config field) current grid type of input source files.
+_______________________________________
+`sources` / N/A (config field)
+
+# INPUT PARAMETERS (configuration fields, optional)
+the following parameters are OPTIONAL for specifying on a per-component basis within `app/regrid-xy/rose-app.conf`.
+_______________________________________
+`outputGridType` / `output_grid_type` (config field) used only for output dir to specify grid type, but does not determine actual output grid type for regridding to fregrid
+_______________________________________
+`fregridRemapFile` / `fregrid_remap_file` (config field) remap file name to use for regridding
+_______________________________________
+`fregridMoreOptions` / `more_options` (config field) field for specifying additional options to `fregrid` that have not-yet been officially implemented. use with caution!
+_______________________________________
+`variables` / `regrid_vars` (config field) list of variable data to regrid within target source files. if unspecified, all variables within the target file of dimension 2 or greater will be regridded.
+_______________________________________
+`outputGridLat` / `output_grid_lat` (config field) latitude resolution for regridded output, also used for remap file targeting if there is no remap file specified.
+_______________________________________
+`outputGridLon` / `output_grid_lon` (config field) latitude resolution for regridded output, also used for remap file targeting if there is no remap file specified.
diff --git a/lib/fre/app/regrid_xy/__init__.py b/lib/fre/app/regrid_xy/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/app/regrid_xy/regrid_xy.py b/lib/fre/app/regrid_xy/regrid_xy.py
new file mode 100644
index 00000000..7b363e32
--- /dev/null
+++ b/lib/fre/app/regrid_xy/regrid_xy.py
@@ -0,0 +1,483 @@
+"""
+ remaps scalar and/or vector fields. It is capable of remapping
+ from a spherical grid onto a different one, e.g. spherical,
+ or tripolar. By default, it does so using a conservative scheme.
+ most valid input args to fregrid are valid in this script
+"""
+
+import subprocess
+import shutil
+import os
+from pathlib import Path
+
+#3rd party
+import metomi.rose.config as rose_cfg
+from netCDF4 import Dataset
+import click
+
+FREGRID_SHARED_FILES='/home/fms/shared_fregrid_remap_files'
+
+def truncate_date(date, freq):
+ """ truncates iso freq to iso date time """
+ freq_in_date_format=freq_to_date_format(freq)
+
+ #in the shell version, this line simply gets run.
+ #we will simply print this command to screen for now. TO DO (maybe we can work around it?)
+ #not clear to me why piping to tr is necessary, doesnt seem to change output at all
+ #print('cylc date --template '+freq_in_date_format+' '+date+' | tr -d T')
+ #output =subprocess.Popen(["cylc", "date", "--template", freq_in_date_format, date,
+ # "|","tr","-d","T"],
+ # stdout=subprocess.PIPE)
+ output =subprocess.Popen(["cylc", "cycle-point", "--template", freq_in_date_format, date],
+ stdout=subprocess.PIPE)
+ bytedate = output.communicate()[0]
+ date=str(bytedate.decode())
+
+ #remove trailing newline
+ date=date[:(len(date)-1)]
+
+ #check for and remove 'T' if present
+ if not date.isnumeric():
+ date=date[:8]+date[-2:]
+ return date
+
+def freq_to_date_format(iso_freq):
+ """
+ Print legacy Bronx-like date template format given a frequency (ISO 8601 duration)
+ """
+ if iso_freq=='P1Y':
+ return 'CCYY'
+ if iso_freq=='P1M':
+ return 'CCYYMM'
+ if iso_freq=='P1D':
+ return 'CCYYMMDD'
+ if (iso_freq[:2]=='PT') and (iso_freq[-1:]=='H'):
+ return 'CCYYMMDDThh'
+ raise ValueError(f'ERROR: Unknown Frequency {iso_freq}')
+
+def test_import():
+ """for quickly testing import within pytest"""
+ return 1
+
+def safe_rose_config_get(config, section, field):
+ """read optional variables from rose configuration, and don't error on None value"""
+ config_dict = config.get( [section,field] )
+ return None if config_dict is None else config_dict.get_value()
+
+
+def get_mosaic_file_name(grid_spec_file, mosaic_type):
+ """read string from a numpy masked array WHY"""
+ grid_spec_nc = Dataset(grid_spec_file, 'r')
+ masked_data = grid_spec_nc[mosaic_type][:].copy() # maskedArray
+ grid_spec_nc.close()
+ unmasked_data = masked_data[~masked_data.mask]
+ file_name = ''.join([ one_char.decode() for one_char in unmasked_data ])
+ return file_name
+
+
+def get_mosaic_grid_file_name(input_mosaic):
+ """get mosaic grid file name from NESTED numpy masked array WHY"""
+ input_mosaic_nc = Dataset(input_mosaic,'r')
+ masked_data = input_mosaic_nc['gridfiles'][0].copy()
+ input_mosaic_nc.close()
+ unmasked_data = masked_data[~masked_data.mask]
+ grid_file_name = ''.join([ one_char.decode() for one_char in unmasked_data ])
+ return grid_file_name
+
+
+def check_interp_method( nc_variable, interp_method):
+ """print warning if optional interp_method clashes with nc file attribute field, if present"""
+ attr_list=nc_variable.ncattrs()
+ if 'interp_method' not in attr_list:
+ pass
+ elif nc_variable.interp_method != interp_method:
+ print(f'WARNING: interp_method={interp_method} differs from what is in target_file')
+ print(f'WARNING: interp_method used may not be {interp_method}!')
+
+
+def check_per_component_settings(component_list, rose_app_cfg):
+ """for a source file ref'd by multiple components check per-component
+ settings for uniqueness. output list of bools of same length to check
+ in componenet loop"""
+ do_regridding = [True] #first component will always be run
+ curr_out_grid_type_list = [safe_rose_config_get( \
+ rose_app_cfg, component_list[0], 'outputGridType')]
+ for i in range( 1, len(component_list) ):
+ next_comp=component_list[i]
+ next_out_grid_type=safe_rose_config_get( rose_app_cfg, next_comp, 'outputGridType')
+ if next_out_grid_type not in curr_out_grid_type_list:
+ do_regridding.append(True)
+ curr_out_grid_type_list.append(next_out_grid_type)
+ else:
+ do_regridding.append(False)
+ if len(do_regridding) != len(component_list) :
+ raise ValueError('problem with checking per-component settings for uniqueness')
+ return do_regridding
+
+
+
+
+def make_component_list(config, source):
+ """make list of relevant component names where source file appears in sources"""
+ comp_list=[] #will not contain env, or command
+ try:
+ for keys, sub_node in config.walk():
+ sources = sub_node.get_value(keys=['sources'])#.split(' ')
+ if any( [ len(keys) < 1,
+ keys[0] in ['env', 'command'],
+ keys[0] in comp_list,
+ sources is None ] ):
+ continue
+ sources = sources.split(' ')
+ if source in sources:
+ comp_list.append(keys[0])
+ except Exception as exc:
+ raise ValueError(f'config = {config} may be an empty file... check the config') \
+ from exc
+ return comp_list
+
+
+def make_regrid_var_list(target_file, interp_method = None):
+ """create default list of variables to be regridded within target file."""
+ fin = Dataset(target_file,'r')
+ all_fin_vars = fin.variables
+ regrid_vars = []
+ for var_name in all_fin_vars:
+ if var_name in ['average_T1','average_T2',
+ 'average_DT','time_bnds' ]:
+ continue
+ if len(all_fin_vars[var_name].shape) < 2 :
+ continue
+ regrid_vars.append(var_name)
+ if interp_method is not None:
+ check_interp_method( all_fin_vars[var_name] , interp_method)
+
+ fin.close()
+ return regrid_vars
+
+
+def regrid_xy(input_dir = None, output_dir = None, begin = None, tmp_dir = None,
+ remap_dir = None, source = None, grid_spec = None, def_xy_interp = None
+ ):
+ """
+ calls fre-nctools' fregrid to regrid net cdf files
+ """
+
+ ## rose config load check
+ config_name = os.getcwd() #REMOVE ME TODO
+ config_name += '/rose-app-run.conf'
+ #config_name += '/rose-app.conf'
+ print(f'config_name = {config_name}')
+ try:
+ rose_app_config = rose_cfg.load(config_name)
+ except Exception as exc:
+ raise ValueError(f'config_name = {config_name} not found.') \
+ from exc
+
+
+ # mandatory arguments- code exits if any of these are not present
+ #input_dir = os.getenv( 'inputDir' )
+ #output_dir = os.getenv( 'outputDir' )
+ #begin = os.getenv( 'begin' )
+ #tmp_dir = os.getenv( 'TMPDIR' )
+ #remap_dir = os.getenv( 'fregridRemapDir' )
+ #source = os.getenv( 'source' )
+ #grid_spec = os.getenv( 'gridSpec' )
+ #def_xy_interp = os.getenv( 'defaultxyInterp' )
+ if None in [ input_dir , output_dir ,
+ begin , tmp_dir ,
+ remap_dir , source ,
+ grid_spec , def_xy_interp ]:
+ print( f'input_dir = { input_dir }\n' + \
+ f'output_dir = { output_dir }\n' + \
+ f'begin = { begin }\n' + \
+ f'tmp_dir = { tmp_dir }\n' + \
+ f'remap_dir = { remap_dir }\n' + \
+ f'source = { source }\n' + \
+ f'grid_spec = { grid_spec }\n' + \
+ f'def_xy_interp = { def_xy_interp }' )
+ raise ValueError(f'a mandatory input argument to regrid_xy is None... \n {locals()}')
+
+
+ def_xy_interp = def_xy_interp.split(',')
+ def_xy_interp[0] = def_xy_interp[0].replace('"', '')
+ def_xy_interp[1] = def_xy_interp[1].replace('"', '')
+ if any( [ def_xy_interp == [] or len(def_xy_interp) != 2 ] ):
+ raise ValueError(
+ f'default xy interpolation has invalid format: \n def_xy_interp = {def_xy_interp}')
+
+ # input dir must exist
+ if not Path( input_dir ).exists():
+ raise OSError(f'input_dir={input_dir} \n does not exist')
+
+ # tmp_dir check
+ if not Path( tmp_dir ).exists():
+ raise OSError(f'tmp_dir={tmp_dir} \n does not exist.')
+
+ # output dir check
+ Path( output_dir ).mkdir( parents = True, exist_ok = True )
+ if not Path( output_dir ).exists() :
+ raise OSError('the following does not exist and/or could not be created:' +
+ f'output_dir=\n{output_dir}')
+
+ # work/ dir check
+ work_dir = tmp_dir + 'work/'
+ Path( work_dir ).mkdir( exist_ok = True )
+ if not Path( work_dir ).exists():
+ raise OSError('the following does not exist and/or could not be created:' +
+ f'work_dir=\n{work_dir}')
+
+
+ # fregrid remap dir check
+ Path(remap_dir).mkdir( exist_ok = True )
+ if not Path( remap_dir ).exists():
+ raise OSError(f'{remap_dir} could not be created')
+
+
+ # grid_spec file management
+ if '.tar' in grid_spec:
+ untar_sp = \
+ subprocess.run( ['tar', '-xvf', grid_spec, '-C', input_dir],
+ check = False , capture_output = True )
+ if untar_sp.returncode != 0:
+ raise OSError(
+ f'untarring of {grid_spec} file failed, ' + \
+ f'ret_code={untar_sp.returncode}, stderr={untar_sp.stderr}')
+ if Path( input_dir+'mosaic.nc' ).exists():
+ grid_spec_file=input_dir+'mosaic.nc'
+ elif Path( input_dir+'grid_spec.nc' ).exists():
+ grid_spec_file=input_dir+'grid_spec.nc'
+ else:
+ raise ValueError(f'grid_spec_file cannot be determined from grid_spec={grid_spec}')
+ else:
+ try: #attempt to copy directly from uncompressed grid_spec location
+ grid_spec_file=input_dir+grid_spec.split('/').pop()
+ shutil.copy(grid_spec, grid_spec_file )
+ except Exception as exc:
+ raise OSError(f'grid_spec={grid_spec} could not be copied.') \
+ from exc
+
+ # ------------------------------------------------------------------ component loop
+ component_list = make_component_list(rose_app_config, source)
+ if len(component_list) == 0:
+ raise ValueError('component list empty- source file not found in any source file list!')
+ if len(component_list) > 1: # check settings for uniqueness
+ do_regridding = \
+ check_per_component_settings( \
+ component_list, rose_app_config)
+ print(f'component_list = {component_list}')
+ print(f'do_regridding = {do_regridding}')
+ else:
+ do_regridding=[True]
+
+ for component in component_list:
+ if not do_regridding[
+ component_list.index(component) ]:
+ continue
+ print(f'\n\nregridding \nsource={source} for \ncomponent={component}\n')
+
+ # mandatory per-component inputs, will error if nothing in rose config
+ input_realm, interp_method, input_grid = None, None, None
+ try:
+ input_realm = rose_app_config.get( [component, 'inputRealm'] ).get_value()
+ interp_method = rose_app_config.get( [component, 'interpMethod'] ).get_value()
+ input_grid = rose_app_config.get( [component, 'inputGrid'] ).get_value()
+ except Exception as exc:
+ raise ValueError('at least one of the following are None: \n' + \
+ f'input_grid=\n{input_grid}\n,input_realm=\n' + \
+ f'{input_realm}\n,/interp_method=\n{interp_method}') \
+ from exc
+ print(f'input_grid = {input_grid }\n' + \
+ f'input_realm = {input_realm }\n' + \
+ f'interp_method = {interp_method }' )
+
+ #target input variable resolution
+ is_tiled = 'cubedsphere' in input_grid
+ target_file = input_dir
+ target_file += f"/{truncate_date(begin,'P1D')}.{source}.tile1.nc" \
+ if is_tiled \
+ else f"/{truncate_date(begin,'P1D')}.{source}.nc"
+ if not Path( target_file ).exists():
+ raise OSError(f'regrid_xy target does not exist. \ntarget_file={target_file}')
+ print(f'target_file={target_file}') #DELETE
+
+
+ # optional per-component inputs
+ output_grid_type = safe_rose_config_get( rose_app_config, component, 'outputGridType')
+ remap_file = safe_rose_config_get( rose_app_config, component, 'fregridRemapFile')
+ more_options = safe_rose_config_get( rose_app_config, component, 'fregridMoreOptions')
+ regrid_vars = safe_rose_config_get( rose_app_config, component, 'variables')
+ output_grid_lon = safe_rose_config_get( rose_app_config, component, 'outputGridLon')
+ output_grid_lat = safe_rose_config_get( rose_app_config, component, 'outputGridLat')
+
+ # if no input args specified, grab the defaul values
+ if any( [ output_grid_lon is None,
+ output_grid_lat is None ] ):
+ output_grid_lon = def_xy_interp[0]
+ output_grid_lat = def_xy_interp[1]
+
+ print( f'output_grid_type = {output_grid_type }\n' + \
+ f'remap_file = {remap_file }\n' + \
+ f'more_options = {more_options }\n' + \
+ f'output_grid_lon = {output_grid_lon }\n' + \
+ f'output_grid_lat = {output_grid_lat }\n' + \
+ f'regrid_vars = {regrid_vars }\n' )
+
+
+
+ # prepare to create input_mosaic via ncks call
+ if input_realm in ['atmos', 'aerosol']:
+ mosaic_type = 'atm_mosaic_file'
+ elif input_realm == 'ocean':
+ mosaic_type = 'ocn_mosaic_file'
+ elif input_realm == 'land':
+ mosaic_type = 'lnd_mosaic_file'
+ else:
+ raise ValueError(f'input_realm={input_realm} not recognized.')
+
+ # this is just to get the grid_file name
+ #print(f'mosaic_type = {mosaic_type}')
+ #print(f'grid_spec_file = {grid_spec_file}')
+ #print(f'input_mosaic = get_mosaic_file_name(grid_spec_file, mosaic_type)')
+
+ # assume input_mosaic near input grid_spec, where intially specified.
+ input_mosaic = input_dir + get_mosaic_file_name(grid_spec_file, mosaic_type)
+ #print(f'input_mosaic = {input_mosaic}')
+
+ ## this is to get the tile1 filename?
+ mosaic_grid_file = input_dir + get_mosaic_grid_file_name(input_mosaic)
+ #print(f'mosaic_grid_file = {mosaic_grid_file}')
+
+ # need source file dimenions for lat/lon
+ source_nx = str(int(Dataset(mosaic_grid_file).dimensions['nx'].size / 2 ))
+ source_ny = str(int(Dataset(mosaic_grid_file).dimensions['ny'].size / 2 ))
+ print(f'source_[nx,ny] = ({source_nx},{source_ny})')
+ Dataset(mosaic_grid_file).close()
+
+
+ if remap_file is not None:
+ try:
+ shutil.copy( remap_file,
+ input_dir+remap_file.split('/').pop() )
+ except Exception as exc:
+ raise OSError('remap_file={remap_file} could not be copied to local dir') \
+ from exc
+ else:
+ print('remap_file was not specified nor found. looking for default one')
+ remap_file= f'fregrid_remap_file_{output_grid_lon}_by_{output_grid_lat}.nc' \
+ if \
+ all( [output_grid_lon is not None,
+ output_grid_lat is not None]) \
+ else \
+ f'fregrid_remap_file_{def_xy_interp(0)}_by_{def_xy_interp(1)}.nc'
+ remap_cache_file = \
+ f'{remap_dir}/{input_grid}/{input_realm}/' + \
+ f'{source_nx}-by-{source_ny}/{interp_method}/{remap_file}'
+ central_remap_cache_file = \
+ f'{FREGRID_SHARED_FILES}/{input_grid}/' + \
+ f'{source_nx}_by_{source_ny}/{remap_file}'
+
+ print(f'remap_file = {remap_file }' + \
+ f'remap_cache_file = {remap_cache_file }' + \
+ f'central_remap_cache_file = {central_remap_cache_file}' )
+
+ if Path( remap_cache_file ).exists():
+ print(f'NOTE: using cached remap file {remap_cache_file}')
+ shutil.copy(remap_cache_file,
+ work_dir+remap_cache_file.split('/').pop())
+ elif Path( central_remap_cache_file ).exists():
+ print(f'NOTE: using centrally cached remap file {remap_cache_file}')
+ shutil.copy(central_remap_cache_file,
+ work_dir+central_remap_cache_file.split('/').pop())
+
+
+
+ # if no variables in config, find the interesting ones to regrid
+ if regrid_vars is None:
+ regrid_vars=make_regrid_var_list( target_file , interp_method)
+ print(f'regrid_vars = {regrid_vars}') #DELETE
+
+ #check if there's anything worth regridding
+ if len(regrid_vars) < 1:
+ raise ValueError('make_regrid_var_list found no vars to regrid. and no vars given. exit')
+ print(f'regridding {len(regrid_vars)} variables: {regrid_vars}')
+ regrid_vars_str=','.join(regrid_vars) # fregrid needs comma-demarcated list of vars
+
+
+
+ # massage input file argument to fregrid.
+ input_file = target_file.replace('.tile1.nc','') \
+ if '.tile1' in target_file \
+ else target_file
+ input_file=input_file.split('/').pop()
+
+ # create output file argument...
+ output_file = target_file.replace('.tile1','') \
+ if 'tile1' in target_file \
+ else target_file
+ output_file = work_dir + output_file.split('/').pop()
+
+ fregrid_command = [
+ 'fregrid',
+ '--debug',
+ '--standard_dimension',
+ '--input_mosaic', f'{input_mosaic}',
+ '--input_dir', f'{input_dir}',
+ '--input_file', f'{input_file}',
+ '--associated_file_dir', f'{input_dir}',
+ '--interp_method', f'{interp_method}',
+ '--remap_file', f'{remap_file}',
+ '--nlon', f'{str(output_grid_lon)}',
+ '--nlat', f'{str(output_grid_lat)}',
+ '--scalar_field', f'{regrid_vars_str}',
+ '--output_file', f'{output_file}']
+ if more_options is not None:
+ fregrid_command.append(f'{more_options}')
+
+ print(f"\n\nabout to run the following command: \n{' '.join(fregrid_command)}\n")
+ fregrid_proc = subprocess.run( fregrid_command, check = False )#i hate it
+ fregrid_rc =fregrid_proc.returncode
+ print(f'fregrid_result.returncode()={fregrid_rc}')
+
+
+ # output wrangling
+
+ # copy the remap file to the cache location
+ if not Path( remap_cache_file ).exists():
+ remap_cache_file_dir='/'.join(remap_cache_file.split('/')[0:-1])
+ Path( remap_cache_file_dir ).mkdir( parents = True , exist_ok = True)
+ print(f'copying \nremap_file={remap_file} to')
+ print(f'remap_cache_file_dir={remap_cache_file_dir}')
+ shutil.copy(remap_file, remap_cache_file_dir)
+
+ # more output wrangling
+ final_output_dir = output_dir \
+ if output_grid_type is None \
+ else output_dir + '/' + output_grid_type
+ Path( final_output_dir ).mkdir( exist_ok = True)
+
+ print(f'TRYING TO COPY {output_file} TO {final_output_dir}')
+ shutil.copy(output_file, final_output_dir)
+
+ continue # end of comp loop, exit or next one.
+
+ print('done running regrid_xy()')
+ return 0
+
+
+@click.command()
+def _regrid_xy( input_dir, output_dir, begin, tmp_dir,
+ remap_dir, source, grid_spec, def_xy_interp ):
+ """ click entrypoint """
+ click.echo(f'(_regrid_xy) locals={locals()}')
+ click.echo( '(_regrid_xy) click entrypoint hit- calling regrid_xy()')
+ return regrid_xy( input_dir, output_dir, begin, tmp_dir,
+ remap_dir, source, grid_spec, def_xy_interp )
+
+
+def main():
+ """steering, local test/debug"""
+ return regrid_xy()
+
+if __name__=='__main__':
+ main()
diff --git a/lib/fre/app/regrid_xy/tests/test_inputs_outputs/input_directory.tar.gz b/lib/fre/app/regrid_xy/tests/test_inputs_outputs/input_directory.tar.gz
new file mode 100644
index 00000000..003d654a
Binary files /dev/null and b/lib/fre/app/regrid_xy/tests/test_inputs_outputs/input_directory.tar.gz differ
diff --git a/lib/fre/app/regrid_xy/tests/test_regrid_xy.py b/lib/fre/app/regrid_xy/tests/test_regrid_xy.py
new file mode 100644
index 00000000..fa511b66
--- /dev/null
+++ b/lib/fre/app/regrid_xy/tests/test_regrid_xy.py
@@ -0,0 +1,494 @@
+""" test regrid_xy functioning as an importable python module """
+from pathlib import Path
+import subprocess
+import os
+
+import pytest
+
+import fre.app.regrid_xy.regrid_xy as rgxy
+
+# directories for tests
+TEST_DIR = os.getcwd() + "/fre/app/regrid_xy/tests/test_inputs_outputs/"
+
+TAR_IN_DIR = TEST_DIR + 'input_directory.tar.gz' #contains in-dir
+IN_DIR = TEST_DIR + 'in-dir/'
+WORK_DIR = TEST_DIR + 'work/'
+
+TEST_OUT_DIR = TEST_DIR + 'out-dir/'
+REMAP_DIR = TEST_DIR + 'remap-dir/'
+
+ALL_TEST_OUT_DIR = TEST_DIR + "out-test-dir/"
+REMAP_TEST_DIR = TEST_DIR + 'remap-test-dir/'
+
+# official data/mosaic stuff
+GOLD_GRID_SPEC = "/archive/gold/datasets/OM4_05/mosaic_c96.v20180227.tar"
+GOLD_GRID_SPEC_NO_TAR = IN_DIR + "mosaic.nc"
+TEST_CDL_GRID_FILE = IN_DIR + "C96_mosaic.cdl"
+
+
+## for rose configuration
+COMPONENT='atmos'
+NLON=288
+NLAT=180
+INPUT_GRID='cubedsphere'
+INPUT_REALM='atmos'
+INTERP_METHOD='conserve_order1'
+YYYYMMDD='20030101'
+SOURCE='atmos_static_cmip' # generally a list, but for tests, only one
+ # note that components will not always contain a
+ # a source file of the same name
+SOURCES_XY = '96-by-96'
+
+WORK_YYYYMMDD_DIR = WORK_DIR + f'{YYYYMMDD}.nc/'
+TEST_NC_GRID_FILE = WORK_YYYYMMDD_DIR + "C96_mosaic.nc" # output of first ncgen test
+
+def test_setup_clean_up(capfd):
+ """ cleanup i/o directories is present for clean regrid_xy testing """
+ try:
+ Path(IN_DIR).unlink()
+ except OSError:
+ pass
+ try:
+ Path(TEST_NC_GRID_FILE).unlink()
+ except OSError:
+ pass
+ try:
+ Path(WORK_DIR).unlink()
+ except OSError:
+ pass
+ try:
+ Path(TEST_OUT_DIR).unlink()
+ except OSError:
+ pass
+ try:
+ Path(ALL_TEST_OUT_DIR).unlink()
+ except OSError:
+ pass
+ try:
+ Path(REMAP_DIR).unlink()
+ except OSError:
+ pass
+ try:
+ Path(REMAP_TEST_DIR).unlink()
+ except OSError:
+ pass
+ assert True
+ out, err = capfd.readouterr()
+
+
+def test_setup_global_work_dirs(capfd):
+ """ create i/o directories for regrid_xy testing """
+ Path(WORK_YYYYMMDD_DIR).mkdir(parents = True, exist_ok = True)
+ assert Path(WORK_YYYYMMDD_DIR).exists()
+
+ Path(REMAP_TEST_DIR).mkdir(exist_ok = True)
+ assert Path(REMAP_TEST_DIR).exists()
+
+ Path(ALL_TEST_OUT_DIR).mkdir(exist_ok = True)
+ assert Path(ALL_TEST_OUT_DIR).exists()
+
+ Path(TEST_OUT_DIR).mkdir(exist_ok = True)
+ assert Path(TEST_OUT_DIR).exists()
+
+ Path(REMAP_DIR).mkdir(exist_ok = True)
+ assert Path(REMAP_DIR).exists()
+ out, err = capfd.readouterr()
+
+
+def test_untar_inputs(capfd):
+ """ untar input directory tarball to create test inputs """
+ ex = ["tar", "-C", TEST_DIR, "-zxvf", TAR_IN_DIR]
+ sp = subprocess.run( ex , check = True )
+ assert all ( [ sp.returncode == 0,
+ Path(IN_DIR).exists() ] )
+ out, err = capfd.readouterr()
+
+
+#@pytest.mark.skip(reason='debug')
+def test_make_ncgen3_nc_inputs(capfd):
+ """
+ set-up test: ncgen3 netcdf file inputs for later steps
+ if the output exists, it will not bother remaking it
+ """
+
+ ncgen3_output = TEST_NC_GRID_FILE
+ ncgen3_input = TEST_CDL_GRID_FILE
+ if Path(ncgen3_output).exists():
+ assert True
+ else:
+ assert Path(TEST_DIR).exists()
+ assert Path(ncgen3_input).exists()
+ ex = [ 'ncgen3', '-k', 'netCDF-4', '-o', ncgen3_output , ncgen3_input ]
+ print (' '.join(ex))
+ sp = subprocess.run( ex , check = True )
+
+ assert all( [ sp.returncode == 0,
+ Path(ncgen3_output).exists() ] )
+ out, err = capfd.readouterr()
+
+
+#@pytest.mark.skip(reason='debug')
+def test_make_ncgen_tile_nc_inputs(capfd):
+ """
+ set-up test: ncgen netcdf tile file inputs for later steps
+ if the output exists, it will not bother remaking it
+ """
+ ncgen_tile_i_nc_output_exists = [ Path( WORK_YYYYMMDD_DIR + \
+ f'{YYYYMMDD}.{SOURCE}.tile{i}.nc' ).exists() \
+ for i in range(1, 6+1) ]
+ if all( ncgen_tile_i_nc_output_exists ):
+ assert True
+ else:
+ for i in range(1, 6+1):
+ ncgen_tile_i_nc_output = WORK_YYYYMMDD_DIR + f'{YYYYMMDD}.{SOURCE}.tile{i}.nc'
+ ncgen_tile_i_cdl_input = IN_DIR + f'{YYYYMMDD}.{SOURCE}.tile{i}.cdl'
+ assert Path(ncgen_tile_i_cdl_input).exists()
+ ex = [ 'ncgen', '-o', ncgen_tile_i_nc_output, ncgen_tile_i_cdl_input ]
+ print (' '.join(ex))
+ sp = subprocess.run( ex , check = True )
+
+ assert all( [sp.returncode == 0,
+ Path(ncgen_tile_i_nc_output).exists()] )
+ out, err = capfd.readouterr()
+
+
+#@pytest.mark.skip(reason='debug')
+def test_make_ncgen_grid_spec_nc_inputs(capfd):
+ """
+ set-up test: ncgen netcdf grid spec tile file inputs for later steps
+ if the output exists, it will not bother remaking it
+ """
+ ncgen_grid_spec_i_nc_output_exists = [ Path( WORK_YYYYMMDD_DIR + \
+ f'{YYYYMMDD}.grid_spec.tile{i}.nc').exists() \
+ for i in range(1, 6+1) ]
+ if all( ncgen_grid_spec_i_nc_output_exists ):
+ assert True
+ else:
+ for i in range(1, 6+1):
+ ncgen_grid_spec_i_nc_output = WORK_YYYYMMDD_DIR + f'{YYYYMMDD}.grid_spec.tile{i}.nc'
+ ncgen_grid_spec_i_cdl_input = IN_DIR + f'{YYYYMMDD}.grid_spec.tile{i}.cdl'
+ ex = [ 'ncgen', '-o', ncgen_grid_spec_i_nc_output, ncgen_grid_spec_i_cdl_input ]
+ print (' '.join(ex))
+ sp = subprocess.run( ex , check = True )
+
+ assert all( [sp.returncode == 0,
+ Path(ncgen_grid_spec_i_nc_output).exists()] )
+ out, err = capfd.readouterr()
+
+
+#@pytest.mark.skip(reason='debug')
+def test_make_hgrid_gold_input(capfd):
+ """
+ set-up test: make C96 gold input via make_hgrid for later steps
+ if the output exists in the desired location, it will not bother remaking it
+ """
+ grid_i_file_targ_loc_exists = [ Path( WORK_YYYYMMDD_DIR + f'C96_grid.tile{i}.nc').exists() \
+ for i in range(1,6+1) ]
+ if all( grid_i_file_targ_loc_exists ):
+ assert True
+ else:
+ # remake gold grid file locally, then move
+ ex = [ "make_hgrid",
+ "--grid_type", "gnomonic_ed",
+ "--nlon", "192",
+ "--grid_name", "C96_grid" ]
+ print (' '.join(ex))
+ sp = subprocess.run( ex , check = True )
+
+ assert sp.returncode == 0
+ out, err = capfd.readouterr()
+
+ # now move the files...
+ for i in range(1, 6+1):
+ grid_i_file_curr_loc = f'C96_grid.tile{i}.nc'
+ grid_i_file_targ_loc = WORK_YYYYMMDD_DIR + f'C96_grid.tile{i}.nc'
+ ex = [ 'mv', '-f', grid_i_file_curr_loc, grid_i_file_targ_loc ]
+ print (' '.join(ex))
+ sp = subprocess.run( ex , check = True )
+
+ assert all( [sp.returncode == 0,
+ Path(grid_i_file_targ_loc).exists()] )
+ out, err = capfd.readouterr()
+
+
+#@pytest.mark.skip(reason='debug')
+def test_make_fregrid_comparison_input(capfd):
+ """
+ set-up test: use fregrid to regrid for later comparison to regrid_xy output
+ if the output exists in the desired location, it will not bother remaking it
+ """
+
+ fregrid_input_mosaic_arg = TEST_NC_GRID_FILE
+ fregrid_input_dir_arg = WORK_YYYYMMDD_DIR
+ fregrid_input_file_arg = f'{YYYYMMDD}.{SOURCE}'
+ fregrid_assoc_file_dir_arg = WORK_YYYYMMDD_DIR
+
+ fregrid_remap_file = f'fregrid_remap_file_{NLON}_by_{NLAT}.nc'
+ fregrid_remap_file_arg = REMAP_TEST_DIR + fregrid_remap_file
+
+ fregrid_nlat_arg = str(NLAT)
+ fregrid_nlon_arg = str(NLON)
+ fregrid_vars_arg = 'grid_xt,grid_yt,orog'
+
+
+ fregrid_output_file_arg = ALL_TEST_OUT_DIR + fregrid_input_file_arg + '.nc'
+
+ ex = [ 'fregrid', '--standard_dimension',
+ '--input_mosaic', fregrid_input_mosaic_arg,
+ '--input_dir', fregrid_input_dir_arg,
+ '--input_file', fregrid_input_file_arg,
+ '--associated_file_dir', fregrid_assoc_file_dir_arg,
+ '--remap_file', fregrid_remap_file_arg,
+ '--nlon', fregrid_nlon_arg,
+ '--nlat', fregrid_nlat_arg,
+ '--scalar_field', fregrid_vars_arg,
+ '--output_file', fregrid_output_file_arg ]
+ print (' \n'.join(ex))
+ sp = subprocess.run( ex , check = True )
+
+ assert all( [ sp.returncode == 0,
+ Path(fregrid_remap_file_arg).exists(),
+ Path(fregrid_output_file_arg).exists() ] )
+ out, err = capfd.readouterr()
+
+
+#@pytest.mark.skip(reason='debug')
+def test_import_regrid_xy(capfd):
+ """
+ check import of regrid_xy as a module
+ """
+ assert all( [ rgxy is not None,
+ rgxy.test_import() == 1 ] )
+ out, err = capfd.readouterr()
+
+#@pytest.mark.skip(reason='debug')
+def test_success_tar_grid_spec_regrid_xy(capfd):
+ """
+ checks for success of regrid_xy with rose app-app run
+ """
+ # this will only work at GFDL for now.
+ if not Path(GOLD_GRID_SPEC).exists():
+ assert True
+ else:
+ # for the time being, still a little dependent on rose for configuration value passing
+ if Path(os.getcwd()+'/rose-app-run.conf').exists():
+ Path(os.getcwd()+'/rose-app-run.conf').unlink()
+
+ with open(os.getcwd()+'/rose-app-run.conf','a',encoding='utf-8') as rose_app_run_config:
+ rose_app_run_config.write( '[command]\n' )
+ rose_app_run_config.write( 'default=regrid-xy\n' )
+ rose_app_run_config.write( '\n' )
+ rose_app_run_config.write( f'[{COMPONENT}]\n' )
+ rose_app_run_config.write( f'sources={SOURCE}\n' )
+ rose_app_run_config.write( f'inputGrid={INPUT_GRID}\n' )
+ rose_app_run_config.write( f'inputRealm={INPUT_REALM}\n' )
+ rose_app_run_config.write( f'interpMethod={INTERP_METHOD}\n' )
+ rose_app_run_config.write( f'outputGridLon={NLON}\n' )
+ rose_app_run_config.write( f'outputGridLat={NLAT}\n' )
+ rose_app_run_config.write( '\n' )
+ assert Path('./rose-app-run.conf').exists()
+
+ rgxy_returncode = rgxy.regrid_xy(
+ input_dir = WORK_YYYYMMDD_DIR,
+ output_dir = TEST_OUT_DIR,
+ begin = f'{YYYYMMDD}T000000',
+ tmp_dir = TEST_DIR,
+ remap_dir = REMAP_DIR,
+ source = SOURCE,
+ grid_spec = GOLD_GRID_SPEC,
+ def_xy_interp = f'"{NLON},{NLAT}"'
+ )
+
+ # uhm....
+ assert rgxy_returncode == 0
+ assert Path( REMAP_DIR + \
+ f'{INPUT_GRID}/{INPUT_REALM}/96-by-96/{INTERP_METHOD}/' + \
+ f'fregrid_remap_file_{NLON}_by_{NLAT}.nc' \
+ ).exists()
+ assert Path( TEST_OUT_DIR ).exists()
+ assert Path( TEST_OUT_DIR + f'{YYYYMMDD}.{SOURCE}.nc' ).exists()
+ assert Path( WORK_DIR ).exists()
+ assert Path( WORK_DIR + f'{YYYYMMDD}.{SOURCE}.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'basin_codes.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile2.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile3.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile4.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile5.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile6.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile1XC96_mosaic_tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile1Xocean_mosaic_tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile2XC96_mosaic_tile2.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile2Xocean_mosaic_tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile3XC96_mosaic_tile3.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile3Xocean_mosaic_tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile4XC96_mosaic_tile4.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile4Xocean_mosaic_tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile5XC96_mosaic_tile5.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile5Xocean_mosaic_tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile6XC96_mosaic_tile6.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic_tile6Xocean_mosaic_tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'hash.md5' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'land_mask_tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'land_mask_tile2.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'land_mask_tile3.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'land_mask_tile4.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'land_mask_tile5.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'land_mask_tile6.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'mosaic.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'ocean_hgrid.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'ocean_mask.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'ocean_mosaic.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'ocean_static.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'ocean_topog.nc' ).exists()
+ out, err = capfd.readouterr()
+ assert True
+
+
+
+
+#@pytest.mark.skip(reason='debug')
+def test_success_no_tar_grid_spec_regrid_xy(capfd):
+ """
+ checks for success of regrid_xy with rose app-app run
+ """
+ # for the time being, still a little dependent on rose for configuration value passing
+ if Path(os.getcwd()+'/rose-app-run.conf').exists():
+ Path(os.getcwd()+'/rose-app-run.conf').unlink()
+
+ with open(os.getcwd()+'/rose-app-run.conf','a',encoding='utf-8') as rose_app_run_config:
+ rose_app_run_config.write( '[command]\n' )
+ rose_app_run_config.write( 'default=regrid-xy\n' )
+ rose_app_run_config.write( '\n' )
+ rose_app_run_config.write( f'[{COMPONENT}]\n' )
+ rose_app_run_config.write( f'sources={SOURCE}\n' )
+ rose_app_run_config.write( f'inputGrid={INPUT_GRID}\n' )
+ rose_app_run_config.write( f'inputRealm={INPUT_REALM}\n' )
+ rose_app_run_config.write( f'interpMethod={INTERP_METHOD}\n' )
+ rose_app_run_config.write( f'outputGridLon={NLON}\n' )
+ rose_app_run_config.write( f'outputGridLat={NLAT}\n' )
+ rose_app_run_config.write( '\n' )
+ assert Path('./rose-app-run.conf').exists()
+
+ rgxy_returncode = rgxy.regrid_xy(
+ input_dir = WORK_YYYYMMDD_DIR,
+ output_dir = TEST_OUT_DIR,
+ begin = f'{YYYYMMDD}T000000',
+ tmp_dir = TEST_DIR,
+ remap_dir = REMAP_DIR,
+ source = SOURCE,
+ grid_spec = GOLD_GRID_SPEC_NO_TAR,
+ def_xy_interp = f'"{NLON},{NLAT}"'
+ )
+
+ # uhm....
+ assert rgxy_returncode == 0
+ assert Path( REMAP_DIR + \
+ f'{INPUT_GRID}/{INPUT_REALM}/96-by-96/{INTERP_METHOD}/' + \
+ f'fregrid_remap_file_{NLON}_by_{NLAT}.nc' \
+ ).exists()
+ assert Path( TEST_OUT_DIR ).exists()
+ assert Path( TEST_OUT_DIR + f'{YYYYMMDD}.{SOURCE}.nc' ).exists()
+ assert Path( WORK_DIR ).exists()
+ assert Path( WORK_DIR + f'{YYYYMMDD}.{SOURCE}.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR ).exists()
+
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile1.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile2.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile3.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile4.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile5.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_grid.tile6.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'C96_mosaic.nc' ).exists()
+ assert Path( WORK_YYYYMMDD_DIR + 'mosaic.nc' ).exists()
+ out, err = capfd.readouterr()
+
+
+
+
+@pytest.mark.skip(reason='debug')
+def test_failure_wrong_datetime_regrid_xy(capfd):
+ """
+ checks for failure of regrid_xy with rose app-run when fed an
+ invalid date for begin
+ """
+ # for the time being, still a little dependent on rose for configuration value passing
+ if Path(os.getcwd()+'/rose-app-run.conf').exists():
+ Path(os.getcwd()+'/rose-app-run.conf').unlink()
+
+ with open(os.getcwd()+'/rose-app-run.conf','a',encoding='utf-8') as rose_app_run_config:
+ rose_app_run_config.write( '[command]\n' )
+ rose_app_run_config.write( 'default=regrid-xy\n' )
+ rose_app_run_config.write( '\n' )
+ rose_app_run_config.write( f'[{COMPONENT}]\n' )
+ rose_app_run_config.write( f'sources={SOURCE}\n' )
+ rose_app_run_config.write( f'inputGrid={INPUT_GRID}\n' )
+ rose_app_run_config.write( f'inputRealm={INPUT_REALM}\n' )
+ rose_app_run_config.write( f'interpMethod={INTERP_METHOD}\n' )
+ rose_app_run_config.write( f'outputGridLon={NLON}\n' )
+ rose_app_run_config.write( f'outputGridLat={NLAT}\n' )
+ rose_app_run_config.write( '\n' )
+ assert Path('./rose-app-run.conf').exists()
+
+ try:
+ rgxy_returncode = rgxy.regrid_xy(
+ input_dir = WORK_YYYYMMDD_DIR,
+ output_dir = TEST_OUT_DIR,
+ begin = '99999999T999999',
+ tmp_dir = TEST_DIR,
+ remap_dir = REMAP_DIR,
+ source = SOURCE,
+ grid_spec = GOLD_GRID_SPEC,
+ def_xy_interp = f'"{NLON},{NLAT}"'
+ )
+ except:
+ # yay good job
+ assert True
+
+ out, err = capfd.readouterr()
+
+
+
+#@pytest.mark.skip(reason='debug')
+
+
+def test_nccmp1_regrid_xy(capfd):
+ """
+ This test compares the output of make_hgrid and fregrid, which are expected to be identical
+ """
+ fregrid_remap_file=f'fregrid_remap_file_{NLON}_by_{NLAT}.nc'
+
+ nccmp_arg1 = REMAP_DIR + INPUT_GRID + '/' + INPUT_REALM + '/' + \
+ SOURCES_XY + '/' + INTERP_METHOD + '/' + fregrid_remap_file
+ nccmp_arg2 = REMAP_TEST_DIR + fregrid_remap_file
+ nccmp= [ 'nccmp', '-m', '--force', nccmp_arg1, nccmp_arg2 ]
+ print (' '.join(nccmp))
+ sp = subprocess.run( nccmp, check = True)
+ assert sp.returncode == 0
+ out, err = capfd.readouterr()
+
+
+def test_nccmp2_regrid_xy(capfd):
+ """
+ This test compares the regridded source file output(s), which are expected to be identical
+ """
+ nccmp_arg1 = TEST_OUT_DIR + f'{YYYYMMDD}.{SOURCE}.nc'
+ nccmp_arg2 = ALL_TEST_OUT_DIR + f'{YYYYMMDD}.{SOURCE}.nc'
+ nccmp= [ 'nccmp', '-m', '--force', nccmp_arg1, nccmp_arg2 ]
+ print (' '.join(nccmp))
+ sp = subprocess.run( nccmp, check = True)
+ assert sp.returncode == 0
+ out, err = capfd.readouterr()
+
+
+@pytest.mark.skip(reason='TODO')
+def test_regrid_one_for_two_comps(capfd):
+ """
+ this test will compare regridding settings for a single source file ref'd in two
+ diff components and regrid that source file twice if the settings are different,
+ and only once if the settings are the same.
+ """
+ assert False
+ out, err = capfd.readouterr()
diff --git a/lib/fre/catalog/__init__.py b/lib/fre/catalog/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/catalog/frecatalog.py b/lib/fre/catalog/frecatalog.py
new file mode 100644
index 00000000..58149e5e
--- /dev/null
+++ b/lib/fre/catalog/frecatalog.py
@@ -0,0 +1,62 @@
+'''
+entry point for fre catalog subcommands
+'''
+
+import click
+#import catalogbuilder
+from catalogbuilder.scripts import gen_intake_gfdl
+from catalogbuilder.scripts import test_catalog
+from catalogbuilder.scripts import combine_cats
+
+
+@click.group(help=click.style(" - access fre catalog subcommands", fg=(64,94,213)))
+def catalog_cli():
+ ''' entry point for click into fre catalog cli calls '''
+
+
+
+@catalog_cli.command()
+#TODO arguments dont have help message. So consider changing arguments to options?
+@click.argument('input_path', required = False, nargs = 1)
+#, help = 'The directory path with the datasets to be cataloged. E.g a GFDL PP path till /pp')
+@click.argument('output_path', required = False, nargs = 1)
+#, help = 'Specify output filename suffix only. e.g. catalog')
+@click.option('--config', required = False, type = click.Path(exists = True), nargs = 1,
+ help = 'Path to your yaml config, Use the config_template in intakebuilder repo')
+@click.option('--filter_realm', nargs = 1)
+@click.option('--filter_freq', nargs = 1)
+@click.option('--filter_chunk', nargs = 1)
+@click.option('--overwrite', is_flag = True, default = False)
+@click.option('--append', is_flag = True, default = False)
+@click.option('--slow', is_flag = True, default = False,
+ help = "Open NetCDF files to retrieve additional vocabulary (standard_name and intrafile static variables")
+@click.pass_context
+def builder(context, input_path = None, output_path = None, config = None, filter_realm = None,
+ filter_freq = None, filter_chunk = None, overwrite = False, append = False, slow = False):
+ # pylint: disable=unused-argument
+ """ - Generate .csv and .json files for catalog """
+ context.forward(gen_intake_gfdl.create_catalog_cli)
+
+@catalog_cli.command()
+@click.argument('json_path', nargs = 1 , required = True)
+@click.argument('json_template_path', nargs = 1 , required = False)
+@click.option('-tf', '--test-failure', is_flag=True, default = False,
+ help="Errors are only printed. Program will not exit.")
+@click.pass_context
+def validate(context, json_path, json_template_path, test_failure):
+ # pylint: disable=unused-argument
+ """ - Validate a catalog against catalog schema """
+ context.forward(test_catalog.main)
+
+@catalog_cli.command()
+@click.option('--input', required = True, multiple = True,
+ help = 'Catalog json files to be merged, space-separated')
+@click.option('--output', required = True, nargs = 1,
+ help = 'Merged catalog')
+@click.pass_context
+def merge(context, input, output):
+ """ - Merge two or more more catalogs into one """
+ context.invoke(combine_cats.combine_cats, inputfiles=input, output_path=output)
+
+if __name__ == "__main__":
+ catalog_cli()
diff --git a/lib/fre/catalog/tests/__init__.py b/lib/fre/catalog/tests/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/catalog/tests/test_fre_catalog.py b/lib/fre/catalog/tests/test_fre_catalog.py
new file mode 100644
index 00000000..7bc36637
--- /dev/null
+++ b/lib/fre/catalog/tests/test_fre_catalog.py
@@ -0,0 +1,5 @@
+#!/usr/bin/env python3
+
+def test_fre_catalog_import():
+ from fre import catalog
+ assert catalog is not None
diff --git a/lib/fre/check/__init__.py b/lib/fre/check/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/check/frecheck.py b/lib/fre/check/frecheck.py
new file mode 100644
index 00000000..9640d138
--- /dev/null
+++ b/lib/fre/check/frecheck.py
@@ -0,0 +1,20 @@
+''' fre check '''
+
+import click
+
+from .frecheckexample import check_test_function
+
+@click.group(help=click.style(" - access fre check subcommands", fg=(162,91,232)))
+def check_cli():
+ ''' entry point to fre check click commands '''
+
+@check_cli.command()
+@click.option('--uppercase', '-u', is_flag=True, help = 'Print statement in uppercase.')
+@click.pass_context
+def function(context, uppercase):
+ # pylint: disable=unused-argument
+ """ - Execute fre check test """
+ context.forward(check_test_function)
+
+if __name__ == "__main__":
+ check_cli()
diff --git a/lib/fre/check/frecheckexample.py b/lib/fre/check/frecheckexample.py
new file mode 100644
index 00000000..daac462a
--- /dev/null
+++ b/lib/fre/check/frecheckexample.py
@@ -0,0 +1,18 @@
+"""
+experimentation file for integrating one file's functions into main prototype fre file
+authored by Bennett.Chang@noaa.gov | bcc2761
+NOAA | GFDL
+"""
+
+import click
+
+@click.command()
+def check_test_function(uppercase=None):
+ """Execute fre list testfunction2."""
+ statement = "testingtestingtestingtesting"
+ if uppercase:
+ statement = statement.upper()
+ click.echo(statement)
+
+if __name__ == '__main__':
+ check_test_function()
diff --git a/lib/fre/cmor/README.md b/lib/fre/cmor/README.md
new file mode 100644
index 00000000..865c1ee7
--- /dev/null
+++ b/lib/fre/cmor/README.md
@@ -0,0 +1,102 @@
+UNDER CONSTRUCTION: old usage notes at the top of `cmor_mixer.py`, re-rigged for markdown and CMIP7.
+
+at PP/AN, module load the latest `fre-cli` that's been pushed to the main branch:
+```
+> module load fre/canopy
+> which fre
+ /home/fms/local/opt/fre-commands/canopy/bin/fre
+```
+
+alternatively, with access to conda:
+```
+> conda activate /nbhome/fms/conda/envs/fre-cli
+> which fre
+ /nbhome/fms/conda/envs/fre-cli/bin/fre
+```
+
+this subtool's help, and command-specific `run` help:
+```
+> fre cmor --help
+ Usage: fre cmor [OPTIONS] COMMAND [ARGS]...
+
+ - access fre cmor subcommands
+
+ Options:
+ --help Show this message and exit.
+
+ Commands:
+ run Rewrite climate model output
+
+
+# subtool command-specific help, e.g. for run
+> fre cmor run --help
+Usage: fre cmor run [OPTIONS]
+
+ Rewrite climate model output files with CMIP-compliant metadata for down-
+ stream publishing
+
+Options:
+ -d, --indir TEXT directory containing netCDF files. keys specified
+ in json_var_list are local variable names used for
+ targeting specific files in this directory
+ [required]
+ -l, --varlist TEXT path pointing to a json file containing directory
+ of key/value pairs. the keys are the 'local' names
+ used in the filename, and the values pointed to by
+ those keys are strings representing the name of the
+ variable contained in targeted files. the key and
+ value are often the same, but it is not required.
+ [required]
+ -r, --table_config TEXT json file containing CMIP-compliant per-
+ variable/metadata for specific MIP table. The MIP
+ table can generally be identified by the specific
+ filename (e.g. 'Omon') [required]
+ -p, --exp_config TEXT json file containing metadata dictionary for
+ CMORization. this metadata is effectively appended
+ to the final output file's header [required]
+ -o, --outdir TEXT directory root that will contain the full output
+ and output directory structure generated by the
+ cmor module upon request. [required]
+ -v, --opt_var_name TEXT optional, specify a variable name to specifically
+ process only filenames matching that variable name.
+ I.e., this string help target local_vars, not
+ target_vars.
+ --help Show this message and exit.
+```
+
+
+the tool requires configuration in the form of variable tables and conventions to work appropriately
+clone the following repository and list the following directory contents to get a sense of what
+the code needs from you to work. a few examples shown in the output below. the CV file is of particular interest/necessity
+```
+> git clone https://github.com/PCMDI/cmip6-cmor-tables.git fre/tests/test_files/cmip6-cmor-tables
+> ls fre/tests/test_files/cmip6-cmor-tables/Tables
+...
+ CMIP6_CV.json
+ CMIP6_formula_terms.json
+ CMIP6_grids.json
+ CMIP6_coordinate.json
+ CMIP6_input_example.json
+...
+ CMIP6_3hr.json
+...
+ CMIP6_Efx.json
+...
+ CMIP6_IyrGre.json
+...
+```
+
+
+Simple example call(s) using fre-cli in the root directory of this repository note the line-continuation character at the end for readability,
+you may wish to avoid it when copy/pasting.
+```
+> fre cmor run \
+ -d fre/tests/test_files/ocean_sos_var_file \
+ -l fre/tests/test_files/varlist \
+ -r fre/tests/test_files/cmip6-cmor-tables/Tables/CMIP6_Omon.json \
+ -p fre/tests/test_files/CMOR_input_example.json \
+ -o fre/tests/test_files/outdir
+```
+
+
+
diff --git a/lib/fre/cmor/__init__.py b/lib/fre/cmor/__init__.py
new file mode 100644
index 00000000..465ad02c
--- /dev/null
+++ b/lib/fre/cmor/__init__.py
@@ -0,0 +1,2 @@
+''' for fre.cmor imports '''
+from .cmor_mixer import cmor_run_subtool
diff --git a/lib/fre/cmor/cmor_mixer.py b/lib/fre/cmor/cmor_mixer.py
new file mode 100644
index 00000000..0f4ef243
--- /dev/null
+++ b/lib/fre/cmor/cmor_mixer.py
@@ -0,0 +1,652 @@
+'''
+python module housing the metadata processing routines utilizing the cmor module, in addition to
+click API entry points
+see README.md for additional information on `fre cmor run` (cmor_mixer.py) usage
+'''
+
+import os
+import glob
+import json
+import subprocess
+from pathlib import Path
+
+import netCDF4 as nc
+import click
+import cmor
+
+# ----- \start consts
+DEBUG_MODE_RUN_ONE = True
+
+# ----- \end consts
+
+### ------ helper functions ------ ###
+def copy_nc(in_nc, out_nc):
+ '''
+ copy target input netcdf file in_nc to target out_nc. I have to think this is not a trivial copy
+ operation, as if it were, using shutil's copy would be sufficient. accepts two arguments
+ in_nc: string, path to an input netcdf file we wish to copy
+ out_nc: string, an output path to copy the targeted input netcdf file to
+ '''
+ print(f'(copy_nc) in_nc: {in_nc}\n'
+ f' out_nc: {out_nc}')
+
+ # input file
+ dsin = nc.Dataset(in_nc)
+
+ # output file, same exact data_model as input file.
+ # note- totally infuriating...
+ # the correct value for the format arg is netCDF4.Dataset.data_model
+ # and NOT netCDF4.Dataset.disk_format
+ dsout = nc.Dataset(out_nc, "w",
+ format = dsin.data_model)
+
+ #Copy dimensions
+ for dname, the_dim in dsin.dimensions.items():
+ dsout.createDimension( dname,
+ len(the_dim) if not the_dim.isunlimited() else None )
+
+ # Copy variables and attributes
+ for v_name, varin in dsin.variables.items():
+ out_var = dsout.createVariable(v_name, varin.datatype, varin.dimensions)
+ out_var.setncatts({k: varin.getncattr(k) for k in varin.ncattrs()})
+ out_var[:] = varin[:]
+ dsout.setncatts({a:dsin.getncattr(a) for a in dsin.ncattrs()})
+
+ # close up
+ dsin.close()
+ dsout.close()
+
+
+def get_var_filenames(indir, var_filenames = None, local_var = None):
+ '''
+ appends files ending in .nc located within indir to list var_filenames accepts three arguments
+ indir: string, representing a path to a directory containing files ending in .nc extension
+ var_filenames: list of strings, empty or non-empty, to append discovered filenames to. the
+ object pointed to by the reference var_filenames is manipulated, and so need
+ not be returned.
+ local_var: string, optional, if not None, will be used for ruling out filename targets
+ '''
+ if var_filenames is None:
+ var_filenames = []
+ filename_pattern='.nc' if local_var is None else f'.{local_var}.nc'
+ print(f'(get_var_filenames) filename_pattern={filename_pattern}')
+ var_filenames_all=glob.glob(f'{indir}/*{filename_pattern}')
+ print(f'(get_var_filenames) var_filenames_all={var_filenames_all}')
+ for var_file in var_filenames_all:
+ var_filenames.append( Path(var_file).name )
+ print(f"(get_var_filenames) var_filenames = {var_filenames}")
+ if len(var_filenames) < 1:
+ raise ValueError(f'target directory had no files with .nc ending. indir =\n {indir}')
+ var_filenames.sort()
+
+
+def get_iso_datetimes(var_filenames, iso_datetime_arr = None):
+ '''
+ appends iso datetime strings found amongst filenames to iso_datetime_arr.
+ var_filenames: non-empty list of strings representing filenames. some of which presumably
+ contain datetime strings
+ iso_datetime_arr: list of strings, empty or non-empty, representing datetimes found in
+ var_filenames entries. the objet pointed to by the reference
+ iso_datetime_arr is manipulated, and so need-not be returned
+ '''
+ if iso_datetime_arr is None:
+ iso_datetime_arr = []
+ for filename in var_filenames:
+ iso_datetime = filename.split(".")[1]
+ if iso_datetime not in iso_datetime_arr:
+ iso_datetime_arr.append(
+ filename.split(".")[1] )
+ iso_datetime_arr.sort()
+ #print(f"(get_iso_datetimes) Available dates: {iso_datetime_arr}")
+ if len(iso_datetime_arr) < 1:
+ raise ValueError('(get_iso_datetimes) ERROR: iso_datetime_arr has length 0!')
+
+def check_dataset_for_ocean_grid(ds):
+ '''
+ checks netCDF4.Dataset ds for ocean grid origin, and throws an error if it finds one. accepts
+ one argument. this function has no return.
+ ds: netCDF4.Dataset object containing variables with associated dimensional information.
+ '''
+ if "xh" in list(ds.variables.keys()):
+ raise NotImplementedError(
+ "(check_dataset_for_ocean_grid) 'xh' found in var_list. ocean grid req'd but not yet unimplemented. stop.")
+
+
+def get_vertical_dimension(ds,target_var):
+ '''
+ determines the vertical dimensionality of target_var within netCDF4 Dataset ds. accepts two
+ arguments and returns an object represnting the vertical dimensions assoc with the target_var.
+ ds: netCDF4.Dataset object containing variables with associated dimensional information.
+ target_var: string, representating a variable contained within the netCDF4.Dataset ds
+
+ '''
+ vert_dim = 0
+ for name, variable in ds.variables.items():
+ # not the var we are looking for? move on.
+ if name != target_var:
+ continue
+ dims = variable.dimensions
+ for dim in dims:
+ # if it is not a vertical axis, move on.
+ print(f'(get_vertical_dimension) dim={dim}')
+ if dim == 'landuse':
+ continue
+ if not (ds[dim].axis and ds[dim].axis == "Z"):
+ continue
+ vert_dim = dim
+ return vert_dim
+
+def create_tmp_dir(outdir):
+ '''
+ creates a tmp_dir based on targeted output directory root. returns the name of the tmp dir.
+ accepts one argument:
+ outdir: string, representing the final output directory root for the cmor modules netcdf
+ file output. tmp_dir will be slightly different depending on the output directory
+ targeted
+ '''
+ print(f"(create_tmp_dir) outdir = {outdir}")
+ tmp_dir = None
+ if any( [ outdir == "/local2",
+ outdir.find("/work") != -1,
+ outdir.find("/net" ) != -1 ] ):
+ print(f'(create_tmp_dir) using /local /work /net ( tmp_dir = {outdir}/ )')
+ tmp_dir = str( Path("{outdir}/").resolve() )
+ else:
+ print(f'(create_tmp_dir) NOT using /local /work /net (tmp_dir = {outdir}/tmp/ )')
+ tmp_dir = str( Path(f"{outdir}/tmp/").resolve() )
+ try:
+ os.makedirs(tmp_dir, exist_ok=True)
+ except Exception as exc:
+ raise OSError('(create_tmp_dir) problem creating temp output directory. stop.') from exc
+ return tmp_dir
+
+
+
+### ------ BULK ROUTINES ------ ###
+def rewrite_netcdf_file_var ( proj_table_vars = None,
+ local_var = None,
+ netcdf_file = None,
+ target_var = None,
+ json_exp_config = None,
+ json_table_config = None):#, tmp_dir = None ):
+ '''
+ rewrite the input netcdf file nc_fl containing target_var in a CMIP-compliant manner.
+ accepts six arguments, all required:
+ proj_table_vars: json dictionary object, variable table read from json_table_config.
+ local_var: string, variable name used for finding files locally containing target_var,
+ this argument is often equal to target_var.
+ netcdf_file: string, representing path to intput netcdf file.
+ target_var: string, representing the variable name attached to the data object in the netcdf file.
+ json_exp_config: string, representing path to json configuration file holding metadata for appending to output
+ this argument is most used for making sure the right grid label is getting attached to the right output
+ json_table_config: string, representing path to json configuration file holding variable names for a given table.
+ proj_table_vars is read from this file, but both are passed anyways.
+ '''
+ print('\n\n-------------------------- START rewrite_netcdf_file_var call -----')
+ print( "(rewrite_netcdf_file_var) input data: " )
+ print(f" local_var = {local_var}" )
+ print(f" target_var = {target_var}")
+
+
+ # open the input file
+ print(f"(rewrite_netcdf_file_var) opening {netcdf_file}" )
+ ds = nc.Dataset(netcdf_file,'a')
+
+
+ # ocean grids are not implemented yet.
+ print( '(rewrite_netcdf_file_var) checking input netcdf file for oceangrid condition')
+ check_dataset_for_ocean_grid(ds)
+
+
+ # figure out the dimension names programmatically TODO
+ # Define lat and lon dimensions
+ # Assume input file is lat/lon grid
+ lat = ds["lat"][:]
+ lon = ds["lon"][:]
+ lat_bnds = ds["lat_bnds"][:]
+ lon_bnds = ds["lon_bnds"][:]
+
+ ## Define time
+ #time = ds["time"][:]
+
+ # read in time_coords + units
+ time_coords = ds["time"][:]
+ time_coord_units = ds["time"].units
+ print(f"(rewrite_netcdf_file_var) time_coord_units = {time_coord_units}")
+
+ # read in time_bnds , if present
+ time_bnds = []
+ try:
+ time_bnds = ds["time_bnds"][:]
+ #print(f"(rewrite_netcdf_file_var) time_bnds = {time_bnds}")
+ except ValueError:
+ print( "(rewrite_netcdf_file_var) WARNING grabbing time_bnds didnt work... moving on")
+
+
+ # read the input variable data, i believe
+ var = ds[target_var][:]
+
+ # determine the vertical dimension by looping over netcdf variables
+ vert_dim = get_vertical_dimension(ds, target_var)
+ print(f"(rewrite_netcdf_file_var) Vertical dimension of {target_var}: {vert_dim}")
+
+ # grab var_dim
+ var_dim = len(var.shape)
+ print(f"(rewrite_netcdf_file_var) var_dim = {var_dim}, local_var = {local_var}")
+
+ # Check var_dim
+ if var_dim not in [3, 4]:
+ raise ValueError(f"var_dim == {var_dim} != 3 nor 4. stop.")
+
+ # Check var_dim and vert_dim and assign lev if relevant.
+ # error if vert_dim wrong given var_dim
+ lev = None
+ if var_dim == 4:
+ if vert_dim not in [ "plev30", "plev19", "plev8",
+ "height2m", "level", "lev", "levhalf"] :
+ raise ValueError(f'var_dim={var_dim}, vert_dim = {vert_dim} is not supported')
+ lev = ds[vert_dim]
+
+
+ # now we set up the cmor module object
+ # initialize CMOR
+ cmor.setup(
+ netcdf_file_action = cmor.CMOR_PRESERVE,
+ set_verbosity = cmor.CMOR_QUIET, #default is CMOR_NORMAL
+ exit_control = cmor.CMOR_NORMAL,
+ logfile = None,
+ create_subdirectories = 1
+ )
+
+ # read experiment configuration file
+ print(f"(rewrite_netcdf_file_var) cmor is opening: json_exp_config = {json_exp_config}")
+ cmor.dataset_json(json_exp_config)
+
+ # load CMOR table
+ print(f"(rewrite_netcdf_file_var) cmor is opening json_table_config = {json_table_config}")
+ cmor.load_table(json_table_config)
+
+ units = proj_table_vars["variable_entry"] [target_var] ["units"]
+ print(f"(rewrite_netcdf_file_var) units={units}")
+
+ cmor_lat = cmor.axis("latitude", coord_vals = lat, cell_bounds = lat_bnds, units = "degrees_N")
+ cmor_lon = cmor.axis("longitude", coord_vals = lon, cell_bounds = lon_bnds, units = "degrees_E")
+ try:
+ print( f"(rewrite_netcdf_file_var) Executing cmor.axis('time', \n"
+ f" coord_vals = \n{time_coords}, \n"
+ f" cell_bounds = time_bnds, units = {time_coord_units}) ")
+ cmor_time = cmor.axis("time", coord_vals = time_coords,
+ cell_bounds = time_bnds, units = time_coord_units)
+ except ValueError as exc:
+ print(f"(rewrite_netcdf_file_var) WARNING exception raised... exc={exc}\n"
+ " cmor_time = cmor.axis('time', \n"
+ " coord_vals = time_coords, units = time_coord_units)")
+ cmor_time = cmor.axis("time", coord_vals = time_coords, units = time_coord_units)
+
+ # initializations
+ save_ps = False
+ ps = None
+ ierr_ap, ierr_b = None, None
+ ips = None
+
+ # set axes for 3-dim case
+ if var_dim == 3:
+ axes = [cmor_time, cmor_lat, cmor_lon]
+ print(f"(rewrite_netcdf_file_var) axes = {axes}")
+ # set axes for 4-dim case
+ elif var_dim == 4:
+
+ if vert_dim in ["plev30", "plev19", "plev8", "height2m"]:
+ cmor_lev = cmor.axis( vert_dim,
+ coord_vals = lev[:], units = lev.units )
+
+ elif vert_dim in ["level", "lev", "levhalf"]:
+ # find the ps file nearby
+ ps_file = netcdf_file.replace(f'.{target_var}.nc', '.ps.nc')
+ ds_ps = nc.Dataset(ps_file)
+ ps = ds_ps['ps'][:].copy()
+ ds_ps.close()
+
+ # assign lev_half specifics
+ if vert_dim == "levhalf":
+ cmor_lev = cmor.axis( "alternate_hybrid_sigma_half",
+ coord_vals = lev[:],
+ units = lev.units )
+ ierr_ap = cmor.zfactor( zaxis_id = cmor_lev,
+ zfactor_name = "ap_half",
+ axis_ids = [cmor_lev, ],
+ zfactor_values = ds["ap_bnds"][:],
+ units = ds["ap_bnds"].units )
+ ierr_b = cmor.zfactor( zaxis_id = cmor_lev,
+ zfactor_name = "b_half",
+ axis_ids = [cmor_lev, ],
+ zfactor_values = ds["b_bnds"][:],
+ units = ds["b_bnds"].units )
+ else:
+ cmor_lev = cmor.axis( "alternate_hybrid_sigma",
+ coord_vals = lev[:],
+ units = lev.units,
+ cell_bounds = ds[vert_dim+"_bnds"] )
+ ierr_ap = cmor.zfactor( zaxis_id = cmor_lev,
+ zfactor_name = "ap",
+ axis_ids = [cmor_lev, ],
+ zfactor_values = ds["ap"][:],
+ zfactor_bounds = ds["ap_bnds"][:],
+ units = ds["ap"].units )
+ ierr_b = cmor.zfactor( zaxis_id = cmor_lev,
+ zfactor_name = "b",
+ axis_ids = [cmor_lev, ],
+ zfactor_values = ds["b"][:],
+ zfactor_bounds = ds["b_bnds"][:],
+ units = ds["b"].units )
+
+ print(f'(rewrite_netcdf_file_var) ierr_ap after calling cmor_zfactor: {ierr_ap}\n'
+ f'(rewrite_netcdf_file_var) ierr_b after calling cmor_zfactor: {ierr_b}' )
+ ips = cmor.zfactor( zaxis_id = cmor_lev,
+ zfactor_name = "ps",
+ axis_ids = [cmor_time, cmor_lat, cmor_lon],
+ units = "Pa" )
+ save_ps = True
+ # assign axes at end of 4-dim case
+ axes = [cmor_time, cmor_lev, cmor_lat, cmor_lon]
+
+
+
+
+ # read positive attribute and create cmor_var? can this return none? TODO
+ positive = proj_table_vars["variable_entry"] [target_var] ["positive"]
+ print(f"(rewrite_netcdf_file_var) positive = {positive}")
+ cmor_var = cmor.variable(target_var, units, axes, positive = positive)
+
+ # Write the output to disk
+ #var = ds[target_var][:] #was this ever needed? why?
+ cmor.write(cmor_var, var)
+ if save_ps:
+ if any( [ ips is None, ps is None ] ):
+ print( '(rewrite_netcdf_file_var) WARNING: ps or ips is None!, but save_ps is True!\n'
+ f' ps = {ps}, ips = {ips}\n'
+ ' skipping ps writing!' )
+ else:
+ cmor.write(ips, ps, store_with = cmor_var)
+ cmor.close(ips, file_name = True, preserve = False)
+ filename = cmor.close(cmor_var, file_name = True, preserve = False)
+ print(f"(rewrite_netcdf_file_var) returned by cmor.close: filename = {filename}")
+ #cmor.close()
+ ds.close()
+
+ print('-------------------------- END rewrite_netcdf_file_var call -----\n\n')
+ return filename
+
+
+def cmorize_target_var_files( indir = None, target_var = None, local_var = None,
+ iso_datetime_arr = None, name_of_set = None,
+ json_exp_config = None, outdir = None,
+ proj_table_vars = None, json_table_config = None ):
+ ''' processes a target directory/file
+ this routine is almost entirely exposed data movement before/after calling
+ rewrite_netcdf_file_var it is also the most hopelessly opaque routine in this entire dang macro.
+ this badboy right here accepts... lord help us... !!!NINE!!! arguments, NINE.
+ indir: string, path to target directories containing netcdf files to cmorize
+ target_var: string, name of variable inside the netcdf file to cmorize
+ local_var: string, value of the variable name in the filename, right before the .nc
+ extension. often identical to target_var but not always.
+ iso_datetime_arr: list of strings, each one a unique ISO datetime string found in targeted
+ netcdf filenames
+ name_of_set: string, representing the post-processing component (GFDL convention) of the
+ targeted files.
+ json_exp_config: see cmor_run_subtool arg desc
+ outdir: string, path to output directory root to move the cmor module output to, including
+ the whole directory structure
+ proj_table_vars: an opened json file object, read from json_table_config
+ json_table_config: see cmor_run_subtool arg desc
+
+ '''
+ print('\n\n-------------------------- START cmorize_target_var_files call -----')
+ print(f"(cmorize_target_var_files) local_var = {local_var} to be used for file-targeting.\n"
+ f" target_var = {target_var} to be used for reading the data \n"
+ " from the file\n"
+ f" outdir = {outdir}")
+
+
+ #determine a tmp dir for working on files.
+ tmp_dir = create_tmp_dir( outdir ) + '/'
+ print(f'(cmorize_target_var_files) will use tmp_dir={tmp_dir}')
+
+
+ # loop over sets of dates, each one pointing to a file
+ nc_fls = {}
+ for i, iso_datetime in enumerate(iso_datetime_arr):
+
+ # why is nc_fls a filled list/array/object thingy here? see above line
+ nc_fls[i] = f"{indir}/{name_of_set}.{iso_datetime}.{local_var}.nc"
+ print(f"(cmorize_target_var_files) input file = {nc_fls[i]}")
+ if not Path(nc_fls[i]).exists():
+ print ("(cmorize_target_var_files) input file(s) not found. Moving on.")
+ continue
+
+
+ # create a copy of the input file with local var name into the work directory
+ nc_file_work = f"{tmp_dir}{name_of_set}.{iso_datetime}.{local_var}.nc"
+
+ print(f"(cmorize_target_var_files) nc_file_work = {nc_file_work}")
+ copy_nc( nc_fls[i], nc_file_work)
+
+ # if the ps file exists, we'll copy it to the work directory too
+ nc_ps_file = nc_fls[i].replace(f'.{local_var}.nc', '.ps.nc')
+ nc_ps_file_work = nc_file_work.replace(f'.{local_var}.nc', '.ps.nc')
+ if Path(nc_ps_file).exists():
+ print(f"(cmorize_target_var_files) nc_ps_file_work = {nc_ps_file_work}")
+ copy_nc(nc_ps_file, nc_ps_file_work)
+
+
+ # TODO think of better way to write this kind of conditional data movement...
+ # now we have a file in our targets, point CMOR to the configs and the input file(s)
+ make_cmor_write_here = None
+ print( Path( tmp_dir ) )
+ print( Path( os.getcwd() ) )
+ if Path( tmp_dir ).is_absolute():
+ print(f'tmp_dir is absolute')
+ make_cmor_write_here = tmp_dir
+ elif Path( tmp_dir ).exists(): # relative to where we are
+ print(f'tmp_dir is relative to CWD!')
+ make_cmor_write_here = os.getcwd() + '/'+tmp_dir # unavoidable, cmor module FORCES write to CWD
+ assert make_cmor_write_here is not None
+
+ gotta_go_back_here=os.getcwd()
+ try:
+ print(f"cd'ing to \n {make_cmor_write_here}" )
+ os.chdir( make_cmor_write_here )
+ except:
+ raise OSError(f'could not chdir to {make_cmor_write_here}')
+
+ print ("(cmorize_target_var_files) calling rewrite_netcdf_file_var")
+ local_file_name = rewrite_netcdf_file_var( proj_table_vars ,
+ local_var ,
+ nc_file_work ,
+ target_var ,
+ json_exp_config ,
+ json_table_config )
+ os.chdir( gotta_go_back_here )
+
+
+ # now that CMOR has rewritten things... we can take our post-rewriting actions
+ # the final output filename will be...
+ print(f'(cmorize_target_var_files) local_file_name={local_file_name}')
+ filename =f"{outdir}/{local_file_name}"
+ print(f"(cmorize_target_var_files) filename = {filename}")
+
+ # the final output file directory will be...
+ filedir = Path(filename).parent
+ print(f"(cmorize_target_var_files) filedir = {filedir}")
+ try:
+ print(f'(cmorize_target_var_files) attempting to create filedir={filedir}')
+ os.makedirs(filedir)
+ except FileExistsError:
+ print(f'(cmorize_target_var_files) WARNING: directory {filedir} already exists!')
+
+ # hmm.... this is making issues for pytest
+ mv_cmd = f"mv {tmp_dir}/{local_file_name} {filedir}"
+ print(f"(cmorize_target_var_files) moving files...\n {mv_cmd}")
+ subprocess.run(mv_cmd, shell = True, check = True)
+
+ # ------ refactor this into function? #TODO
+ # ------ what is the use case for this logic really??
+ filename_no_nc = filename[:filename.rfind(".nc")]
+ chunk_str = filename_no_nc[-6:]
+ if not chunk_str.isdigit():
+ print(f'(cmorize_target_var_files) WARNING: chunk_str is not a digit: '
+ f'chunk_str = {chunk_str}')
+ filename_corr = "{filename[:filename.rfind('.nc')]}_{iso_datetime}.nc"
+ mv_cmd = f"mv {filename} {filename_corr}"
+ print(f"(cmorize_target_var_files) moving files, strange chunkstr logic...\n {mv_cmd}")
+ subprocess.run(mv_cmd, shell = True, check = True)
+ # ------ end refactor this into function?
+
+ # delete files in work dirs
+ if Path(nc_file_work).exists():
+ Path(nc_file_work).unlink()
+
+ if Path(nc_ps_file_work).exists():
+ Path(nc_ps_file_work).unlink()
+
+ if DEBUG_MODE_RUN_ONE:
+ print(f'WARNING: DEBUG_MODE_RUN_ONE is True!!!!')
+ print(f'WARNING: done processing one file!!!')
+ break
+
+
+
+
+def cmor_run_subtool( indir = None,
+ json_var_list = None,
+ json_table_config = None,
+ json_exp_config = None ,
+ outdir = None, opt_var_name = None
+ ):
+ '''
+ primary steering function for the cmor_mixer tool, i.e essentially main. Accepts six args:
+ indir: string, directory containing netCDF files. keys specified in json_var_list are local
+ variable names used for targeting specific files
+ json_var_list: string, path pointing to a json file containing directory of key/value
+ pairs. the keys are the "local" names used in the filename, and the
+ values pointed to by those keys are strings representing the name of the
+ variable contained in targeted files. the key and value are often the same,
+ but it is not required.
+ json_table_config: json file containing CMIP-compliant per-variable/metadata for specific
+ MIP table. The MIP table can generally be identified by the specific
+ filename (e.g. "Omon")
+ json_exp_config: json file containing metadata dictionary for CMORization. this metadata is effectively
+ appended to the final output file's header
+ outdir: string, directory root that will contain the full output and output directory
+ structure generated by the cmor module upon request.
+ opt_var_name: string, optional, specify a variable name to specifically process only filenames matching
+ that variable name. I.e., this string help target local_vars, not target_vars.
+ '''
+ # check req'd inputs
+ if None in [indir, json_var_list, json_table_config, json_exp_config, outdir]:
+ raise ValueError(f'(cmor_run_subtool) all input arguments except opt_var_name are required!\n'
+ ' [indir, json_var_list, json_table_config, json_exp_config, outdir] = \n'
+ f' [{indir}, {json_var_list}, {json_table_config}, '
+ ' {json_exp_config}, {outdir}]' )
+
+ # open CMOR table config file
+ print( '(cmor_run_subtool) getting table variables from json_table_config = \n'
+ f' {json_table_config}' )
+ try:
+ with open( json_table_config, "r", encoding = "utf-8") as table_config_file:
+ proj_table_vars=json.load(table_config_file)
+
+ except Exception as exc:
+ raise FileNotFoundError(
+ f'ERROR: json_table_config file cannot be opened.\n'
+ f' json_table_config = {json_table_config}' ) from exc
+
+ # now resolve the json_table_config path after confirming it can be open
+ json_table_config= str( Path(json_table_config).resolve() )
+
+ # open input variable list
+ print('(cmor_run_subtool) opening variable list json_var_list')
+ try:
+ with open( json_var_list, "r", encoding = "utf-8" ) as var_list_file:
+ var_list = json.load( var_list_file )
+
+ except Exception as exc:
+ raise FileNotFoundError(
+ f'ERROR: json_var_list file cannot be opened.\n'
+ f' json_var_list = {json_var_list}' ) from exc
+
+ # make sure the exp config exists too while we're at it...
+ if Path(json_exp_config).exists(): # if so, resolve to absolute path
+ json_exp_config = str( Path( json_exp_config).resolve() )
+ else:
+ raise FileNotFoundError(
+ f'ERROR: json_exp_config file cannot be opened.\n'
+ f' json_exp_config = {json_exp_config}' )
+
+ # loop over entries in the json_var_list, read into var_list
+ for local_var in var_list:
+
+ # if its not in the table configurations variable_entry list, skip
+ if var_list[local_var] not in proj_table_vars["variable_entry"]:
+ print(f"(cmor_run_subtool) WARNING: skipping local_var = {local_var} /\n"
+ f" target_var = {var_list[local_var]}\n"
+ " ... target_var not found in CMOR variable group")
+ continue
+
+ if all( [ opt_var_name is not None,
+ local_var != opt_var_name ] ):
+ print(f'(cmor_run_subtool) WARNING: skipping local_var={local_var} as it is not equal\n'
+ ' to the opt_var_name argument.')
+ continue
+
+ # it is in there, get the name of the data inside the netcdf file.
+ target_var=var_list[local_var] # often equiv to local_var but not necessarily.
+ if local_var != target_var:
+ print(f'(cmor_run_subtool) WARNING: local_var == {local_var} \n'
+ f' != {target_var} == target_var\n'
+ f' i am expecting {local_var} to be in the filename, and i expect the variable\n'
+ f' in that file to be {target_var}')
+
+
+ # examine input directory to obtain a list of input file targets
+ var_filenames = []
+ get_var_filenames(indir, var_filenames, local_var)
+ print(f"(cmor_run_subtool) found filenames = \n {var_filenames}")
+
+ # examine input files to obtain target date ranges
+ iso_datetime_arr = []
+ get_iso_datetimes(var_filenames, iso_datetime_arr)
+ print(f"(cmor_run_subtool) found iso datetimes = \n {iso_datetime_arr}")
+
+ # name_of_set == component label...
+ # which is not relevant for CMOR/CMIP... or is it?
+ name_of_set = var_filenames[0].split(".")[0]
+ print(f"(cmor_run_subtool) setting name_of_set = {name_of_set}")
+
+
+
+ print(f'(cmor_run_subtool) ..............beginning CMORization for {local_var}/\n'
+ f' {target_var}..........')
+ cmorize_target_var_files(
+ indir, target_var, local_var, iso_datetime_arr, # OK
+ name_of_set, json_exp_config,
+ outdir,
+ proj_table_vars, json_table_config # a little redundant
+ )
+
+ if DEBUG_MODE_RUN_ONE:
+ print(f'WARNING: DEBUG_MODE_RUN_ONE is True. breaking var_list loop')
+ break
+ return 0
+
+
+@click.command()
+def _cmor_run_subtool(indir = None,
+ json_var_list = None, json_table_config = None, json_exp_config = None,
+ outdir = None, opt_var_name = None):
+ ''' entry point to fre cmor run for click. see cmor_run_subtool for argument descriptions.'''
+ return cmor_run_subtool(indir, json_var_list, json_table_config, json_exp_config, outdir, opt_var_name)
+
+
+if __name__ == '__main__':
+ cmor_run_subtool()
diff --git a/lib/fre/cmor/frecmor.py b/lib/fre/cmor/frecmor.py
new file mode 100644
index 00000000..e882186a
--- /dev/null
+++ b/lib/fre/cmor/frecmor.py
@@ -0,0 +1,66 @@
+''' fre cmor '''
+
+import click
+
+from .cmor_mixer import _cmor_run_subtool
+
+@click.group(help=click.style(" - access fre cmor subcommands", fg=(232,91,204)))
+def cmor_cli():
+ ''' entry point to fre cmor click commands '''
+
+@cmor_cli.command()
+@click.option("-d", "--indir",
+ type=str,
+ help="directory containing netCDF files. keys specified in json_var_list are local " + \
+ "variable names used for targeting specific files in this directory",
+ required=True)
+@click.option("-l", "--varlist",
+ type=str,
+ help="path pointing to a json file containing directory of key/value pairs. " + \
+ "the keys are the \'local\' names used in the filename, and the values " + \
+ "pointed to by those keys are strings representing the name of the variable " + \
+ "contained in targeted files. the key and value are often the same, " + \
+ "but it is not required.",
+ required=True)
+@click.option("-r", "--table_config",
+ type=str,
+ help="json file containing CMIP-compliant per-variable/metadata for specific " + \
+ "MIP table. The MIP table can generally be identified by the specific " + \
+ "filename (e.g. \'Omon\')",
+ required=True)
+@click.option("-p", "--exp_config",
+ type=str,
+ help="json file containing metadata dictionary for CMORization. this metadata is " + \
+ "effectively appended to the final output file's header",
+ required=True)
+@click.option("-o", "--outdir",
+ type=str,
+ help="directory root that will contain the full output and output directory " + \
+ "structure generated by the cmor module upon request.",
+ required=True)
+@click.option('-v', "--opt_var_name",
+ type = str,
+ help="optional, specify a variable name to specifically process only filenames " + \
+ "matching that variable name. I.e., this string help target local_vars, not " + \
+ "target_vars.",
+ required=False)
+@click.pass_context
+def run(context, indir, varlist, table_config, exp_config, outdir, opt_var_name):
+ # pylint: disable=unused-argument
+ """
+ Rewrite climate model output files with CMIP-compliant metadata for down-stream publishing
+ """
+ context.invoke(
+ _cmor_run_subtool,
+ indir = indir,
+ json_var_list = varlist,
+ json_table_config = table_config,
+ json_exp_config = exp_config,
+ outdir = outdir,
+ opt_var_name = opt_var_name
+ )
+ # context.forward(
+ # _cmor_run_subtool() )
+
+if __name__ == "__main__":
+ cmor_cli()
diff --git a/lib/fre/cmor/tests/__init__.py b/lib/fre/cmor/tests/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/cmor/tests/test_cmor_run_subtool.py b/lib/fre/cmor/tests/test_cmor_run_subtool.py
new file mode 100644
index 00000000..043b5705
--- /dev/null
+++ b/lib/fre/cmor/tests/test_cmor_run_subtool.py
@@ -0,0 +1,249 @@
+''' tests for fre.cmor.cmor_run_subtool '''
+import subprocess
+import shutil
+from pathlib import Path
+from datetime import date
+
+import git
+
+import fre
+
+# where are we? we're running pytest from the base directory of this repo
+ROOTDIR = 'fre/tests/test_files'
+
+# setup- cmip/cmor variable table(s)
+CLONE_CMIP_TABLE_URL = \
+ 'https://github.com/PCMDI/cmip6-cmor-tables.git'
+CLONE_REPO_PATH = \
+ f'{ROOTDIR}/cmip6-cmor-tables'
+TABLE_CONFIG = \
+ f'{CLONE_REPO_PATH}/Tables/CMIP6_Omon.json'
+
+def test_setup_cmor_cmip_table_repo():
+ ''' setup routine, if it doesnt exist, clone the repo holding CMOR/CMIP6 tables '''
+ if Path(TABLE_CONFIG).exists():
+ pass
+ else:
+ git.Repo.clone_from(
+ CLONE_CMIP_TABLE_URL,
+ CLONE_REPO_PATH )
+ assert Path(TABLE_CONFIG).exists()
+
+# explicit inputs to tool
+INDIR = f'{ROOTDIR}/ocean_sos_var_file'
+VARLIST = f'{ROOTDIR}/varlist'
+EXP_CONFIG = f'{ROOTDIR}/CMOR_input_example.json'
+OUTDIR = f'{ROOTDIR}/outdir'
+TMPDIR = f'{OUTDIR}/tmp'
+
+# determined by cmor_run_subtool
+YYYYMMDD = date.today().strftime('%Y%m%d')
+CMOR_CREATES_DIR = \
+ 'CMIP6/CMIP6/ISMIP6/PCMDI/PCMDI-test-1-0/piControl-withism/r3i1p1f1/Omon/sos/gn'
+FULL_OUTPUTDIR = \
+ f"{OUTDIR}/{CMOR_CREATES_DIR}/v{YYYYMMDD}"
+FULL_OUTPUTFILE = \
+f"{FULL_OUTPUTDIR}/sos_Omon_PCMDI-test-1-0_piControl-withism_r3i1p1f1_gn_199307-199807.nc"
+
+# FYI but helpful for tests
+FILENAME = 'ocean_monthly_1x1deg.199301-199712.sos.nc' # unneeded, this is mostly for reference
+FULL_INPUTFILE=f"{INDIR}/{FILENAME}"
+
+def test_setup_fre_cmor_run_subtool(capfd):
+ ''' checks for outputfile from prev pytest runs, removes it if it's present.
+ this routine also checks to make sure the desired input file is present'''
+ if Path(FULL_OUTPUTFILE).exists():
+ Path(FULL_OUTPUTFILE).unlink()
+ if Path(OUTDIR).exists():
+ shutil.rmtree(OUTDIR)
+ assert not any ( [ Path(FULL_OUTPUTFILE).exists(),
+ Path(OUTDIR).exists() ] )
+ assert Path(FULL_INPUTFILE).exists()
+ _out, _err = capfd.readouterr()
+
+def test_fre_cmor_run_subtool_case1(capfd):
+ ''' fre cmor run, test-use case '''
+
+ #debug
+ #print(
+ # f"fre.cmor.cmor_run_subtool("
+ # f"\'{INDIR}\',"
+ # f"\'{VARLIST}\',"
+ # f"\'{TABLE_CONFIG}\',"
+ # f"\'{EXP_CONFIG}\',"
+ # f"\'{OUTDIR}\'"
+ # ")"
+ #)
+
+ # test call, where meat of the workload gets done
+ fre.cmor.cmor_run_subtool(
+ indir = INDIR,
+ json_var_list = VARLIST,
+ json_table_config = TABLE_CONFIG,
+ json_exp_config = EXP_CONFIG,
+ outdir = OUTDIR
+ )
+
+ assert all( [ Path(FULL_OUTPUTFILE).exists(),
+ Path(FULL_INPUTFILE).exists() ] )
+ _out, _err = capfd.readouterr()
+
+def test_fre_cmor_run_subtool_case1_output_compare_data(capfd):
+ ''' I/O data-only comparison of test case1 '''
+ print(f'FULL_OUTPUTFILE={FULL_OUTPUTFILE}')
+ print(f'FULL_INPUTFILE={FULL_INPUTFILE}')
+
+ nccmp_cmd= [ "nccmp", "-f", "-d",
+ f"{FULL_INPUTFILE}",
+ f"{FULL_OUTPUTFILE}" ]
+ print(f"via subprocess, running {' '.join(nccmp_cmd)}")
+ result = subprocess.run( ' '.join(nccmp_cmd),
+ shell=True,
+ check=False,
+ capture_output=True
+ )
+ # err_list has length two if end in newline
+ err_list = result.stderr.decode().split('\n')
+ expected_err = \
+ "DIFFER : FILE FORMATS : NC_FORMAT_64BIT <> NC_FORMAT_NETCDF4_CLASSIC"
+ assert all( [result.returncode == 1,
+ len(err_list)==2,
+ '' in err_list,
+ expected_err in err_list ] )
+ _out, _err = capfd.readouterr()
+
+def test_fre_cmor_run_subtool_case1_output_compare_metadata(capfd):
+ ''' I/O metadata-only comparison of test case1 '''
+ print(f'FULL_OUTPUTFILE={FULL_OUTPUTFILE}')
+ print(f'FULL_INPUTFILE={FULL_INPUTFILE}')
+
+ nccmp_cmd= [ "nccmp", "-f", "-m", "-g",
+ f"{FULL_INPUTFILE}",
+ f"{FULL_OUTPUTFILE}" ]
+ print(f"via subprocess, running {' '.join(nccmp_cmd)}")
+ result = subprocess.run( ' '.join(nccmp_cmd),
+ shell=True,
+ check=False
+ )
+
+ assert result.returncode == 1
+ _out, _err = capfd.readouterr()
+
+
+# FYI, but again, helpful for tests
+FILENAME_DIFF = \
+ 'ocean_monthly_1x1deg.199301-199712.sosV2.nc'
+FULL_INPUTFILE_DIFF = \
+ f"{INDIR}/{FILENAME_DIFF}"
+VARLIST_DIFF = \
+ f'{ROOTDIR}/varlist_local_target_vars_differ'
+def test_setup_fre_cmor_run_subtool_case2(capfd):
+ ''' make a copy of the input file to the slightly different name.
+ checks for outputfile from prev pytest runs, removes it if it's present.
+ this routine also checks to make sure the desired input file is present'''
+ if Path(FULL_OUTPUTFILE).exists():
+ Path(FULL_OUTPUTFILE).unlink()
+ assert not Path(FULL_OUTPUTFILE).exists()
+
+ if Path(OUTDIR+'/CMIP6').exists():
+ shutil.rmtree(OUTDIR+'/CMIP6')
+ assert not Path(OUTDIR+'/CMIP6').exists()
+
+
+ # VERY ANNOYING !!! FYI WARNING TODO
+ if Path(TMPDIR).exists():
+ try:
+ shutil.rmtree(TMPDIR)
+ except OSError as exc:
+ print(f'WARNING: TMPDIR={TMPDIR} could not be removed.')
+ print( ' this does not matter that much, but is unfortunate.')
+ print( ' supicion: something the cmor module is using is not being closed')
+
+ #assert not Path(TMPDIR).exists() # VERY ANNOYING !!! FYI WARNING TODO
+
+ # VERY ANNOYING !!! FYI WARNING TODO
+ if Path(OUTDIR).exists():
+ try:
+ shutil.rmtree(OUTDIR)
+ except OSError as exc:
+ print(f'WARNING: OUTDIR={OUTDIR} could not be removed.')
+ print( ' this does not matter that much, but is unfortunate.')
+ print( ' supicion: something the cmor module is using is not being closed')
+
+ #assert not Path(OUTDIR).exists() # VERY ANNOYING !!! FYI WARNING TODO
+
+ # make a copy of the usual test file.
+ if not Path(FULL_INPUTFILE_DIFF).exists():
+ shutil.copy(
+ Path(FULL_INPUTFILE),
+ Path(FULL_INPUTFILE_DIFF) )
+ assert Path(FULL_INPUTFILE_DIFF).exists()
+ _out, _err = capfd.readouterr()
+
+def test_fre_cmor_run_subtool_case2(capfd):
+ ''' fre cmor run, test-use case2 '''
+
+ #debug
+ #print(
+ # f"fre.cmor.cmor_run_subtool("
+ # f"\'{INDIR}\',"
+ # f"\'{VARLIST_DIFF}\',"
+ # f"\'{TABLE_CONFIG}\',"
+ # f"\'{EXP_CONFIG}\',"
+ # f"\'{OUTDIR}\'"
+ # ")"
+ #)
+
+ # test call, where meat of the workload gets done
+ fre.cmor.cmor_run_subtool(
+ indir = INDIR,
+ json_var_list = VARLIST_DIFF,
+ json_table_config = TABLE_CONFIG,
+ json_exp_config = EXP_CONFIG,
+ outdir = OUTDIR
+ )
+
+ # check we ran on the right input file.
+ assert all( [ Path(FULL_OUTPUTFILE).exists(),
+ Path(FULL_INPUTFILE_DIFF).exists() ] )
+ _out, _err = capfd.readouterr()
+
+def test_fre_cmor_run_subtool_case2_output_compare_data(capfd):
+ ''' I/O data-only comparison of test case2 '''
+ print(f'FULL_OUTPUTFILE={FULL_OUTPUTFILE}')
+ print(f'FULL_INPUTFILE_DIFF={FULL_INPUTFILE_DIFF}')
+
+ nccmp_cmd= [ "nccmp", "-f", "-d",
+ f"{FULL_INPUTFILE_DIFF}",
+ f"{FULL_OUTPUTFILE}" ]
+ print(f"via subprocess, running {' '.join(nccmp_cmd)}")
+ result = subprocess.run( ' '.join(nccmp_cmd),
+ shell=True,
+ check=False,
+ capture_output=True
+ )
+
+ err_list = result.stderr.decode().split('\n')#length two if end in newline
+ expected_err="DIFFER : FILE FORMATS : NC_FORMAT_64BIT <> NC_FORMAT_NETCDF4_CLASSIC"
+ assert all( [result.returncode == 1,
+ len(err_list)==2,
+ '' in err_list,
+ expected_err in err_list ] )
+ _out, _err = capfd.readouterr()
+
+def test_fre_cmor_run_subtool_case2_output_compare_metadata(capfd):
+ ''' I/O metadata-only comparison of test case2 '''
+ print(f'FULL_OUTPUTFILE={FULL_OUTPUTFILE}')
+ print(f'FULL_INPUTFILE_DIFF={FULL_INPUTFILE_DIFF}')
+
+ nccmp_cmd= [ "nccmp", "-f", "-m", "-g",
+ f"{FULL_INPUTFILE_DIFF}",
+ f"{FULL_OUTPUTFILE}" ]
+ print(f"via subprocess, running {' '.join(nccmp_cmd)}")
+ result = subprocess.run( ' '.join(nccmp_cmd),
+ shell=True,
+ check=False
+ )
+
+ assert result.returncode == 1
+ _out, _err = capfd.readouterr()
diff --git a/lib/fre/coveragerc b/lib/fre/coveragerc
new file mode 100644
index 00000000..0e936b95
--- /dev/null
+++ b/lib/fre/coveragerc
@@ -0,0 +1,15 @@
+# https://pytest-cov.readthedocs.io/en/latest/config.html
+[run]
+omit =
+ fre/tests/*
+ fre/app/generate_time_averages/tests/*
+ fre/app/regrid_xy/tests/*
+ fre/catalog/tests/*
+ fre/check
+ fre/cmor/tests/*
+ fre/list
+ fre/make/tests/*
+ fre/pp/tests/*
+ fre/run
+ fre/test
+ fre/yamltools/tests/*
diff --git a/lib/fre/fre.py b/lib/fre/fre.py
new file mode 100644
index 00000000..388a4588
--- /dev/null
+++ b/lib/fre/fre.py
@@ -0,0 +1,40 @@
+"""
+Main host file for FRE-CLI program scripts
+authored by Bennett.Chang@noaa.gov | bcc2761
+NOAA | GFDL
+2023-2024
+principal click group for main/fre allows for subgroup functions to
+be called via this script. I.e. 'fre' is the entry point
+"""
+
+import click
+from .lazy_group import LazyGroup
+
+@click.group(
+ cls = LazyGroup,
+ lazy_subcommands = {"pp": ".pp.frepp.pp_cli",
+ "catalog": ".catalog.frecatalog.catalog_cli",
+ "list": ".list.frelist.list_cli",
+ "check": ".check.frecheck.check_cli",
+ "run": ".run.frerun.run_cli",
+ "test": ".test.fretest.test_cli",
+ "yamltools": ".yamltools.freyamltools.yamltools_cli",
+ "make": ".make.fremake.make_cli",
+ "app": ".app.freapp.app_cli",
+ "cmor": ".cmor.frecmor.cmor_cli" },
+ help = click.style(
+ "'fre' is the main CLI click group that houses the other tool groups as lazy subcommands.",
+ fg='cyan')
+)
+
+
+@click.version_option(
+ package_name = "fre-cli"
+)
+
+def fre():
+ ''' entry point function to subgroup functions '''
+
+
+if __name__ == '__main__':
+ fre()
diff --git a/lib/fre/gfdl_msd_schemas/.github/CODEOWNERS b/lib/fre/gfdl_msd_schemas/.github/CODEOWNERS
new file mode 100644
index 00000000..22c95973
--- /dev/null
+++ b/lib/fre/gfdl_msd_schemas/.github/CODEOWNERS
@@ -0,0 +1,4 @@
+* @ceblanton @J-Lentz
+FMS/diag_table.yaml @uramirez8707
+FMS/data_table.yaml @J-Lentz
+FMS/field_table.yaml @J-Lentz
diff --git a/lib/fre/gfdl_msd_schemas/FMS/data_table.json b/lib/fre/gfdl_msd_schemas/FMS/data_table.json
new file mode 100644
index 00000000..f0683374
--- /dev/null
+++ b/lib/fre/gfdl_msd_schemas/FMS/data_table.json
@@ -0,0 +1,114 @@
+{
+ "title": "Data table schema for data_override",
+ "type": "object",
+ "properties": {
+ "data_table": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "properties": {
+ "grid_name": {
+ "type": "string",
+ "enum": ["OCN", "LND", "ATM", "ICE"]
+ },
+ "fieldname_in_model": {
+ "type": "string",
+ "minLength": 1
+ },
+ "override_file": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "properties": {
+ "file_name": {
+ "type": "string",
+ "minLength": 1
+ },
+ "fieldname_in_file": {
+ "type": "string",
+ "minLength": 1
+ },
+ "interp_method": {
+ "type": "string",
+ "enum": ["bilinear", "bicubic", "none"]
+ },
+ "multi_file": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "properties": {
+ "next_file_name": {
+ "type": "string",
+ "minLength": 1
+ },
+ "prev_file_name": {
+ "type": "string",
+ "minLength": 1
+ }
+ },
+ "required": ["next_file_name", "prev_file_name"],
+ "additionalProperties": false
+ }
+ },
+ "external_weights": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "properties": {
+ "file_name": {
+ "type": "string",
+ "minLength": 1
+ },
+ "source": {
+ "type": "string",
+ "minLength": 1,
+ "enum": ["fregrid"]
+ }
+ },
+ "required": ["file_name", "source"],
+ "additionalProperties": false
+ }
+ }
+ },
+ "required": ["file_name", "fieldname_in_file", "interp_method"],
+ "additionalProperties": false
+ }
+ },
+ "factor": {
+ "type": "number"
+ },
+ "subregion": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "properties": {
+ "type": {
+ "type": "string",
+ "enum": ["inside_region", "outside_region"]
+ },
+ "lon_start": {
+ "type": "number"
+ },
+ "lon_end": {
+ "type": "number"
+ },
+ "lat_start": {
+ "type": "number"
+ },
+ "lat_end": {
+ "type": "number"
+ }
+ },
+ "required": ["type", "lon_start", "lon_end", "lat_start", "lat_end"],
+ "additionalProperties": false
+ }
+ }
+ },
+ "required": ["grid_name", "fieldname_in_model", "factor"],
+ "additionalProperties": false
+ }
+ }
+ },
+ "required": ["data_table"],
+ "additionalProperties": false
+}
diff --git a/lib/fre/gfdl_msd_schemas/FMS/diag_table.json b/lib/fre/gfdl_msd_schemas/FMS/diag_table.json
new file mode 100644
index 00000000..a968ca6b
--- /dev/null
+++ b/lib/fre/gfdl_msd_schemas/FMS/diag_table.json
@@ -0,0 +1,250 @@
+{
+ "$schema": "http://json-schema.org/draft-07/schema#",
+ "type": "object",
+ "required": ["title", "base_date"],
+ "additionalProperties": false,
+ "properties": {
+ "title": {
+ "type": "string",
+ "minLength": 1
+ },
+ "base_date": {
+ "type": "string",
+ "anyOf": [
+ {"enum": ["$baseDate"]},
+ {"pattern": "^([0-9]*[1-9][0-9]* ){3}([0-9]+ ){2}[0-9]+$"}
+ ]
+ },
+ "diag_files": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "required": ["file_name", "freq", "time_units", "unlimdim"],
+ "additionalProperties": false,
+ "properties": {
+ "file_name": {
+ "type": "string",
+ "minLength": 1
+ },
+ "freq": {
+ "anyOf": [
+ {"type": "string"},
+ {"type": "number"}
+ ],
+ "pattern": "^(-1|0|((-1|[0-9]+) +(seconds|minutes|hours|days|months|years)( *, *)?)+)$"
+ },
+ "time_units": {
+ "type": "string",
+ "enum": ["seconds", "minutes", "hours", "days", "months", "years"]
+ },
+ "unlimdim": {
+ "type": "string",
+ "minLength": 1
+ },
+ "write_file": {
+ "type": "boolean"
+ },
+ "global_meta": {
+ "type": "array",
+ "minItems": 1,
+ "maxItems": 1,
+ "items": {
+ "type": "object",
+ "additionalProperties": {
+ "oneOf": [
+ {"type": "string"},
+ {"type": "number"},
+ {"type": "boolean"}
+ ]
+ }
+ }
+ },
+ "sub_region": {
+ "type": "array",
+ "minItems": 1,
+ "maxItems": 1,
+ "required": ["grid_type", "corner1", "corner2", "corner3", "corner4"],
+ "properties": {
+ "grid_type": {
+ "type": "string",
+ "enum": ["indices", "latlon"]
+ },
+ "corner1": {
+ "type": "array",
+ "minItems": 2,
+ "maxItems": 2,
+ "items": {
+ "type": "number"
+ }
+ },
+ "corner2": {
+ "type": "array",
+ "minItems": 2,
+ "maxItems": 2,
+ "items": {
+ "type": "number"
+ }
+ },
+ "corner3": {
+ "type": "array",
+ "minItems": 2,
+ "maxItems": 2,
+ "items": {
+ "type": "number"
+ }
+ },
+ "corner4": {
+ "type": "array",
+ "minItems": 2,
+ "maxItems": 2,
+ "items": {
+ "type": "number"
+ }
+ },
+ "tile": {
+ "type": "number"
+ }
+ }
+ },
+ "new_file_freq": {
+ "type": "string",
+ "pattern": "^(0*[1-9][0-9]* (seconds|minutes|hours|days|months|years), )*0*[1-9][0-9]* (seconds|minutes|hours|days|months|years)$"
+ },
+ "start_time": {
+ "type": "array",
+ "minItems": 6,
+ "maxItems": 6,
+ "items": {
+ "type": "integer",
+ "minimum": 0
+ }
+ },
+ "file_duration": {
+ "type": "string",
+ "pattern": "^(0*[1-9][0-9]* (seconds|minutes|hours|days|months|years), )*0*[1-9][0-9]* (seconds|minutes|hours|days|months|years)$"
+ },
+ "is_ocean": {
+ "type": "boolean"
+ },
+ "kind": {
+ "type": "string",
+ "enum": ["r4", "r8", "i4", "i8"]
+ },
+ "module": {
+ "type": "string",
+ "minLength": 1
+ },
+ "reduction": {
+ "type": "string",
+ "pattern": "^average$|^min$|^max$|^none$|^rms$|^sum$|^diurnal[1-9]+|^pow[1-9]+"
+ },
+ "varlist": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "required": ["var_name"],
+ "additionalProperties": false,
+ "properties": {
+ "kind": {
+ "type": "string",
+ "enum": ["r4", "r8", "i4", "i8"]
+ },
+ "module": {
+ "type": "string",
+ "minLength": 1
+ },
+ "reduction": {
+ "type": "string",
+ "pattern": "^average$|^min$|^max$|^none$|^rms$|^sum$|^diurnal[1-9]+|^pow[1-9]+"
+ },
+ "var_name": {
+ "type": "string",
+ "minLength": 1
+ },
+ "write_var": {
+ "type": "boolean"
+ },
+ "output_name": {
+ "type": "string",
+ "minLength": 1
+ },
+ "long_name": {
+ "type": "string",
+ "minLength": 1
+ },
+ "attributes": {
+ "type": "array",
+ "minItems": 1,
+ "maxItems": 1,
+ "items": {
+ "type": "object",
+ "additionalProperties": {
+ "oneOf": [
+ {"type": "string"},
+ {"type": "number"},
+ {"type": "boolean"}
+ ]
+ }
+ }
+ },
+ "zbounds": {
+ "type": "string"
+ }
+ }
+ }
+ }
+ },
+ "allOf": [
+ {
+ "if": {
+ "required": ["kind"]
+ },
+ "else": {
+ "properties": {
+ "varlist": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "required": ["kind"]
+ }
+ }
+ }
+ }
+ },
+ {
+ "if": {
+ "required": ["module"]
+ },
+ "else": {
+ "properties": {
+ "varlist": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "required": ["module"]
+ }
+ }
+ }
+ }
+ },
+ {
+ "if": {
+ "required": ["reduction"]
+ },
+ "else": {
+ "properties": {
+ "varlist": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "required": ["reduction"]
+ }
+ }
+ }
+ }
+ }
+ ]
+ }
+ }
+ }
+}
diff --git a/lib/fre/gfdl_msd_schemas/FMS/field_table.json b/lib/fre/gfdl_msd_schemas/FMS/field_table.json
new file mode 100644
index 00000000..64e069cb
--- /dev/null
+++ b/lib/fre/gfdl_msd_schemas/FMS/field_table.json
@@ -0,0 +1,48 @@
+{
+ "title": "Field table for field_manager",
+ "type": "object",
+ "properties": {
+ "field_table": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "properties": {
+ "field_type": {
+ "type": "string"
+ },
+ "modlist": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "properties": {
+ "model_type": {
+ "type": "string",
+ "enum": ["coupler_mod", "atmos_mod", "ocean_mod", "land_mod", "ice_mod"]
+ },
+ "varlist": {
+ "type": "array",
+ "items": {
+ "type": "object",
+ "properties": {
+ "variable": {
+ "type": "string"
+ }
+ },
+ "required": ["variable"],
+ "additionalProperties": true
+ }
+ }
+ },
+ "required": ["model_type"],
+ "additionalProperties": false
+ }
+ }
+ },
+ "required": ["field_type"],
+ "additionalProperties": false
+ }
+ }
+ },
+ "required": ["field_table"],
+ "additionalProperties": false
+}
diff --git a/lib/fre/gfdl_msd_schemas/FRE/fre_make.json b/lib/fre/gfdl_msd_schemas/FRE/fre_make.json
new file mode 100644
index 00000000..44a153f4
--- /dev/null
+++ b/lib/fre/gfdl_msd_schemas/FRE/fre_make.json
@@ -0,0 +1,212 @@
+{
+ "$schema": "http://json-schema.org/draft-06/schema#",
+ "type": "object",
+ "additionalProperties": false,
+ "properties": {
+ "name": {
+ "description": "The name of the experiment",
+ "type": "string"
+ },
+ "platform": {
+ "description": "The platforms listed in the command",
+ "type": "string"
+ },
+ "target": {
+ "description": "The targets listed in the command",
+ "type": "string"
+ },
+ "build": {
+ "type": "object",
+ "additionalProperties": false,
+ "properties": {
+ "compileYaml": {
+ "description": "Path to the compile yaml.",
+ "type": "string"
+ },
+ "platformYaml": {
+ "description": "Path to the platform yaml.",
+ "type": "string"
+ }
+ }
+ },
+ "compile": {
+ "description": "The source code descriptions",
+ "$ref": "#/$defs/Compile"
+ },
+ "platforms": {
+ "description": "FRE platforms",
+ "type": "array",
+ "items": {"$ref": "#/$defs/Platform"}
+ }
+ },
+ "$defs": {
+ "Compile": {
+ "type": "object",
+ "properties": {
+ "experiment": {
+ "description": "The name of the model",
+ "type": "string"
+ },
+ "container_addlibs": {
+ "description": "Libraries and packages needed for linking in the container",
+ "oneOf": [
+ {"type": "null"},
+ {
+ "type": "array",
+ "items": {"type": "string"}
+ }
+ ]
+ },
+ "baremetal_linkerflags": {
+ "description": "Linker flags of libraries and packages needed for linking in the bare-metal build",
+ "oneOf": [
+ {"type": "null"},
+ {
+ "type": "array",
+ "items": {"type": "string"}
+ }
+ ]
+ },
+ "src": {
+ "type": "array",
+ "items": {"$ref": "#/$defs/Src"}
+ }
+ }
+ },
+ "Src": {
+ "type": "object",
+ "properties": {
+ "component": {
+ "description": "The name of the model component",
+ "type": "string"
+ },
+ "repo": {
+ "anyOf": [
+ {
+ "description": "The URL of the code repository",
+ "type": "array",
+ "items": {
+ "type": "string",
+ "format": "uri",
+ "qt-uri-protocols": [
+ "https"
+ ],
+ "qt-uri-extensions": [
+ ".git"
+ ]
+ }
+ },
+ {
+ "description": "The URL of the code repository",
+ "type": "string",
+ "format": "uri",
+ "qt-uri-protocols": [
+ "https"
+ ],
+ "qt-uri-extensions": [
+ ".git"
+ ]
+ }
+ ]
+ },
+ "cppdefs": {
+ "description": "String of CPPDEFs to include in compiling the component",
+ "type": "string"
+ },
+ "branch": {
+ "anyOf": [
+ {
+ "description": "The version of code to clone",
+ "type": "array",
+ "items": {
+ "type": "string"
+ }
+ },
+ {
+ "description": "The version of code to clone",
+ "type": "string"
+ }
+ ]
+ },
+ "otherFlags": {
+ "description": "String of Include flags necessary to retrieve other code needed",
+ "type": "string"
+ },
+ "requires": {
+ "description": "list of componets that this component depends on",
+ "type": "array",
+ "items": {"type": "string"}
+ },
+ "paths": {
+ "description": "A list of the paths in the component to compile",
+ "type": "array",
+ "items": {"type": "string"}
+ },
+ "doF90Cpp": {
+ "description": "True if the preprocessor needs to be run",
+ "type": "boolean"
+ },
+ "makeOverrides": {
+ "description": "Overrides openmp target for MOM6",
+ "type": "string"
+ }
+ }
+ },
+ "Platform": {
+ "type": "object",
+ "properties": {
+ "name": {
+ "description": "The name of the platform",
+ "type": "string"
+ },
+ "compiler": {
+ "description": "The compiler used to build the model",
+ "type": "string"
+ },
+ "modulesInit": {
+ "description": "Array of commands to run before loading modules",
+ "type": "array",
+ "items": {"type": "string"}
+ },
+ "modules": {
+ "description": "List (array) of modules to load",
+ "type": "array",
+ "items": {"type": "string"}
+ },
+ "fc": {
+ "description": "The Fortran compiler",
+ "type": "string"
+ },
+ "cc": {
+ "description": "The C compiler",
+ "type": "string"
+ },
+ "mkTemplate": {
+ "description": "Path to the mk template file",
+ "type": "string"
+ },
+ "modelRoot": {
+ "description": "Path to the root for all model install files",
+ "type": "string"
+ },
+ "RUNenv": {
+ "description": "Commands needed at the beginning of a RUN in dockerfile",
+ "type": ["array"],
+ "items": {"type": "string"}
+ },
+ "container": {
+ "description": "True/False if using container to compile",
+ "type": "boolean"
+ },
+ "containerBuild": {
+ "description": "Program used to build the container",
+ "type": "string"
+ },
+ "containerRun": {
+ "description": "Program used to run the container",
+ "type": "string"
+ }
+ }
+ }
+ }
+}
diff --git a/lib/fre/gfdl_msd_schemas/README.md b/lib/fre/gfdl_msd_schemas/README.md
new file mode 100644
index 00000000..7fd70f3a
--- /dev/null
+++ b/lib/fre/gfdl_msd_schemas/README.md
@@ -0,0 +1,26 @@
+# About
+This repository contains schemas for various types of YAML files that are used
+by FMS and FRE at GFDL. These schemas are written in the
+[JSON Schema](https://json-schema.org/) format, and are intended for use with
+the `validate_schema.py` tool in
+[fms_yaml_tools](https://github.com/NOAA-GFDL/fms_yaml_tools).
+
+# Inventory
+* `FMS/data_table.json`
+* `FMS/diag_table.json`
+* `FMS/field_table.json`
+
+# Disclaimer
+
+The United States Department of Commerce (DOC) GitHub project code is provided
+on an 'as is' basis and the user assumes responsibility for its use. DOC has
+relinquished control of the information and no longer has responsibility to
+protect the integrity, confidentiality, or availability of the information. Any
+claims against the Department of Commerce stemming from the use of its GitHub
+project will be governed by all applicable Federal law. Any reference to
+specific commercial products, processes, or services by service mark,
+trademark, manufacturer, or otherwise, does not constitute or imply their
+endorsement, recommendation or favoring by the Department of Commerce. The
+Department of Commerce seal and logo, or the seal and logo of a DOC bureau,
+shall not be used in any manner to imply endorsement of any commercial product
+or activity by DOC or the United States Government.
diff --git a/lib/fre/lazy_group.py b/lib/fre/lazy_group.py
new file mode 100644
index 00000000..1ceefca7
--- /dev/null
+++ b/lib/fre/lazy_group.py
@@ -0,0 +1,43 @@
+'''
+for lazy-style loading of commands and subcommands while using click
+see https://click.palletsprojects.com/en/8.1.x/complex/
+'''
+
+import importlib
+import click
+
+class LazyGroup(click.Group):
+ ''' class defining lazygroup command/subcommand loading '''
+ def __init__(self, *args, lazy_subcommands=None, **kwargs):
+ super().__init__(*args, **kwargs)
+ # lazy_subcommands is a map of the form:
+ #
+ # {command-name} -> {module-name}.{command-object-name}
+ #
+ self.lazy_subcommands = lazy_subcommands or {}
+
+ def list_commands(self, ctx):
+ base = super().list_commands(ctx)
+ lazy = sorted(self.lazy_subcommands.keys())
+ return base + lazy
+
+ def get_command(self, ctx, cmd_name):
+ if cmd_name in self.lazy_subcommands:
+ return self._lazy_load(cmd_name)
+ return super().get_command(ctx, cmd_name)
+
+ def _lazy_load(self, cmd_name):
+ # lazily loading a command, first get the module name and attribute name
+ import_path = self.lazy_subcommands[cmd_name]
+ modname, cmd_object_name = import_path.rsplit(".", 1)
+ # do the import
+ mod = importlib.import_module(modname, package="fre")
+ # get the Command object from that module
+ cmd_object = getattr(mod, cmd_object_name)
+ # check the result to make debugging easier
+ if not isinstance(cmd_object, click.BaseCommand):
+ raise ValueError(
+ f"Lazy loading of {import_path} failed by returning "
+ "a non-command object"
+ )
+ return cmd_object
diff --git a/lib/fre/list/__init__.py b/lib/fre/list/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/list/frelist.py b/lib/fre/list/frelist.py
new file mode 100644
index 00000000..fea3b6ea
--- /dev/null
+++ b/lib/fre/list/frelist.py
@@ -0,0 +1,20 @@
+''' fre list '''
+
+import click
+
+from .frelistexample import list_test_function
+
+@click.group(help=click.style(" - access fre list subcommands", fg=(232,204,91)))
+def list_cli():
+ ''' entry point to fre list click commands '''
+
+@list_cli.command()
+@click.option('--uppercase', '-u', is_flag=True, help = 'Print statement in uppercase.')
+@click.pass_context
+def function(context, uppercase):
+ # pylint: disable=unused-argument
+ """ - Execute fre list test """
+ context.forward(list_test_function)
+
+if __name__ == "__main__":
+ list_cli()
diff --git a/lib/fre/list/frelistexample.py b/lib/fre/list/frelistexample.py
new file mode 100644
index 00000000..7e40e195
--- /dev/null
+++ b/lib/fre/list/frelistexample.py
@@ -0,0 +1,18 @@
+"""
+experimentation file for integrating one file's functions into main prototype fre file
+authored by Bennett.Chang@noaa.gov | bcc2761
+NOAA | GFDL
+"""
+
+import click
+
+@click.command()
+def list_test_function(uppercase=None):
+ """Execute fre list testfunction2."""
+ statement = "testingtestingtestingtesting"
+ if uppercase:
+ statement = statement.upper()
+ click.echo(statement)
+
+if __name__ == '__main__':
+ list_test_function()
diff --git a/lib/fre/make/README.md b/lib/fre/make/README.md
new file mode 100644
index 00000000..961e564c
--- /dev/null
+++ b/lib/fre/make/README.md
@@ -0,0 +1,180 @@
+# **Fremake Canopy**
+Through the fre-cli, `fre make` can be used to create and run a checkout script, makefile, and compile a model.
+
+* Fremake Canopy Supports:
+ - multiple targets; use `-t` flag to define each target
+ - bare-metal build
+ - container creation
+ - parallel checkouts for bare-metal build**
+
+** **Note: Users will not be able to create containers without access to podman**
+
+The fremake canopy fre-cli subcommands are described below ([Subtools](#subtools)), as well as a Guide on the order in which to use them ([Guide](#guide)).
+
+Additionally, as mentioned, multiple targets can be used more multiple target-platform combinations. Below is an example of this usage for both the bare-metal build and container build, using the AM5 model
+
+- [Bare-metal Example](#bare-metal-build)
+- [Bare-metal Multi-target Example](#bare-metal-build-multi-target)
+- [Container Example](#container-build)
+
+## **Usage (Users)**
+* Refer to fre-cli [README.md](https://github.com/NOAA-GFDL/fre-cli/blob/main/README.md) for foundational fre-cli usage guide and tips.
+
+## **Quickstart**
+### **Bare-metal Build:**
+```bash
+# Create checkout script
+fre make create-checkout -y am5.yaml -p ncrc5.intel23 -t prod
+
+# Create and run checkout script
+fre make create-checkout -y am5.yaml -p ncrc5.intel23 -t prod --execute
+
+# Create Makefile
+fre make create-makefile -y am5.yaml -p ncrc5.intel23 -t prod
+
+# Create the compile script
+fre make create-compile -y am5.yaml -p ncrc5.intel23 -t prod
+
+# Create and run the compile script
+fre make create-compile -y am5.yaml -p ncrc5.intel23 -t prod --execute
+```
+### **Bare-metal Build Multi-target:**
+```bash
+# Create checkout script
+fre make create-checkout -y am5.yaml -p ncrc5.intel23 -t prod -t debug
+
+# Create and run checkout script
+fre make create-checkout -y am5.yaml -p ncrc5.intel23 -t prod -t debug --execute
+
+# Create Makefile
+fre make create-makefile -y am5.yaml -p ncrc5.intel23 -t prod -t debug
+
+# Create the compile script
+fre make create-compile -y am5.yaml -p ncrc5.intel23 -t prod -t debug
+
+# Create and run the compile script
+fre make create-compile -y am5.yaml -p ncrc5.intel23 -t prod -t debug --execute
+```
+
+### **Container Build:**
+In order for the container to build successfully, a `-npc`, or `--no-parallel-checkout` is needed.
+```bash
+# Create checkout script
+fre make create-checkout -y am5.yaml -p hpcme.2023 -t prod -npc
+
+# Create and run checkout script
+fre make create-checkout -y am5.yaml -p hpcme.2023 -t prod -npc --execute
+
+# Create Makefile
+fre make create-makefile -y am5.yaml -p hpcme.2023 -t prod
+
+# Create Dockerfile
+fre make create-dockerfile -y am5.yaml -p hpcme.2023 -t prod
+
+# Create and run the Dockerfile
+fre make create-dockerfile -y am5.yaml -p hpcme.2023 -t prod --execute
+```
+
+### **Run all of fremake:**
+```bash
+# Bare-metal
+fre make run-fremake -y am5.yaml -p ncrc5.intel23 -t prod
+
+# Container
+fre make run-fremake -y am5.yaml -p hpcme.2023 -t prod -npc
+```
+
+## Subtools
+- `fre make create-checkout [options]`
+ - Purpose: Creates the checkout script and can check out source code (with execute option)
+ - Options:
+ - `-y, --yamlfile [experiment yaml] (required)`
+ - `-p, --platform [platform] (required)`
+ - `-t, --target [target] (required)`
+ - `-j, --jobs [number of jobs to run simultneously]`
+ - `-npc, --no-parallel-checkout (for container build)`
+ - `-e, --execute`
+
+- `fre make create-makefile [options]`
+ - Purpose: Creates the makefile
+ - Options:
+ - `-y, --yamlfile [experiment yaml] (required)`
+ - `-p, --platform [platform] (required)`
+ - `-t, --target [target] (required)`
+
+- `fre make create-compile [options]`
+ - Purpose: Creates the compile script and compiles the model (with execute option)
+ - Options:
+ - `-y, --yamlfile [experiment yaml] (required)`
+ - `-p, --platform [platform] (required)`
+ - `-t, --target [target] (required)`
+ - `-j, --jobs [number of jobs to run simultneously]`
+ - `-n, --parallel [number of concurrent modile compiles]`
+ - `-e, --execute`
+
+- `fre make create-dockerfile [options]`
+ - Purpose: Creates the dockerfile and creates the container (with execute option)
+ - With the creation of the dockerfile, the Makefile, checkout script, and any other necessary script is copied into the container from a temporary location
+ - Options:
+ - `-y, --yamlfile [experiment yaml] (required)`
+ - `-p, --platform [platform] (required)`
+ - `-t, --target [target] (required)`
+ - `-e, --execute`
+
+- `fre make run-fremake [options]`
+ - Purpose: Create the checkout script, Makefile, compile script, and dockerfile (platform dependent) for the compilation of the model
+ - Options:
+ - `-y, --yamlfile [experiment yaml] (required)`
+ - `-p, --platform [platform] (required)`
+ - `-t, --target [target] (required)`
+ - `-npc, --no-parallel-checkout (for container build)`
+ - `-j, --jobs [number of jobs to run simultneously]`
+ - `-n, --parallel [number of concurrent modile compiles]`
+
+## Guide
+In order to use the `fre make` tools, remember to create a combined yaml first. This can be done with the `fre yamltools combine-yamls` tool. This combines the model, compile, platform, experiment, and any analysis yamls into ONE yaml file for parsing and validation.
+
+To combine:
+`fre yamltools combine-yamls -y [model yaml file] -e [experiment name] -p [platform] -t [target]`
+
+### **Bare-metal Build:**
+```bash
+# Create checkout script
+fre make create-checkout -y [model yaml file] -p [platform] -t [target]
+
+# Create and run checkout script
+fre make create-checkout -y [model yaml file] -p [platform] -t [target] --execute
+
+# Create Makefile
+fre make create-makefile -y [model yaml file] -p [platform] -t [target]
+
+# Creat the compile script
+fre make create-compile -y [model yaml file] -p [platform] -t [target]
+
+# Create and run the compile script
+fre make create-compile -y [model yaml file] -p [platform] -t [target] --execute
+
+# Run all of fremake
+fre make run-fremake -y [model yaml file] -p [platform] -t [target] [other options...]
+```
+
+### **Container Build:**
+For the container build, parallel checkouts are not supported, so the `-npc` options must be used for the checkout script. In addition the platform must be a container platform.
+
+***To reiterate, users will not be able to create containers unless they have podman access on gaea.***
+```bash
+# Create checkout script
+fre make create-checkout -y [model yaml file] -p [CONTAINER PLATFORM] -t [target] -npc
+
+# Create and run checkout script
+fre make create-checkout -y [model yaml file] -p [CONTAINER PLATFORM] -t [target] --execute
+
+# Create Makefile
+fre make create-makefile -y [model yaml file] -p [CONTAINER PLATFORM] -t [target]
+
+#Create a Dockerfile
+fre make create-dockerfile -y [model yaml file] -p [CONTAINER PLATFORM] -t [target]
+
+# Create and run the Dockerfile
+fre make create-dockerfile -y [model yaml file] -p [CONTAINER PLATFORM] -t [target] --execute
+```
diff --git a/lib/fre/make/__init__.py b/lib/fre/make/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/make/createCheckout.py b/lib/fre/make/createCheckout.py
new file mode 100644
index 00000000..fc4d2df4
--- /dev/null
+++ b/lib/fre/make/createCheckout.py
@@ -0,0 +1,116 @@
+#!/usr/bin/python3
+
+import os
+import subprocess
+import logging
+import sys
+import click
+import fre.yamltools.combine_yamls as cy
+from .gfdlfremake import varsfre, yamlfre, checkout, targetfre
+
+def checkout_create(yamlfile,platform,target,no_parallel_checkout,jobs,execute,verbose):
+ # Define variables
+ yml = yamlfile
+ name = yamlfile.split(".")[0]
+ run = execute
+ jobs = str(jobs)
+ pcheck = no_parallel_checkout
+
+ if pcheck:
+ pc = ""
+ else:
+ pc = " &"
+
+ if verbose:
+ logging.basicConfig(level=logging.INFO)
+ else:
+ logging.basicConfig(level=logging.ERROR)
+
+ srcDir="src"
+ checkoutScriptName = "checkout.sh"
+ baremetalRun = False # This is needed if there are no bare metal runs
+
+ ## Split and store the platforms and targets in a list
+ plist = platform
+ tlist = target
+
+ # Combine model, compile, and platform yamls
+ # Default behavior - combine yamls / rewrite combined yaml
+ comb = cy.init_compile_yaml(yml,platform,target)
+ full_combined = cy.get_combined_compileyaml(comb)
+
+ ## Get the variables in the model yaml
+ freVars = varsfre.frevars(full_combined)
+
+ ## Open the yaml file, validate the yaml, and parse as fremakeYaml
+ modelYaml = yamlfre.freyaml(full_combined,freVars)
+ fremakeYaml = modelYaml.getCompileYaml()
+
+ ## Error checking the targets
+ for targetName in tlist:
+ target = targetfre.fretarget(targetName)
+
+ ## Loop through the platforms specified on the command line
+ ## If the platform is a baremetal platform, write the checkout script and run it once
+ ## This should be done separately and serially because bare metal platforms should all be using
+ ## the same source code.
+ for platformName in plist:
+ if modelYaml.platforms.hasPlatform(platformName):
+ pass
+ else:
+ raise ValueError (platformName + " does not exist in platforms.yaml")
+ ( compiler, modules, modulesInit, fc, cc, modelRoot,
+ iscontainer, mkTemplate, containerBuild, ContainerRun,
+ RUNenv ) = modelYaml.platforms.getPlatformFromName(platformName)
+
+ ## Create the source directory for the platform
+ if iscontainer is False:
+ srcDir = modelRoot + "/" + fremakeYaml["experiment"] + "/src"
+ # if the source directory does not exist, it is created
+ if not os.path.exists(srcDir):
+ os.system("mkdir -p " + srcDir)
+ # if the checkout script does not exist, it is created
+ if not os.path.exists(srcDir+"/checkout.sh"):
+ freCheckout = checkout.checkout("checkout.sh",srcDir)
+ freCheckout.writeCheckout(modelYaml.compile.getCompileYaml(),jobs,pc)
+ freCheckout.finish(pc)
+ # Make checkout script executable
+ os.chmod(srcDir+"/checkout.sh", 0o744)
+ print("\nCheckout script created in "+ srcDir + "/checkout.sh \n")
+
+ # Run the checkout script
+ if run is True:
+ freCheckout.run()
+ else:
+ sys.exit()
+ else:
+ print("\nCheckout script PREVIOUSLY created in "+ srcDir + "/checkout.sh \n")
+ if run == True:
+ try:
+ subprocess.run(args=[srcDir+"/checkout.sh"], check=True)
+ except:
+ print("\nThere was an error with the checkout script "+srcDir+"/checkout.sh.",
+ "\nTry removing test folder: " + modelRoot +"\n")
+ raise
+ else:
+ sys.exit()
+
+ else:
+ image="ecpe4s/noaa-intel-prototype:2023.09.25"
+ bldDir = modelRoot + "/" + fremakeYaml["experiment"] + "/exec"
+ tmpDir = "tmp/"+platformName
+ freCheckout = checkout.checkoutForContainer("checkout.sh", srcDir, tmpDir)
+ freCheckout.writeCheckout(modelYaml.compile.getCompileYaml(),jobs,pc)
+ freCheckout.finish(pc)
+ print("\nCheckout script created at " + tmpDir + "/checkout.sh" + "\n")
+
+@click.command()
+def _checkout_create(yamlfile,platform,target,no_parallel_checkout,jobs,execute,verbose):
+ '''
+ Decorator for calling checkout_create - allows the decorated version
+ of the function to be separate from the undecorated version
+ '''
+ return checkout_create(yamlfile,platform,target,no_parallel_checkout,jobs,execute,verbose)
+
+if __name__ == "__main__":
+ checkout_create()
diff --git a/lib/fre/make/createCompile.py b/lib/fre/make/createCompile.py
new file mode 100644
index 00000000..36068329
--- /dev/null
+++ b/lib/fre/make/createCompile.py
@@ -0,0 +1,99 @@
+#!/usr/bin/python3
+
+import os
+import sys
+import logging
+from pathlib import Path
+from multiprocessing.dummy import Pool
+import click
+from .gfdlfremake import varsfre, yamlfre, targetfre, buildBaremetal
+import fre.yamltools.combine_yamls as cy
+
+def compile_create(yamlfile,platform,target,jobs,parallel,execute,verbose):
+ # Define variables
+ yml = yamlfile
+ name = yamlfile.split(".")[0]
+ nparallel = parallel
+ jobs = str(jobs)
+ run = execute
+
+ if verbose:
+ logging.basicCOnfig(level=logging.INFO)
+ else:
+ logging.basicConfig(level=logging.ERROR)
+
+ srcDir="src"
+ checkoutScriptName = "checkout.sh"
+ baremetalRun = False # This is needed if there are no bare metal runs
+
+ ## Split and store the platforms and targets in a list
+ plist = platform
+ tlist = target
+
+ # Combined compile yaml file
+ combined = Path(f"combined-{name}.yaml")
+
+ ## If combined yaml exists, note message of its existence
+ ## If combined yaml does not exist, combine model, compile, and platform yamls
+ full_combined = cy.combined_compile_existcheck(combined,yml,platform,target)
+
+ ## Get the variables in the model yaml
+ freVars = varsfre.frevars(full_combined)
+
+ ## Open the yaml file and parse as fremakeYaml
+ modelYaml = yamlfre.freyaml(full_combined,freVars)
+ fremakeYaml = modelYaml.getCompileYaml()
+
+ ## Error checking the targets
+ for targetName in tlist:
+ target = targetfre.fretarget(targetName)
+
+ fremakeBuildList = []
+ ## Loop through platforms and targets
+ for platformName in plist:
+ for targetName in tlist:
+ target = targetfre.fretarget(targetName)
+ if modelYaml.platforms.hasPlatform(platformName):
+ pass
+ else:
+ raise ValueError (platformName + " does not exist in " + modelYaml.combined.get("compile").get("platformYaml"))
+
+ (compiler,modules,modulesInit,fc,cc,modelRoot,iscontainer,mkTemplate,containerBuild,ContainerRun,RUNenv)=modelYaml.platforms.getPlatformFromName(platformName)
+ ## Make the bldDir based on the modelRoot, the platform, and the target
+ srcDir = modelRoot + "/" + fremakeYaml["experiment"] + "/src"
+ ## Check for type of build
+ if iscontainer is False:
+ baremetalRun = True
+ bldDir = modelRoot + "/" + fremakeYaml["experiment"] + "/" + platformName + "-" + target.gettargetName() + "/exec"
+ os.system("mkdir -p " + bldDir)
+ ## Create a list of compile scripts to run in parallel
+ fremakeBuild = buildBaremetal.buildBaremetal(exp = fremakeYaml["experiment"],
+ mkTemplatePath = mkTemplate,
+ srcDir = srcDir,
+ bldDir = bldDir,
+ target = target,
+ modules = modules,
+ modulesInit = modulesInit,
+ jobs = jobs)
+ for c in fremakeYaml['src']:
+ fremakeBuild.writeBuildComponents(c)
+ fremakeBuild.writeScript()
+ fremakeBuildList.append(fremakeBuild)
+ click.echo("\nCompile script created at " + bldDir + "/compile.sh" + "\n")
+ if run:
+ if baremetalRun:
+ pool = Pool(processes=nparallel) # Create a multiprocessing Pool
+ pool.map(buildBaremetal.fremake_parallel,fremakeBuildList) # process data_inputs iterable with pool
+ else:
+ sys.exit()
+
+@click.command()
+def _compile_create(yamlfile,platform,target,jobs,parallel,execute,verbose):
+ '''
+ Decorator for calling compile_create - allows the decorated version
+ of the function to be separate from the undecorated version
+ '''
+ return compile_create(yamlfile,platform,target,jobs,parallel,execute,verbose)
+
+if __name__ == "__main__":
+ compile_create()
diff --git a/lib/fre/make/createDocker.py b/lib/fre/make/createDocker.py
new file mode 100644
index 00000000..59b73ee9
--- /dev/null
+++ b/lib/fre/make/createDocker.py
@@ -0,0 +1,89 @@
+#!/usr/bin/python3
+
+import os
+import sys
+from pathlib import Path
+import click
+#from .gfdlfremake import varsfre, targetfre, makefilefre, platformfre, yamlfre, buildDocker
+from .gfdlfremake import varsfre, targetfre, yamlfre, buildDocker
+import fre.yamltools.combine_yamls as cy
+
+def dockerfile_create(yamlfile,platform,target,execute):
+ srcDir="src"
+ checkoutScriptName = "checkout.sh"
+ baremetalRun = False # This is needed if there are no bare metal runs
+ ## Split and store the platforms and targets in a list
+ plist = platform
+ tlist = target
+ yml = yamlfile
+ name = yamlfile.split(".")[0]
+ run = execute
+
+ # Combined compile yaml file
+ combined = Path(f"combined-{name}.yaml")
+
+ ## If combined yaml exists, note message of its existence
+ ## If combined yaml does not exist, combine model, compile, and platform yamls
+ full_combined = cy.combined_compile_existcheck(combined,yml,platform,target)
+
+ ## Get the variables in the model yaml
+ freVars = varsfre.frevars(full_combined)
+
+ ## Open the yaml file and parse as fremakeYaml
+ modelYaml = yamlfre.freyaml(full_combined,freVars)
+ fremakeYaml = modelYaml.getCompileYaml()
+
+ fremakeBuildList = []
+ ## Loop through platforms and targets
+ for platformName in plist:
+ for targetName in tlist:
+ targetObject = targetfre.fretarget(targetName)
+ if modelYaml.platforms.hasPlatform(platformName):
+ pass
+ else:
+ raise ValueError (platformName + " does not exist in " + \
+ modelYaml.combined.get("compile").get("platformYaml"))
+
+ ( compiler, modules, modulesInit, fc, cc, modelRoot,
+ iscontainer, mkTemplate, containerBuild, containerRun,
+ RUNenv ) = modelYaml.platforms.getPlatformFromName(platformName)
+
+ ## Make the bldDir based on the modelRoot, the platform, and the target
+ srcDir = modelRoot + "/" + fremakeYaml["experiment"] + "/src"
+ ## Check for type of build
+ if iscontainer is True:
+ image="ecpe4s/noaa-intel-prototype:2023.09.25"
+ bldDir = modelRoot + "/" + fremakeYaml["experiment"] + "/exec"
+ tmpDir = "tmp/"+platformName
+
+ dockerBuild = buildDocker.container(base = image,
+ exp = fremakeYaml["experiment"],
+ libs = fremakeYaml["container_addlibs"],
+ RUNenv = RUNenv,
+ target = targetObject)
+ dockerBuild.writeDockerfileCheckout("checkout.sh", tmpDir+"/checkout.sh")
+ dockerBuild.writeDockerfileMakefile(tmpDir+"/Makefile", tmpDir+"/linkline.sh")
+
+ for c in fremakeYaml['src']:
+ dockerBuild.writeDockerfileMkmf(c)
+
+ dockerBuild.writeRunscript(RUNenv,containerRun,tmpDir+"/execrunscript.sh")
+ currDir = os.getcwd()
+ click.echo("\ntmpDir created in " + currDir + "/tmp")
+ click.echo("Dockerfile created in " + currDir +"\n")
+
+ if run:
+ dockerBuild.build(containerBuild, containerRun)
+ else:
+ sys.exit()
+
+@click.command()
+def _dockerfile_create(yamlfile,platform,target,execute):
+ '''
+ Decorator for calling dockerfile_create - allows the decorated version
+ of the function to be separate from the undecorated version
+ '''
+ return dockerfile_create(yamlfile,platform,target,execute)
+
+if __name__ == "__main__":
+ dockerfile_create()
diff --git a/lib/fre/make/createMakefile.py b/lib/fre/make/createMakefile.py
new file mode 100644
index 00000000..eaf340dd
--- /dev/null
+++ b/lib/fre/make/createMakefile.py
@@ -0,0 +1,89 @@
+#!/usr/bin/python3
+
+import os
+from pathlib import Path
+
+import click
+
+from .gfdlfremake import makefilefre, varsfre, targetfre, yamlfre
+import fre.yamltools.combine_yamls as cy
+
+def makefile_create(yamlfile,platform,target):
+ srcDir="src"
+ checkoutScriptName = "checkout.sh"
+ baremetalRun = False # This is needed if there are no bare metal runs
+ ## Split and store the platforms and targets in a list
+ plist = platform
+ tlist = target
+ yml = yamlfile
+ name = yamlfile.split(".")[0]
+
+ combined = Path(f"combined-{name}.yaml")
+
+ ## If combined yaml exists, note message of its existence
+ ## If combined yaml does not exist, combine model, compile, and platform yamls
+ full_combined = cy.combined_compile_existcheck(combined,yml,platform,target)
+
+ ## Get the variables in the model yaml
+ freVars = varsfre.frevars(full_combined)
+
+ ## Open the yaml file and parse as fremakeYaml
+ modelYaml = yamlfre.freyaml(full_combined,freVars)
+ fremakeYaml = modelYaml.getCompileYaml()
+
+ fremakeBuildList = []
+ ## Loop through platforms and targets
+ for platformName in plist:
+ for targetName in tlist:
+ targetObject = targetfre.fretarget(targetName)
+ if modelYaml.platforms.hasPlatform(platformName):
+ pass
+ else:
+ raise ValueError (platformName + " does not exist in " + modelYaml.combined.get("compile").get("platformYaml"))
+
+ (compiler,modules,modulesInit,fc,cc,modelRoot,iscontainer,mkTemplate,containerBuild,ContainerRun,RUNenv)=modelYaml.platforms.getPlatformFromName(platformName)
+ ## Make the bldDir based on the modelRoot, the platform, and the target
+ srcDir = modelRoot + "/" + fremakeYaml["experiment"] + "/src"
+ ## Check for type of build
+ if iscontainer is False:
+ baremetalRun = True
+ bldDir = modelRoot + "/" + fremakeYaml["experiment"] + "/" + platformName + "-" + targetObject.gettargetName() + "/exec"
+ os.system("mkdir -p " + bldDir)
+ ## Create the Makefile
+ freMakefile = makefilefre.makefile(exp = fremakeYaml["experiment"],
+ libs = fremakeYaml["baremetal_linkerflags"],
+ srcDir = srcDir,
+ bldDir = bldDir,
+ mkTemplatePath = mkTemplate)
+ # Loop through components and send the component name, requires, and overrides for the Makefile
+ for c in fremakeYaml['src']:
+ freMakefile.addComponent(c['component'],c['requires'],c['makeOverrides'])
+ freMakefile.writeMakefile()
+ click.echo("\nMakefile created at " + bldDir + "/Makefile" + "\n")
+ else:
+ image="ecpe4s/noaa-intel-prototype:2023.09.25"
+ bldDir = modelRoot + "/" + fremakeYaml["experiment"] + "/exec"
+ tmpDir = "tmp/"+platformName
+ freMakefile = makefilefre.makefileContainer(exp = fremakeYaml["experiment"],
+ libs = fremakeYaml["container_addlibs"],
+ srcDir = srcDir,
+ bldDir = bldDir,
+ mkTemplatePath = mkTemplate,
+ tmpDir = tmpDir)
+
+ # Loop through compenents and send the component name and requires for the Makefile
+ for c in fremakeYaml['src']:
+ freMakefile.addComponent(c['component'],c['requires'],c['makeOverrides'])
+ freMakefile.writeMakefile()
+ click.echo("\nMakefile created at " + bldDir + "/Makefile" + "\n")
+
+@click.command()
+def _makefile_create(yamlfile,platform,target):
+ '''
+ Decorator for calling makefile_create - allows the decorated version
+ of the function to be separate from the undecorated version
+ '''
+ return makefile_create(yamlfile,platform,target)
+
+if __name__ == "__main__":
+ makefile_create()
diff --git a/lib/fre/make/fremake.py b/lib/fre/make/fremake.py
new file mode 100644
index 00000000..6dfb9a34
--- /dev/null
+++ b/lib/fre/make/fremake.py
@@ -0,0 +1,213 @@
+import click
+from fre.make import createCheckout
+from fre.make import createMakefile
+from fre.make import createCompile
+from fre.make import createDocker
+from fre.make import runFremake
+
+yamlfile_opt_help = """Experiment yaml compile FILE
+"""
+experiment_opt_help = """Name of experiment"""
+platform_opt_help = """Hardware and software FRE platform space separated list of STRING(s).
+This sets platform-specific data and instructions
+"""
+target_opt_help = """a space separated list of STRING(s) that defines compilation settings and
+linkage directives for experiments. Predefined targets refer to groups of directives that exist in
+the mkmf template file (referenced in buildDocker.py). Possible predefined targets include 'prod',
+'openmp', 'repro', 'debug, 'hdf5'; however 'prod', 'repro', and 'debug' are mutually exclusive
+(cannot not use more than one of these in the target list). Any number of targets can be used.
+"""
+parallel_opt_help = """Number of concurrent model compiles (default 1)
+"""
+jobs_opt_help = """Number of jobs to run simultaneously. Used for make -jJOBS and git clone
+recursive --jobs=JOBS
+"""
+no_parallel_checkout_opt_help = """Use this option if you do not want a parallel checkout.
+The default is to have parallel checkouts.
+"""
+verbose_opt_help = """Get verbose messages (repeat the option to increase verbosity level)
+"""
+
+
+
+@click.group(help=click.style(" - access fre make subcommands", fg=(210,73,57)))
+def make_cli():
+ pass
+
+@make_cli.command()
+@click.option("-y",
+ "--yamlfile",
+ type = str,
+ help = yamlfile_opt_help,
+ required = True) # use click.option() over click.argument(), we want help statements
+@click.option("-p",
+ "--platform",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = platform_opt_help, required = True)
+@click.option("-t", "--target",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = target_opt_help,
+ required = True)
+@click.option("-n",
+ "--parallel",
+ type = int,
+ metavar = '',
+ default = 1,
+ help = parallel_opt_help)
+@click.option("-j",
+ "--jobs",
+ type = int,
+ metavar = '',
+ default = 4,
+ help = jobs_opt_help)
+@click.option("-npc",
+ "--no-parallel-checkout",
+ is_flag = True,
+ help = no_parallel_checkout_opt_help)
+@click.option("-e",
+ "--execute",
+ is_flag = True,
+ default = False,
+ help = "Use this to run the created compilation script.")
+@click.option("-v",
+ "--verbose",
+ is_flag = True,
+ help = verbose_opt_help)
+@click.pass_context
+def run_fremake(context, yamlfile, platform, target, parallel, jobs, no_parallel_checkout, execute, verbose):
+ """ - Perform all fremake functions to run checkout and compile model"""
+ context.forward(runFremake._fremake_run)
+
+####
+@make_cli.command()
+@click.option("-y",
+ "--yamlfile",
+ type = str,
+ help = yamlfile_opt_help,
+ required = True) # use click.option() over click.argument(), we want help statements
+@click.option("-p",
+ "--platform",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = platform_opt_help,
+ required = True)
+@click.option("-t", "--target",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = target_opt_help,
+ required = True)
+@click.option("-j",
+ "--jobs",
+ type = int,
+ metavar = '',
+ default = 4,
+ help = jobs_opt_help)
+@click.option("-npc",
+ "--no-parallel-checkout",
+ is_flag = True,
+ help = no_parallel_checkout_opt_help)
+@click.option("--execute",
+ is_flag = True,
+ default = False,
+ help = "Use this to run the created checkout script.")
+@click.option("-v",
+ "--verbose",
+ is_flag = True,
+ help = verbose_opt_help)
+@click.pass_context
+def create_checkout(context,yamlfile,platform,target,no_parallel_checkout,jobs,execute,verbose):
+ """ - Write the checkout script """
+ context.forward(createCheckout._checkout_create)
+
+#####
+@make_cli.command
+@click.option("-y",
+ "--yamlfile",
+ type = str,
+ help = yamlfile_opt_help,
+ required = True) # use click.option() over click.argument(), we want help statements
+@click.option("-p",
+ "--platform",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = platform_opt_help, required = True)
+@click.option("-t", "--target",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = target_opt_help,
+ required = True)
+@click.pass_context
+def create_makefile(context,yamlfile,platform,target):
+ """ - Write the makefile """
+ context.forward(createMakefile._makefile_create)
+
+#####
+
+@make_cli.command
+@click.option("-y",
+ "--yamlfile",
+ type = str,
+ help = yamlfile_opt_help,
+ required = True) # use click.option() over click.argument(), we want help statements
+@click.option("-p",
+ "--platform",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = platform_opt_help, required = True)
+@click.option("-t", "--target",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = target_opt_help,
+ required = True)
+@click.option("-j",
+ "--jobs",
+ type = int,
+ metavar = '',
+ default = 4,
+ help = jobs_opt_help)
+@click.option("-n",
+ "--parallel",
+ type = int,
+ metavar = '', default = 1,
+ help = parallel_opt_help)
+@click.option("--execute",
+ is_flag = True,
+ default = False,
+ help = "Use this to run the created checkout script.")
+@click.option("-v",
+ "--verbose",
+ is_flag = True,
+ help = verbose_opt_help)
+@click.pass_context
+def create_compile(context,yamlfile,platform,target,jobs,parallel,execute,verbose):
+ """ - Write the compile script """
+ context.forward(createCompile._compile_create)
+
+@make_cli.command
+@click.option("-y",
+ "--yamlfile",
+ type = str,
+ help = yamlfile_opt_help,
+ required = True) # use click.option() over click.argument(), we want help statements
+@click.option("-p",
+ "--platform",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = platform_opt_help, required = True)
+@click.option("-t", "--target",
+ multiple = True, # replaces nargs = -1, since click.option()
+ type = str,
+ help = target_opt_help,
+ required = True)
+@click.option("--execute",
+ is_flag = True,
+ help = "Build Dockerfile that has been generated by create-docker.")
+@click.pass_context
+def create_dockerfile(context,yamlfile,platform,target,execute):
+ """ - Write the dockerfile """
+ context.forward(createDocker._dockerfile_create)
+
+if __name__ == "__main__":
+ make_cli()
diff --git a/lib/fre/make/gfdlfremake/README.md b/lib/fre/make/gfdlfremake/README.md
new file mode 100644
index 00000000..18aa8d44
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/README.md
@@ -0,0 +1,147 @@
+# FREMAKE CANOPY
+The fremake canopy prototype embodies the checking out of source code and compilation of the model. This rewrite was developed with the idea of portability and flexibilty in mind.
+
+Fremake canopy is written in python and can support:
+- Bare-metal and container builds (with container platform)
+ - bare-metal build:
+ - supports parallel checkouts (default behavior)
+ - supports parallel model builds
+ - container:
+ - e4s container with spack stack package list
+- Multiple platforms and targets can be given (for multiple platform-target combinations)
+- One yaml format
+- Additional library support if needed by model
+
+Fremake has also been integrated as a fre-cli tool, creating a more modular and interactive experience.
+
+## Get Started
+Steps for how to build an example with the AM5 model, for both the bare-metal compilation and container, are outlined below.
+
+- ### [Set-up environment](#how-to-get-started)
+ - Fremake tools are provided through the fre-cli
+ - For more information on the fre-cli, see its associated github page: https://github.com/NOAA-GFDL/fre-cli
+- ### [Bare-metal build](#how-to-build-the-bare-metal-example)
+ - Note: make sure to have the correct experiment yaml, compile yaml, and platform yaml available for the model compilation
+- ### [Container build](#how-to-build-the-container-example)
+ - Note: make sure to have the correct experiment yaml, compile yaml and platform yaml available for the model compilation
+- ### [Full fremake: bare-metal and container](#how-to-run-complete-fremake)
+ - this can be done for either bare-metal or container builds
+ - this tool is available to give the option to run fremake fully, not in steps
+
+
+## YAMLS:
+experiment yaml: This is the main yaml and gives the path to the other yamls
+```yaml
+platformYaml: path to yaml containing the platforms
+layoutYaml: path to the yaml containing the layouts (not used currently)
+compileYaml: path to the yaml with the model source code and build information
+experiments: path to the yaml that has information about expeiment configurations (not currently used)
+
+```
+
+compile yaml: Containers information about source code and how to build
+```yaml
+experiment: name of the model
+container_addlibs: ["comma separated list of additional packages/libraries"]
+baremetal_linkerflags: ["comma separated list of linker flags needed for each additional package/library"]
+src:
+ - component: The name of the component model
+ requires: [Array of required components]
+ repo: scalar URL of the repo --OR-- [array of URLs to the code repos if more than one is required]
+ branch: scalar (or array of same size of repo) of the version of the code to clone
+ doF90Cpp: True if the F90cpp needs to be done (land)
+ cppdefs: a single string containing all CPPDEFs to add during compilations
+ paths: [array of paths to build]
+ otherFlags: Additional flags defined as environment variables in the experiment.yaml
+```
+
+platform yaml: User defined platform specifications. This will require more user input that bronx
+```yaml
+platforms:
+ - name: the platform name
+ compiler: the compiler you are using
+ modulesInit: ["array of commands that are needed to load modules." , "each command must end with a newline character"]
+ modules: [array of modules to load including compiler]
+ fc: the name of the fortran compiler
+ cc: the name of the C compiler
+ mkTemplate: The location of the mkmf make template
+ modelRoot: The root directory of the model (where src, exec, experiments will go)
+ container: True if this is a container platform
+ containerBuild: "podman" - the container build program
+ containerRun: "apptainer" - the container run program
+```
+
+## How to Get started
+Set up the environment:
+```bash
+# Load the fre-cli (if on gaea or somewhere where the fre-cli is available)
+module load fre/canopy
+
+# If the fre-cli is not available, follow steps in the fre-cli repository to either create your own conda environment with the fre-cli or activate the fre-cli environment available
+# If no fre-cli env available:
+conda create -n fremake-env # create environment
+conda config --append channels noaa-gfdl # append necessary channels
+conda config --append channels conda-forge # append necessary channels
+conda activate fremake-env # activate environment
+conda install noaa-gfdl::fre-cli # install the fre-cli
+```
+
+### How to build the Bare-metal example
+```bash
+# Have access to yamls
+# older versino: git clone -b 2023.00 https://gitlab.gfdl.noaa.gov/portable_climate/fremake_canopy.git
+
+git clone https://gitlab.gfdl.noaa.gov/portable_climate/fremake_canopy.git
+cd fremake_canopy/yamls
+
+# Create and run checkout
+# `-e` will run the checkout script after creating it
+# without `-e`, the checkout script will just be created
+fre make create-checkout -y am5.yaml -p ncrc5.intel -t prod -e
+
+# Create the Makefile
+fre make create-makefile -y am5.yaml -p ncrc5.intel -t prod
+
+# Create and run the compile script
+# `-e` will run the compile script after creating it
+# without `-e`, the compile script will just be created
+fre make create-compile -y am5.yaml -p ncrc5.intel -t prod -e
+```
+*Corresponding files, such as the checkout script, makefile, compile script, and experiment executable will be in the `/exec` folder created in `$HOME/$USER/fremake-canopy/am5/ncrc5.intel-prod/test`*
+
+### How to build the Container example
+```bash
+# Have access to yamls
+# older version: git clone -b 2023.00 https://gitlab.gfdl.noaa.gov/portable_climate/fremake_canopy.git
+
+git clone https://gitlab.gfdl.noaa.gov/portable_climate/fremake_canopy.git
+cd fremake_canopy/yamls
+
+# Create and run checkout
+# `-e` will run the checkout script after creating it
+# without `-e`, the checkout script will just be created
+# Be sure to specify `-npc` for non-parallel checkout
+fre make create-checkout -y am5.yaml -p hpcme.2023 -t prod -npc -e
+
+# Create the Makefile
+fre make create-makefile -y am5.yaml -p hpcme.2023 -t prod
+
+# Create and run the dockerfile
+# `-e` will run the dockerfile after creating it, to build a container
+# without `-e`, the dockerfile will just be created
+fre make create-dockerfile -y am5.yaml -p hpcme.2023 -t prod -e
+```
+*Corresponding files, such as the checkout script and makefile, will be in a tmp location and copied into the container via the dockerfile. The dockerfile,, as well as the container sif file, will be created in the users current location (yaml folder).*
+
+### How to run complete fremake
+```bash
+# Have access to yamls
+# older version: git clone -b 2023.00 https://gitlab.gfdl.noaa.gov/portable_climate/fremake_canopy.git
+
+git clone https://gitlab.gfdl.noaa.gov/portable_climate/fremake_canopy.git
+cd fremake_canopy/yamls
+
+# Run fremake
+# Can run fremake for bare-metal or container; be sure to give the correct platform for your build
+fre make run-fremake -y am5.yaml -p [ncrc5.intel OR hpcme.2023] -t prod
+```
diff --git a/lib/fre/make/gfdlfremake/buildBaremetal.py b/lib/fre/make/gfdlfremake/buildBaremetal.py
new file mode 100644
index 00000000..9e742980
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/buildBaremetal.py
@@ -0,0 +1,153 @@
+#!/usr/bin/python3
+## \date 2023
+## \author Tom Robinson
+## \email thomas.robinson@noaa.gov
+## \description
+
+import subprocess
+import os
+
+def fremake_parallel(fremakeBuildList):
+ """
+ Brief: Called for parallel execution purposes. Runs the builds.
+ Param:
+ - fremakeBuildList : fremakeBuild object list passes by pool.map
+ """
+ fremakeBuildList.run()
+
+class buildBaremetal():
+ """
+ Brief: Creates the build script to compile the model
+ Param:
+ - self : The buildScript object
+ - exp : The experiment name
+ - mkTemplatePath : The template used by mkmf to compile the model
+ - srcDir : The source directory
+ - bldDir : The build directory
+ - modules : The list of modules to load before compilation
+ - modulesInit : A list of commands with new line characters to initialize modules
+ """
+ def __init__(self,exp,mkTemplatePath,srcDir,bldDir,target,modules,modulesInit,jobs):
+ """
+ Initialize variables and set-up the compile script.
+ """
+ self.e = exp
+ self.t = target.gettargetName()
+ self.src = srcDir
+ self.bld = bldDir
+ self.make = "make --jobs="+str(jobs)+" "+target.getmakeline_add() #make line
+ self.mkmf = True
+ self.template = mkTemplatePath
+ self.modules = ""
+ for m in modules:
+ self.modules = f"{self.modules} {m}"
+
+ ## Set up the top portion of the compile script
+ self.setup=[ "#!/bin/sh -fx \n",
+ f"bld_dir={self.bld}/ \n",
+ f"src_dir={self.src}/ \n",
+ f"mkmf_template={self.template} \n"]
+ if self.modules != "":
+ self.setup.extend(modulesInit) #extend - this is a list
+ self.setup.append(f"module load {self.modules} \n") # Append -this is a single string
+
+ ## Create the build directory
+ os.system(f"mkdir -p {self.bld}")
+
+ ## Create the compile script
+ self.f=open(f"{self.bld}/compile.sh","w")
+ self.f.writelines(self.setup)
+
+ def writeBuildComponents(self, c):
+ """
+ Brief: Adds components to the build script
+ Param:
+ - self : The build script object
+ - c : Component from the compile yaml
+ """
+ # Shorthand for component
+ comp = c["component"]
+
+ # Make the component directory
+ self.f.write(f"\n mkdir -p $bld_dir/{comp}\n")
+
+ # Get the paths needed for compiling
+ pstring = ""
+ for paths in c["paths"]:
+ pstring = pstring+"$src_dir/"+paths+" "
+
+ # Run list_paths
+ self.f.write(f" list_paths -l -o $bld_dir/{comp}/pathnames_{comp} {pstring}\n")
+ self.f.write(f" cd $bld_dir/{comp}\n")
+
+ # Create the mkmf line
+ # If this lib doesnt have any code dependencies and
+ # it requires the preprocessor (no -o and yes --use-cpp)
+ if c["requires"] == [] and c["doF90Cpp"]:
+ self.f.write(" mkmf -m Makefile -a $src_dir -b $bld_dir "
+ "-p lib"+comp+".a -t $mkmf_template --use-cpp "
+ "-c \""+c["cppdefs"]+"\" "+c["otherFlags"]
+ +" $bld_dir/"+comp+"/pathnames_"+comp+" \n")
+ elif c["requires"] == []: # If this lib doesnt have any code dependencies (no -o)
+ self.f.write(" mkmf -m Makefile -a $src_dir -b $bld_dir "
+ "-p lib"+comp+".a -t $mkmf_template -c \""
+ +c["cppdefs"]+"\" "+c["otherFlags"]
+ +" $bld_dir/"+comp+"/pathnames_"+comp+" \n")
+ else: #Has requirements
+ #Set up the requirements as a string to inclue after the -o
+ reqstring = ""
+ for r in c["requires"]:
+ reqstring = reqstring+"-I$bld_dir/"+r+" "
+
+ #Figure out if we need the preprocessor
+ if c["doF90Cpp"]:
+ self.f.write(" mkmf -m Makefile -a $src_dir -b $bld_dir "
+ "-p lib"+comp+".a -t $mkmf_template --use-cpp "
+ "-c \""+c["cppdefs"]+"\" -o \""+reqstring+"\" "
+ +c["otherFlags"]+" $bld_dir/"+comp+"/pathnames_"+comp+" \n")
+ else:
+ self.f.write(" mkmf -m Makefile -a $src_dir -b $bld_dir "
+ "-p lib"+comp+".a -t $mkmf_template -c \""
+ +c["cppdefs"]+"\" -o \""+reqstring+"\" "+c["otherFlags"]
+ +" $bld_dir/"+comp+"/pathnames_"+comp+" \n")
+
+##TODO: add targets input
+ def writeScript(self):
+ """
+ Brief: Finishes and writes the build script
+ Param:
+ - self : The buildScript object
+ """
+ self.f.write(f"cd {self.bld}\n")
+ self.f.write(f"{self.make}\n")
+ self.f.close()
+
+ # Make compile script executable
+ os.chmod(self.bld+"/compile.sh", 0o744)
+
+## TODO run as a batch job on the login cluster
+ def run(self):
+ """
+ Brief: Run the build script
+ Param:
+ - self : The dockerfile object
+ """
+ command = [self.bld+"/compile.sh"]
+
+ # Run compile script
+ p1 = subprocess.Popen(command, stdout=subprocess.PIPE,stderr=subprocess.STDOUT)
+
+ # Direct output to log file as well
+ p2 = subprocess.Popen(["tee",self.bld+"/log.compile"], stdin=p1.stdout)
+
+ # Allow process1 to receive SIGPIPE is process2 exits
+ p1.stdout.close()
+ p2.communicate()
+
+ # wait for process1 to finish before checking return code
+ p1.wait()
+ if p1.returncode != 0:
+ print(f"\nThere was an error running {self.bld}/compile.sh")
+ print(f"Check the log file: {self.bld}/log.compile")
+ else:
+ print(f"\nSuccessful run of {self.bld}/compile.sh")
diff --git a/lib/fre/make/gfdlfremake/buildDocker.py b/lib/fre/make/gfdlfremake/buildDocker.py
new file mode 100644
index 00000000..491d5c2c
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/buildDocker.py
@@ -0,0 +1,208 @@
+#!/usr/bin/python3
+## \date 2023
+## \author Tom Robinson
+## \email thomas.robinson@noaa.gov
+## \description
+
+import os
+
+class container():
+ """
+ Brief: Opens the Dockerfile for writing
+ Param:
+ - self : The dockerfile object
+ - base : The docker base image to start from
+ - libs : Additional libraries defined by user
+ - exp : The experiment name
+ - RUNenv : The commands that have to be run at
+ the beginning of a RUN in the dockerfile
+ to set up the environment
+ """
+ def __init__(self,base,exp,libs,RUNenv,target):
+ """
+ Initialize variables and write to the dockerfile
+ """
+ self.base = base
+ self.e = exp
+ self.l = libs
+ self.src = "/apps/"+self.e+"/src"
+ self.bld = "/apps/"+self.e+"/exec"
+ self.mkmf = True
+ self.target = target
+ self.template = "/apps/mkmf/templates/hpcme-intel21.mk"
+
+ # Set up spack loads in RUN commands in dockerfile
+ if RUNenv == "":
+ self.setup = ["RUN \\ \n"]
+ else:
+ self.setup = ["RUN "+RUNenv[0]+" \\ \n"]
+ self.setup
+ for env in RUNenv[1:]:
+ self.setup.append(" && "+env+" \\ \n")
+ if self.l:
+ for l in self.l:
+ self.setup.append(" && spack load "+l+" \\ \n")
+
+ # Clone and copy mkmf through Dockerfile
+ self.mkmfclone=["RUN cd /apps \\ \n",
+ " && git clone --recursive https://github.com/NOAA-GFDL/mkmf \\ \n",
+ " && cp mkmf/bin/* /usr/local/bin \n"]
+
+ # Set bld_dir, src_dir, mkmf_template
+ self.bldsetup=["RUN bld_dir="+self.bld+" \\ \n",
+ " && src_dir="+self.src+" \\ \n",
+ " && mkmf_template="+self.template+ " \\ \n"]
+ self.d=open("Dockerfile","w")
+ self.d.writelines("FROM "+self.base+" \n")
+
+ def writeDockerfileCheckout(self, cScriptName, cOnDisk):
+ """
+ Brief: writes to the checkout part of the Dockerfile and sets up the compile
+ Param:
+ - self : The dockerfile object
+ - cScriptName : The name of the checkout script in the container
+ - cOnDisk : The relative path to the checkout script on disk
+ """
+ self.checkoutPath = self.src+"/"+ cScriptName
+ self.d.write("COPY " + cOnDisk +" "+ self.checkoutPath +" \n")
+ self.d.write("RUN chmod 744 "+self.src+"/checkout.sh \n")
+ self.d.writelines(self.setup)
+ self.d.write(" && "+self.src+"/checkout.sh \n")
+ # Clone mkmf
+ self.d.writelines(self.mkmfclone)
+
+ def writeDockerfileMakefile(self, makefileOnDiskPath, linklineonDiskPath):
+ """
+ Brief: Copies the Makefile into the bldDir in the dockerfile
+ Param:
+ - self : The dockerfile object
+ - makefileOnDiskPath : The path to Makefile on the local disk
+ - linklineonDiskPath : The path to the link line script on the local disk
+ """
+ # Set up the bldDir
+ # If no additional libraries defined
+ if self.l == None:
+ self.bldCreate=["RUN mkdir -p "+self.bld+" \n",
+ "COPY "+ makefileOnDiskPath +" "+self.bld+"/Makefile \n"]
+ self.d.writelines(self.bldCreate)
+ # If additional libraries defined
+ if self.l != None:
+ self.bldCreate=["RUN mkdir -p "+self.bld+" \n",
+ "COPY "+ makefileOnDiskPath +" "+self.bld+"/Makefile \n",
+ "RUN chmod +rw "+self.bld+"/Makefile \n",
+ "COPY "+ linklineonDiskPath +" "+self.bld+"/linkline.sh \n",
+ "RUN chmod 744 "+self.bld+"/linkline.sh \n"]
+ self.d.writelines(self.bldCreate)
+ self.d.writelines(self.setup)
+ self.d.write(" && "+self.bld+"/linkline.sh \n")
+
+ def writeDockerfileMkmf(self, c):
+ """
+ Brief: Adds components to the build part of the Dockerfile
+ Param:
+ - self : The dockerfile object
+ - c : Component from the compile yaml
+ """
+ # Set up the compile variables
+ self.d.writelines(self.bldsetup)
+
+ # Shorthand for component
+ comp = c["component"]
+
+ # Make the component directory
+ self.d.write(" && mkdir -p $bld_dir/"+comp+" \\ \n")
+
+ # Get the paths needed for compiling
+ pstring = ""
+ for paths in c["paths"]:
+ pstring = pstring+"$src_dir/"+paths+" "
+
+ # Run list_paths
+ self.d.write(" && list_paths -l -o $bld_dir/"+comp+"/pathnames_"+comp+" "+pstring+" \\ \n")
+ self.d.write(" && cd $bld_dir/"+comp+" \\ \n")
+
+ # Create the mkmf line
+ if c["requires"] == [] and c["doF90Cpp"]: # If this lib doesnt have any code dependencies and it requires the preprocessor (no -o and yes --use-cpp)
+ self.d.write(" && mkmf -m Makefile -a $src_dir -b $bld_dir -p lib"+comp+".a -t $mkmf_template --use-cpp -c \""+c["cppdefs"]+"\" "+c["otherFlags"]+" $bld_dir/"+comp+"/pathnames_"+comp+" \n")
+ elif c["requires"] == []: # If this lib doesnt have any code dependencies (no -o)
+ self.d.write(" && mkmf -m Makefile -a $src_dir -b $bld_dir -p lib"+comp+".a -t $mkmf_template -c \""+c["cppdefs"]+"\" "+c["otherFlags"]+" $bld_dir/"+comp+"/pathnames_"+comp+" \n")
+ else: #Has requirements
+ #Set up the requirements as a string to inclue after the -o
+ reqstring = ""
+ for r in c["requires"]:
+ reqstring = reqstring+"-I$bld_dir/"+r+" "
+
+ #Figure out if we need the preprocessor
+ if c["doF90Cpp"]:
+ self.d.write(" && mkmf -m Makefile -a $src_dir -b $bld_dir -p lib"+comp+".a -t $mkmf_template --use-cpp -c \""+c["cppdefs"]+"\" -o \""+reqstring+"\" "+c["otherFlags"]+" $bld_dir/"+comp+"/pathnames_"+comp+" \n")
+ else:
+ self.d.write(" && mkmf -m Makefile -a $src_dir -b $bld_dir -p lib"+comp+".a -t $mkmf_template -c \""+c["cppdefs"]+"\" -o \""+reqstring+"\" "+c["otherFlags"]+" $bld_dir/"+comp+"/pathnames_"+comp+" \n")
+
+ def writeRunscript(self,RUNenv,containerRun,runOnDisk):
+ """
+ Brief: Writes a runscript to set up spack loads/environment
+ in order to run the executable in the container;
+ runscript copied into container
+ Param:
+ - self : The dockerfile object
+ - RUNEnv : The commands that have to be run at
+ the beginning of a RUN in the dockerfile
+ - containerRun : The container platform used with `exec`
+ to run the container; apptainer
+ or singularity used
+ - runOnDisk : The path to the run script on the local disk
+ """
+ #create runscript in tmp - create spack environment, install necessary packages,
+ self.createscript = ["#!/bin/bash \n",
+ "export BACKUP_LD_LIBRARY_PATH=$LD_LIBRARY_PATH\n",
+ "# Set up spack loads\n",
+ RUNenv[0]+"\n"]
+ with open(runOnDisk,"w") as f:
+ f.writelines(self.createscript)
+ f.write("# Load spack packages\n")
+ for env in RUNenv[1:]:
+ f.write(env+"\n")
+
+ if self.l:
+ for l in self.l:
+ self.spackloads = "spack load "+l+"\n"
+ f.write(self.spackloads)
+
+ f.write("export LD_LIBRARY_PATH=$BACKUP_LD_LIBRARY_PATH:$LD_LIBRARY_PATH\n")
+ f.write("# Run executable\n")
+ f.write(self.bld+"/"+self.e+".x\n")
+ #copy runscript into container in dockerfile
+ self.d.write("COPY "+runOnDisk+" "+self.bld+"/execrunscript.sh\n")
+ #make runscript executable
+ self.d.write("RUN chmod 744 "+self.bld+"/execrunscript.sh\n")
+ #link runscript to more general location (for frerun container usage)
+ self.d.write("RUN mkdir -p /apps/bin \ \n")
+ self.d.write(" && ln -sf "+self.bld+"/execrunscript.sh "+"/apps/bin/execrunscript.sh \n")
+ #finish the dockerfile
+ self.d.writelines(self.setup)
+ self.d.write(" && cd "+self.bld+" && make -j 4 "+self.target.getmakeline_add()+"\n")
+ self.d.write('ENTRYPOINT ["/bin/bash"]')
+ self.d.close()
+
+ def createBuildScript(self,containerBuild,containerRun):
+ """
+ Brief: Writes out the build commands for the created dockerfile in a script,
+ which builds the dockerfile and then converts the format to a singularity image file.
+ Param:
+ - self : The dockerfile object
+ - containerBuild : The tool used to build the container;
+ docker or podman used
+ - containerRun : The container platform used with `exec` to
+ run the container; apptainer or singularity used
+ """
+ self.userScript = ["#!/bin/bash\n"]
+ self.userScript.append(containerBuild+" build -f Dockerfile -t "+self.e+":"+self.target.gettargetName()+"\n")
+ self.userScript.append("rm -f "+self.e+".tar "+self.e+".sif\n")
+ self.userScript.append(containerBuild+" save -o "+self.e+"-"+self.target.gettargetName()+".tar localhost/"+self.e+":"+self.target.gettargetName()+"\n")
+ self.userScript.append(containerRun+" build --disable-cache "+self.e+"-"+self.target.gettargetName()+".sif docker-archive://"+self.e+"-"+self.target.gettargetName()+".tar\n")
+ self.userScriptFile = open("createContainer.sh","w")
+ self.userScriptFile.writelines(self.userScript)
+ self.userScriptFile.close()
+ os.chmod("createContainer.sh", 0o744)
+ self.userScriptPath = os.getcwd()+"/createContainer.sh"
+
diff --git a/lib/fre/make/gfdlfremake/checkout.py b/lib/fre/make/gfdlfremake/checkout.py
new file mode 100644
index 00000000..d3fa5d4e
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/checkout.py
@@ -0,0 +1,154 @@
+import os
+import subprocess
+
+## TODO: Add parallelizations using () and simplify
+def writeRepo(file,repo,component,srcDir,branch,add,multi,jobs,pc):
+ """
+ Brief: Creates the clone lines for the checkout script
+ Param:
+ - file Checkout script file
+ - repo the repo(s) to clone
+ - component Model component name
+ - srcDir The source directory
+ - branch The version to clone/checkout
+ - add Additional instrcutions after the clone
+ - multi True if a component has more than one repo to clone
+ """
+ ## Write message about cloning repo and branch in component
+ file.write("echo cloning "+repo+" -b "+branch+" into "+srcDir+"/"+component+"\n")
+
+ ## If this component has multiple repos, clone everything in the component folder
+ ## If it's not multi, then use the component name (comp) as the folder name to clone into
+ if multi:
+ file.write("mkdir -p "+component+"\n")
+ file.write("cd "+component+"\n")
+ comp=""
+ else:
+ comp=component
+
+ ## Check if there is a branch/version and then write the clone line;
+ ## record the pid of that clone in dictionary `pids` if parallel
+ ## checkout option is defined
+ if pc:
+ if branch=="":
+ file.write("(git clone --recursive --jobs="+jobs+" "+repo+" "+comp+")"+pc+"\n")
+ if multi:
+ r=repo.split("/")[4].strip(".git")
+ file.write("pids+=("+r+"pid:$!)\n")
+ else:
+ file.write("pids+=("+comp+"pid:$!)\n")
+ else:
+ file.write("(git clone --recursive --jobs="+jobs+" "+repo+" -b "+branch+" "+comp+")"+pc+"\n")
+ if multi:
+ r=repo.split("/")[4].strip(".git")
+ file.write("pids+=("+r+"pid:$!)\n")
+ else:
+ file.write("pids+=("+comp+"pid:$!)\n")
+ else:
+ if branch=="":
+ file.write("git clone --recursive --jobs="+jobs+" "+repo+" "+comp+"\n")
+ else:
+ file.write("git clone --recursive --jobs="+jobs+" "+repo+" -b "+branch+" "+comp+"\n")
+
+ ## Make sure to go back up in the folder structure
+ if multi:
+ file.write("cd .. \n")
+ if add!="":
+ file.write(add)
+
+class checkout():
+ """
+ Brief: Class to create the checkout script
+ """
+ def __init__(self,fname,srcDir):
+ """
+ Brief: Opens the checkout script with the specified name
+ Param:
+ - self The checkout script object
+ - fname The file name of the checkout script
+ - srcDir The source directory where fname will be run and source will exist
+ """
+ self.fname = fname
+ self.src = srcDir
+ os.system("mkdir -p "+self.src)
+ ##TODO: Force checkout
+ os.system("rm -rf "+self.src+"/*")
+ self.checkoutScript = open(self.src+"/"+fname, 'w')
+ self.checkoutScript.write("#!/bin/sh -f \n")
+ self.checkoutScript.write("export GIT_TERMINAL_PROMPT=0 \n")
+ def writeCheckout(self,y,jobs,pc):
+ """
+ Brief: Writes the contents of the checkout script by looping through the input yaml
+ Param:
+ - self The checkout script object
+ - y The fremake compile yaml
+ """
+ self.checkoutScript.write("cd "+self.src +"\n")
+ for c in y['src']:
+ if type(c['repo']) is list and type(c['branch']) is list:
+ for (repo,branch) in zip(c['repo'],c['branch']):
+ writeRepo(self.checkoutScript,repo,c['component'],self.src,branch,c['additionalInstructions'],True,jobs,pc)
+ else:
+ writeRepo(self.checkoutScript,c['repo'],c['component'],self.src,c['branch'],c['additionalInstructions'],False,jobs,pc)
+
+ def finish (self,pc):
+ """
+ Brief: If pc is defined: Loops through dictionary of pids,
+ waits for each pid individually, writes exit code in
+ `check` list;
+ allows checkoutscript to exit if exit code is not 0;
+ closes the checkout script when writing is done
+ Param:
+ - self The checkout script object
+ """
+ if pc:
+ self.checkoutScript.write('for id in ${pids[@]}; do\n wait ${id##*:}\n check+=("clone of ${id%%:*} exited with status $?")\ndone\n')
+ self.checkoutScript.write('for stat in "${check[@]}"; do\n echo $stat \n if [ ${stat##* } -ne 0 ]; then\n exit ${stat##* }\n fi\ndone')
+ self.checkoutScript.close()
+ else:
+ self.checkoutScript.close()
+
+## TODO: batch script building
+ def run (self):
+ """
+ Brief: Runs the checkout script
+ Param:
+ - self The checkout script object
+ """
+ try:
+ subprocess.run(args=[self.src+"/"+self.fname], check=True)
+ except:
+ print("There was an error with the checkout script "+self.src+"/"+self.fname)
+ raise
+###################################################################################################
+## Subclass for container checkout
+class checkoutForContainer(checkout):
+ """
+ Brief: Subclass for container checkout
+ """
+ def __init__(self,fname,srcDir,tmpdir):
+ """
+ Brief: Opens the checkout script with the specified name
+ Param:
+ - self : The checkout script object
+ - fname : The file name of the checkout script
+ - srcDir : The source directory where fname will be run and source will exist
+ - tmpdir : The relative path on disk that fname will be created (and copied from into the
+ container)
+ """
+ self.fname = fname
+ self.src = srcDir
+ self.tmpdir = tmpdir
+ os.system("mkdir -p "+self.tmpdir)
+ os.system("rm -rf "+self.tmpdir+"/*")
+ self.checkoutScript = open(self.tmpdir+"/"+fname, 'w')
+ self.checkoutScript.write("#!/bin/sh -fx \n")
+ self.checkoutScript.write("export GIT_TERMINAL_PROMPT=0 \n")
+
+ def cleanup (self):
+ """
+ Brief: Removes the self.tmpdir and contents
+ Param:
+ - self The checkout script object
+ """
+ os.system("rm -rf "+self.tmpdir)
diff --git a/lib/fre/make/gfdlfremake/makefilefre.py b/lib/fre/make/gfdlfremake/makefilefre.py
new file mode 100644
index 00000000..dfe15fff
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/makefilefre.py
@@ -0,0 +1,204 @@
+import os
+import textwrap
+
+def linklineBuild(self):
+ """
+ Brief: Writes the link line for bare metal and container builds
+ Param:
+ - self The Makefile object
+ """
+ linkline=""
+
+#if additional libraries are defined, populate the link line with the correct information for libraries
+## CONTAINER; write a script that will execute in the container, to fill in link line with additional libraries in Makefile
+ if "tmp" in self.filePath:
+ with open(self.filePath+"/linkline.sh","a") as fh:
+ fh.write("set -- ")
+ for l in self.l:
+ fh.write(l+" ")
+ fh.write("\n")
+
+ self.linklinecreate = '''
+ line=''
+ for l in $@; do
+ loc=$(spack location -i $l)
+ libraries=$(ls $loc/lib)
+ if echo "$libraries" | grep -q "_d"; then
+ for i in $libraries; do
+ if [ "$i" != "cmake" ] && echo "$i" | grep -q "_d"; then
+ ln1=${i%.*}
+ ln2=${ln1#???}
+ line=$line" -L$loc/lib -l$ln2"
+ fi
+ done
+ else
+ for i in $libraries; do
+ if [ "$i" != "cmake" ]; then
+ ln1=${i%.*}
+ ln2=${ln1#???}
+ line=$line" -L$loc/lib -l$ln2"
+ fi
+ done
+ fi
+ done
+ '''
+
+ with open(self.filePath+"/linkline.sh","a") as fh:
+ fh.writelines(textwrap.dedent(self.linklinecreate))
+ fh.write("MF_PATH='/apps/"+self.e+"/exec/Makefile'\n")
+ fh.write('sed -i "/MK_TEMPLATE = /a LL = $line" $MF_PATH\n')
+ fh.write("sed -i 's|\($^\) \($(LDFLAGS)\)|\\1 $(LL) \\2|' $MF_PATH\n")
+
+## BARE METAL; if addlibs defined on bare metal, include those additional libraries in link line
+ elif "tmp" not in self.filePath:
+ for l in self.l: # baremetal_linkerflags
+ linkline = linkline + " " + l
+ os.system(f"sed -i '/MK_TEMPLATE = /a LL = {linkline}' {self.filePath}/Makefile")
+ os.system(f"sed -i 's|\($(LDFLAGS)\)|$(LL) \\1|' {self.filePath}/Makefile")
+
+class makefile():
+ def __init__(self,exp,libs,srcDir,bldDir,mkTemplatePath):
+ """
+ Brief: Opens Makefile and sets the experiment and other common variables
+ Param:
+ - self The Makefile object
+ - exp Experiment name
+ - libs Additional libraries/linker flags defined by user
+ - srcDir The path to the source directory
+ - bldDir The path to the build directory
+ - mkTemplatePath The path of the template .mk file for compiling
+ """
+ self.e = exp
+ self.l = libs
+ self.src = srcDir
+ self.bld = bldDir
+ self.template = mkTemplatePath
+ self.c =[] #components
+ self.r=[] #requires
+ self.o=[] #overrides
+ os.system("mkdir -p "+self.bld)
+ self.filePath = self.bld # Needed so that the container and bare metal builds can
+ # use the same function to create the Makefile
+
+ def addComponent (self,c,r,o):
+ """
+ Brief: Adds a component and corresponding requires to the list
+ Param:
+ - self The Makefile object
+ - c The component
+ - r The requires for that componenet
+ - o The overrides for that component
+ """
+ self.c.append(c)
+ self.r.append(r)
+ self.o.append(o)
+
+ def createLibstring (self,c,r,o):
+ """
+ Brief: Sorts the component by how many requires there are for that component
+ Param:
+ - self The Makefile object
+ - c The component
+ - r The requires for that component
+ - o The overrides for that component
+ """
+ # org_comp : returns a zip object
+ org_comp = zip(self.c,self.r,self.o)
+ # Sort zip object so that the component with the most requires (self.r) is listed first, and so on
+ sort = sorted(org_comp,key=lambda values:len(values[1]),reverse=True)
+
+ return sort
+
+ def writeMakefile (self):
+ """
+ Brief: Writes the Makefile. Should be called after all components are added
+ Param:
+ - self The Makefile object
+ """
+ # Get the list of all of the libraries
+ sd=self.createLibstring(self.c,self.r,self.o)
+ libstring=" "
+ for i in sd:
+ lib=i[0]
+ libstring = libstring+lib+"/lib"+lib+".a "
+
+ # Open the Makefile for Writing
+ with open(self.filePath+"/Makefile","w") as fh:
+ # Write the header information for the Makefile
+ fh.write("# Makefile for "+self.e+"\n")
+ fh.write("SRCROOT = "+self.src+"/\n")
+ fh.write("BUILDROOT = "+self.bld+"/\n")
+ fh.write("MK_TEMPLATE = "+self.template+"\n")
+ fh.write("include $(MK_TEMPLATE)"+"\n")
+
+ # Write the main experiment compile
+ fh.write(self.e+".x: "+libstring+"\n")
+ fh.write("\t$(LD) $^ $(LDFLAGS) -o $@ $(STATIC_LIBS)"+"\n")
+
+ # Write the link line script with user-provided libraries
+ if self.l:
+ linklineBuild(self)
+
+ # Write the individual component library compiles
+ with open(self.filePath+"/Makefile","a") as fh:
+ for (c,r,o) in sd:
+ libstring = " "
+ for lib in r:
+ libstring = libstring+lib+"/lib"+lib+".a "
+ cstring = c+"/lib"+c+".a: "
+ fh.write(cstring+libstring+" FORCE"+"\n")
+ if o == "":
+ fh.write("\t$(MAKE) SRCROOT=$(SRCROOT) BUILDROOT=$(BUILDROOT) MK_TEMPLATE=$(MK_TEMPLATE) --directory="+c+" $(@F)\n")
+ else:
+ fh.write("\t$(MAKE) SRCROOT=$(SRCROOT) BUILDROOT=$(BUILDROOT) MK_TEMPLATE=$(MK_TEMPLATE) "+o+" --directory="+c+" $(@F)\n")
+ fh.write("FORCE:\n")
+ fh.write("\n")
+
+ # Set up the clean
+ fh.write("clean:\n")
+ for c in self.c:
+ fh.write("\t$(MAKE) --directory="+c+" clean\n")
+
+ # Set up localize
+ fh.write("localize:\n")
+ for c in self.c:
+ fh.write("\t$(MAKE) -f $(BUILDROOT)"+c+" localize\n")
+
+ # Set up distclean
+ fh.write("distclean:\n")
+ for c in self.c:
+ fh.write("\t$(RM) -r "+c+"\n")
+ fh.write("\t$(RM) -r "+self.e+"\n")
+ fh.write("\t$(RM) -r Makefile \n")
+
+### This seems incomplete? ~ ejs
+## The makefile class for a container. It gets built into a temporary directory so it can be copied
+## into the container.
+## \param exp Experiment name
+## \param libs Additional libraries/linker flags defined by user
+## \param srcDir The path to the source directory
+## \param bldDir The path to the build directory
+## \param mkTemplatePath The path of the template .mk file for compiling
+## \param tmpDir A local path to temporarily store files build to be copied to the container
+class makefileContainer(makefile):
+ def __init__(self,exp,libs,srcDir,bldDir,mkTemplatePath,tmpDir):
+ self.e = exp
+ self.l = libs
+ self.src = srcDir
+ self.bld = bldDir
+ self.template = mkTemplatePath
+ self.tmpDir = tmpDir
+ self.c =[] #components
+ self.r=[] #requires
+ self.o=[] #overrides
+ os.system("mkdir -p "+self.tmpDir)
+ self.filePath = self.tmpDir # Needed so that the container and bare metal builds can
+ # use the same function to create the Makefile
+
+ def getTmpDir(self):
+ """
+ Brief: Return the tmpDir
+ Param:
+ - self The makefile object
+ """
+ return self.tmpDir
diff --git a/lib/fre/make/gfdlfremake/platformfre.py b/lib/fre/make/gfdlfremake/platformfre.py
new file mode 100644
index 00000000..4f8d0eed
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/platformfre.py
@@ -0,0 +1,108 @@
+import yaml
+
+class platforms ():
+ def __init__(self,platforminfo):
+ """
+ Param:
+ - self The platform yaml object
+ - platforminfo dictionary with platform information
+ from the combined yaml
+ """
+ self.yaml = platforminfo
+
+ ## Check the yaml for errors/omissions
+ ## Loop through the platforms
+ for p in self.yaml:
+ ## Check the platform name
+ try:
+ p["name"]
+ except:
+ raise Exception("At least one of the platforms is missing a name in "+fname+"\n")
+ ## Check the compiler
+ try:
+ p["compiler"]
+ except:
+ raise Exception("You must specify a compiler in your "+p["name"]+" platform in the file "+fname+"\n")
+ ## Check for the Fortran (fc) and C (cc) compilers
+ try:
+ p["fc"]
+ except:
+ raise Exception("You must specify the name of the Fortran compiler as fc on the "+p["name"]+" platform in the file "+fname+"\n")
+ try:
+ p["cc"]
+ except:
+ raise Exception("You must specify the name of the Fortran compiler as cc on the "+p["name"]+" platform in the file "+fname+"\n")
+ ## Check for modules to load
+ try:
+ p["modules"]
+ except:
+ p["modules"]=[""]
+ ## Check for modulesInit to set up the modules environment
+ try:
+ p["modulesInit"]
+ except:
+ p["modulesInit"]=[""]
+ ## Get the root for the build
+ try:
+ p["modelRoot"]
+ except:
+ p["modelRoot"] = "/apps"
+ ## Check if we are working with a container and get the info for that
+ try:
+ p["container"]
+ except:
+ p["container"] = False
+ p["RUNenv"] = [""]
+ p["containerBuild"] = ""
+ p["containerRun"] = ""
+ if p["container"]:
+ ## Check the container builder
+ try:
+ p["containerBuild"]
+ except:
+ raise Exception("You must specify the program used to build the container (containerBuild) on the "+p["name"]+" platform in the file "+fname+"\n")
+ if p["containerBuild"] != "podman" and p["containerBuild"] != "docker":
+ raise ValueError("Container builds only supported with docker or podman, but you listed "+p["containerBuild"]+"\n")
+ ## Check for container environment set up for RUN commands
+ try:
+ p["RUNenv"]
+ except:
+ p["RUNenv"] = ""
+ ## Check the container runner
+ try:
+ p["containerRun"]
+ except:
+ raise Exception("You must specify the program used to run the container (containerRun) on the "+p["name"]+" platform in the file "+fname+"\n")
+ if p["containerRun"] != "apptainer" and p["containerRun"] != "singularity":
+ raise ValueError("Container builds only supported with apptainer, but you listed "+p["containerRun"]+"\n")
+ ## set the location of the mkTemplate.
+ ## In a container, it uses the hpc-me template cloned from mkmf
+ p["mkTemplate"] = "/apps/mkmf/templates/hpcme-intel21.mk"
+ else:
+ try:
+ p["mkTemplate"]
+ except:
+ raise ValueError("The non-container platform "+p["name"]+" must specify a mkTemplate \n")
+
+ def hasPlatform(self,name):
+ """
+ Brief: Checks if the platform yaml has the named platform
+ """
+ for p in self.yaml:
+ if p["name"] == name:
+ return True
+ return False
+
+ def getPlatformsYaml(self):
+ """
+ Brief: Get the platform yaml
+ """
+ return self.yaml
+
+ def getPlatformFromName(self,name):
+ """
+ Brief: Get the platform information from the name of the platform
+ """
+ for p in self.yaml:
+ if p["name"] == name:
+ return (p["compiler"], p["modules"], p["modulesInit"], p["fc"], p["cc"], p["modelRoot"],p["container"], p["mkTemplate"],p["containerBuild"], p["containerRun"], p["RUNenv"])
diff --git a/lib/fre/make/gfdlfremake/targetfre.py b/lib/fre/make/gfdlfremake/targetfre.py
new file mode 100644
index 00000000..0b2b5964
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/targetfre.py
@@ -0,0 +1,81 @@
+class fretarget:
+ """
+ Class: Stores information about the target
+ """
+ def __init__(self,t):
+ """
+ Brief: Sets up information about the target and handles errors
+ Note: The default target is prod
+ Param:
+ - self the fretarget object
+ - t The target string
+ """
+ self.target = t # The target string
+
+ ## Split the target string
+ targ = self.target.split('-')
+ self.makeline_add = ""
+ self.debug = False
+ self.repro = False
+ self.prod = False
+
+ ## Parse the target string for prod, repro, and debug. Set up what to add to the
+ ## make line during compile when using mkmf builds
+ for target in targ:
+ if target == "debug":
+ targ = target.upper()
+ self.makeline_add = self.makeline_add + targ + "=on "
+ self.debug = True
+ elif target == "prod":
+ targ = target.upper()
+ self.makeline_add = self.makeline_add + targ + "=on "
+ self.prod = True
+ elif target == "repro":
+ targ = target.upper()
+ self.makeline_add = self.makeline_add + targ + "=on "
+ self.repro = True
+
+ ## Check to see if openmp is included in the target and add that to the makeline add string
+ if target == "openmp":
+ targ = target.upper()
+ self.makeline_add = self.makeline_add + targ + "=on "
+ self.openmp = True
+ else:
+ self.openmp = False
+
+ ## Check to make sure only one of the prod, debug, repro are used
+ errormsg = "You can only list one mutually exclusive target, but your target '"+self.target+"' lists more than one of the following targets: \n debug \n prod \n repro"
+ if self.debug:
+ try:
+ if self.repro or self.prod == True:
+ raise ValueError(errormsg)
+ except ValueError:
+ raise
+ elif self.repro:
+ try:
+ if self.prod == True:
+ raise ValueError(errormsg)
+ except ValueError:
+ raise
+ else:
+ try:
+ if self.prod == False:
+ raise ValueError("Your target '"+self.target+"' needs to include one of the following: prod, repro, debug")
+ except ValueError:
+ raise
+
+ def gettargetName(self):
+ """
+ Brief: Returns the name of the target
+ Param:
+ - self The fretarget object
+ """
+ return self.target
+
+ def getmakeline_add(self):
+ """
+ Brief: Returns the makeline_add
+ Param:
+ - self The fretarget object
+ """
+ return self.makeline_add
diff --git a/lib/fre/make/gfdlfremake/varsfre.py b/lib/fre/make/gfdlfremake/varsfre.py
new file mode 100644
index 00000000..c153e845
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/varsfre.py
@@ -0,0 +1,56 @@
+import yaml
+import os
+import re
+
+## Removes enclosing braces or parentheses and optionally leading dollar sign from a string
+## \param inString A string from which characters will be removed
+## \param bound A set of characters to be removed from the beginning and end of the string
+## \param leadingDollar (optional) If True, removed leading dollar sign
+def removeEnclosing(inString, bound='()', leadingDollar=True):
+ if leadingDollar:
+ inString = inString.lstrip('$')
+ return inString.lstrip(bound[0]).rstrip(bound[1])
+
+## Retrieves an environmental variable based on a string of the form $(VAR)
+## \param inString A string that specifies the variable to be retrieved
+def getEnvSub(inString):
+ return os.getenv(removeEnclosing(inString.group(), bound='{}'))
+
+## Replaces all instances of Linux environment variables (${VAR}) in a string using getEnvSub
+## \param string A string to have the variables replaced
+def envReplace(string):
+ return re.sub("\$\{\w+\}", getEnvSub, string)
+
+## Reads stores and replaces the fre variables set in the model YAML
+class frevars():
+## Grabs the FRE variables from the model yaml
+## \param self The frevars object
+## \param y The model yaml file name
+ def __init__(self,y):
+ with open(y, 'r') as file:
+ self.modelyaml = yaml.safe_load(file)
+## Substitutes fre variables with the format $(variable) into the input string
+## \string The string that contains the variables
+## \returns String with the fre variables filled in
+ def freVarReplace(self,string):
+ ## Retrieves a value from the modelyaml based on a key of the form $(VAR)
+ ## \param inString A string that specifies the value to be retrieved
+ def getVarYamlSub(inString):
+ return self.modelyaml[removeEnclosing(inString.group())]
+ return re.sub("\$\(\w+\)", getVarYamlSub, string)
+
+## Wrapper that relaces environment ${} and FRE $() variables
+## \param self the FRE yaml varaibles (FRE properties)
+## \param string The YAML string that is having its variables replaced
+## \returns string with the environment and FRE variables replaced
+ def freVarSub(self, string):
+ tmpString = envReplace(string)
+ returnString = self.freVarReplace(tmpString)
+ return returnString
+
+## Wrapper that takes in a string (yaml) and fills in the FRE and Environment variables
+## \param y Path to yaml file whose variables need to be filled in
+ def fillInYamlWithVars(self,y):
+ with open(y, 'r') as file:
+ yamlString=read(file)
+ return self.freVarSub(yamlString)
diff --git a/lib/fre/make/gfdlfremake/yamlfre.py b/lib/fre/make/gfdlfremake/yamlfre.py
new file mode 100644
index 00000000..72458a8c
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/yamlfre.py
@@ -0,0 +1,199 @@
+import os
+import json
+from pathlib import Path
+import yaml
+from jsonschema import validate, ValidationError, SchemaError
+from . import platformfre
+
+def parseCompile(fname,v):
+ """
+ Brief: Open the yaml file and parse as fremakeYaml
+ Param:
+ - fname the name of the yaml file to parse
+ - v the FRE yaml variables
+ """
+ # Open the yaml file and parse as fremakeYaml
+ with open(fname, 'r') as yamlfile:
+ y = yaml.safe_load(v.freVarSub(yamlfile.read()))
+
+ return y
+
+##### THIS SEEMS UNUSED
+## \brief Checks the yaml for variables. Required variables will dump and error. Non-required variables will
+## set a default value for the variable
+#def yamlVarCheck(var,val="",req=False,err="error"):
+# """
+# Brief: Checks the yaml for variables. Required variables will dump and error.
+# Non-required variables will set a default value for the variable
+# Param:
+# - var A variable in the yaml
+# - val a default value for var
+# - req if true, the variable is required in the yaml and an exception will be raised
+# - err An error message to print if the variable is required and doesn't exist
+# """
+# try:
+# var
+# except:
+# if req:
+# print (err)
+# raise
+# else:
+# var = val
+
+class compileYaml():
+ """
+ Brief: This will read the compile yaml for FRE and then fill in any of the missing non-required variables
+ """
+ def __init__(self,compileinfo):
+ """
+ Brief: Read get the compile yaml and fill in the missing pieces
+ Param:
+ - self the compile Yaml object
+ - compileinfo dictionary with compile information from the combined yaml
+ """
+ # compile information from the combined yaml
+ self.yaml = compileinfo
+
+ ## Check the yaml for required things
+ ## Check for required experiment name
+ try:
+ self.yaml["experiment"]
+ except:
+ print("You must set an experiment name to compile \n")
+ raise
+ ## Check for optional libraries and packages for linking in container
+ try:
+ self.yaml["container_addlibs"]
+ except:
+ self.yaml["container_addlibs"]=""
+ ## Check for optional libraries and packages for linking on bare-metal system
+ try:
+ self.yaml["baremetal_linkerflags"]
+ except:
+ self.yaml["baremetal_linkerflags"]=""
+ ## Check for required src
+ try:
+ self.yaml["src"]
+ except:
+ print("You must set a src to specify the sources in modelRoot/"+self.yaml["experiment"]+"\n")
+ raise
+ ## Loop through the src array
+ for c in self.yaml['src']:
+ ## Check for required componenet name
+ try:
+ c['component']
+ except:
+ print("You must set the 'componet' name for each src component")
+ raise
+ ## Check for required repo url
+ try:
+ c['repo']
+ except:
+ print("'repo' is missing from the component "+c['component']+" in "+self.yaml["experiment"]+"\n")
+ raise
+ # Check for optional branch. Otherwise set it to blank
+ try:
+ c['branch']
+ except:
+ c['branch']=""
+ # Check for optional cppdefs. Otherwise set it to blank
+ try:
+ c['cppdefs']
+ except:
+ c['cppdefs']=""
+ # Check for optional doF90Cpp. Otherwise set it to False
+ try:
+ c['doF90Cpp']
+ except:
+ c['doF90Cpp']=False
+ # Check for optional additional instructions. Otherwise set it to blank
+ try:
+ c['additionalInstructions']
+ except:
+ c['additionalInstructions']=""
+ # Check for optional paths. Otherwise set it to blank
+ try:
+ c['paths']
+ except:
+ c['paths']=[c['component']]
+ # Check for optional requires. Otherwise set it to blank
+ try:
+ c['requires']
+ except:
+ c['requires']=[]
+ # Check for optional overrides. Otherwise set it to blank
+ try:
+ c['makeOverrides']
+ except:
+ c['makeOverrides']=""
+ # Check for optional flags. Otherwise set it to blank.
+ try:
+ c["otherFlags"]
+ except:
+ c["otherFlags"]=""
+
+ def getCompileYaml(self):
+ """
+ Brief: Returns the compile yaml
+ """
+ try:
+ self.yaml
+ except:
+ print ("You must initialize the compile YAML object before you try to get the yaml \n")
+ raise
+ return self.yaml
+
+class freyaml():
+ """
+ Brief: This will take the combined yaml file, parse information, and fill in missing variables
+ to make the full freyaml that can be used and checked
+ Note:
+ - platformYaml: platforms.yaml
+ - compileYaml: compile.yaml
+ """
+ def __init__(self,combinedyaml,v):
+ """
+ Param:
+ - self The freyaml object
+ - combinedyaml The name of the combined yaml file
+ - v FRE yaml variables
+ """
+ self.combinedfile = combinedyaml
+
+ self.freyaml = parseCompile(self.combinedfile, v)
+
+ #get compile info
+ self.compiledict = self.freyaml.get("compile")
+ self.compile = compileYaml(self.compiledict)
+ self.compileyaml = self.compile.getCompileYaml()
+
+ #self.freyaml.update(self.compileyaml)
+
+ #get platform info
+ self.platformsdict = self.freyaml.get("platforms")
+ self.platforms = platformfre.platforms(self.platformsdict)
+ self.platformsyaml = self.platforms.getPlatformsYaml()
+
+ #self.freyaml.update(self.platformsyaml)
+
+ ## VALIDATION OF COMBINED YAML FOR COMPILATION
+ fremake_package_dir = Path(__file__).resolve().parents[2]
+ schema_path = os.path.join(fremake_package_dir, 'gfdl_msd_schemas', 'FRE', 'fre_make.json')
+ with open(schema_path, 'r') as f:
+ s = f.read()
+ schema = json.loads(s)
+
+ validate(instance=self.freyaml,schema=schema)
+ print("\nCOMBINED YAML VALID")
+
+ def getCompileYaml(self):
+ """
+ Brief: Returns the compile yaml
+ """
+ return self.compileyaml
+
+ def getPlatformsYaml(self):
+ """
+ Brief: Returns the compile yaml
+ """
+ return self.platformsyaml
diff --git a/lib/fre/make/gfdlfremake/yamls/SHiELD_example/SHiELD.yaml b/lib/fre/make/gfdlfremake/yamls/SHiELD_example/SHiELD.yaml
new file mode 100644
index 00000000..9f063d0a
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/yamls/SHiELD_example/SHiELD.yaml
@@ -0,0 +1,10 @@
+platformYaml: platforms.yaml
+compileYaml: compile.yaml
+fv3_release: main
+phy_release: main
+fms_release: "2023.02"
+drivers_release: main
+coupler_release: "2023.02"
+FMSincludes: "-IFMS/fms2_io/include -IFMS/include -IFMS/mpp/include"
+momIncludes: "-Imom6/MOM6-examples/src/MOM6/pkg/CVMix-src/include"
+INTEL: intel-classic
diff --git a/lib/fre/make/gfdlfremake/yamls/SHiELD_example/compile.yaml b/lib/fre/make/gfdlfremake/yamls/SHiELD_example/compile.yaml
new file mode 100644
index 00000000..a83bb1ce
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/yamls/SHiELD_example/compile.yaml
@@ -0,0 +1,38 @@
+experiment: shield_nh
+compileInclude: "-IFMS/fms2_io/include -IFMS/include -IFMS/mpp/include"
+container_addlibs: ["bacio","sp","w3emc","w3nco"]
+baremetal_addlibs: ["-L/autofs/ncrc-svm1_proj/epic/spack-stack/spack-stack-1.6.0/envs/unified-env/install/intel/2023.1.0/bacio-2.4.1-wrykbu2/lib -lbacio_4", "-L/autofs/ncrc-svm1_proj/epic/spack-stack/spack-stack-1.6.0/envs/unified-env/install/intel/2023.1.0/bacio-2.4.1-wrykbu2/lib -lbacio_8", "-L/autofs/ncrc-svm1_proj/epic/spack-stack/spack-stack-1.6.0/envs/unified-env/install/intel/2023.1.0/sp-2.5.0-7bumbmx/lib64 -lsp_d", "-L/autofs/ncrc-svm1_proj/epic/spack-stack/spack-stack-1.6.0/envs/unified-env/install/intel/2023.1.0/w3emc-2.10.0-zmuykep/lib64 -lw3emc_d", "-L/autofs/ncrc-svm1_proj/epic/spack-stack/spack-stack-1.6.0/envs/unified-env/install/intel/2023.1.0/w3nco-2.4.1-76qm6h2/lib -lw3nco_d"]
+src:
+ - component: "FMS"
+ repo: "https://github.com/NOAA-GFDL/FMS.git"
+ cppdefs: "-Duse_libMPI -Duse_netCDF -Duse_LARGEFILE -DHAVE_SCHED_GETAFFINITY -DINTERNAL_FILE_NML -DGFS_PHYS -DGFS_CONSTANTS -DHAVE_GETTID"
+ branch: "$(fms_release)"
+ - component: "SHiELD_physics"
+ requires: ["FMS"]
+ repo: "https://github.com/NOAA-GFDL/SHiELD_physics.git"
+ branch: "$(phy_release)"
+ paths: [SHiELD_physics/gsmphys,
+ SHiELD_physics/GFS_layer,
+ SHiELD_physics/IPD_layer]
+ cppdefs: "-Duse_libMPI -Duse_netCDF -DHAVE_SCHED_GETAFFINITY -DSPMD -Duse_LARGEFILE -DGFS_PHYS -DUSE_GFSL63 -DNEW_TAUCTMAX -DNEMS_GSM -DINTERNAL_FILE_NML -DMOIST_CAPPA -DUSE_COND"
+ otherFlags: "$(FMSincludes)"
+ - component: "fv3"
+ requires: ["FMS", "SHiELD_physics"]
+ repo: ["https://github.com/NOAA-GFDL/GFDL_atmos_cubed_sphere.git",
+ "https://github.com/NOAA-GFDL/atmos_drivers.git"]
+ cppdefs: "-Duse_libMPI -Duse_netCDF -DHAVE_SCHED_GETAFFINITY -DSPMD -Duse_LARGEFILE -DGFS_PHYS -DUSE_GFSL63 -DNEW_TAUCTMAX -DNEMS_GSM -DINTERNAL_FILE_NML -DMOIST_CAPPA -DUSE_COND"
+ branch: ["$(fv3_release)","$(drivers_release)"]
+ paths: [SHiELD_physics/FV3GFS/,
+ fv3/atmos_drivers/SHiELD/atmos_model.F90,
+ fv3/GFDL_atmos_cubed_sphere/driver/SHiELD/atmosphere.F90,
+ fv3/GFDL_atmos_cubed_sphere/tools/,
+ fv3/GFDL_atmos_cubed_sphere/model/,
+ fv3/GFDL_atmos_cubed_sphere/GFDL_tools/fv_diag_column.F90]
+ otherFlags: "$(FMSincludes)"
+ - component: "FMScoupler"
+ requires: ["FMS", "SHiELD_physics", "fv3"]
+ repo: "https://github.com/NOAA-GFDL/FMScoupler.git"
+ cppdefs: "-Duse_libMPI -Duse_netCDF -DHAVE_SCHED_GETAFFINITY -DSPMD -Duse_LARGEFILE -DGFS_PHYS -DUSE_GFSL63 -DNEW_TAUCTMAX -DNEMS_GSM -DINTERNAL_FILE_NML -DMOIST_CAPPA -DUSE_COND"
+ branch: "$(coupler_release)"
+ paths: ["FMScoupler/SHiELD/coupler_main.F90"]
+ otherFlags: "$(FMSincludes)"
diff --git a/lib/fre/make/gfdlfremake/yamls/SHiELD_example/platforms.yaml b/lib/fre/make/gfdlfremake/yamls/SHiELD_example/platforms.yaml
new file mode 100644
index 00000000..9f72043b
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/yamls/SHiELD_example/platforms.yaml
@@ -0,0 +1,26 @@
+platforms:
+ - name: ncrc5.intel
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: ["$(INTEL)/2022.2.1","fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/SHiELDtest
+ - name: ncrc5.intel23
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: ["$(INTEL)/2023.1.0","fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/SHiELDtest
+ - name: hpcme.2023
+ compiler: intel
+ RUNenv: [". /spack/share/spack/setup-env.sh", "spack load libyaml", "spack load netcdf-fortran@4.5.4", "spack load hdf5@1.12.1"]
+ modelRoot: /apps
+ fc: mpiifort
+ cc: mpiicc
+ container: True
+ containerBuild: "podman"
+ containerRun: "apptainer"
diff --git a/lib/fre/make/gfdlfremake/yamls/am5.yaml b/lib/fre/make/gfdlfremake/yamls/am5.yaml
new file mode 100644
index 00000000..4b7bf8d4
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/yamls/am5.yaml
@@ -0,0 +1,6 @@
+platformYaml: platforms.yaml
+compileYaml: compile.yaml
+release: f1a1r1
+INTEL: "intel-classic"
+FMSincludes: "-IFMS/fms2_io/include -IFMS/include -IFMS/mpp/include"
+momIncludes: "-Imom6/MOM6-examples/src/MOM6/pkg/CVMix-src/include"
diff --git a/lib/fre/make/gfdlfremake/yamls/compile.yaml b/lib/fre/make/gfdlfremake/yamls/compile.yaml
new file mode 100644
index 00000000..5200599c
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/yamls/compile.yaml
@@ -0,0 +1,66 @@
+experiment: "am5"
+container_addlibs:
+baremetal_linkerflags:
+src:
+ - component: "FMS"
+ repo: "https://github.com/NOAA-GFDL/FMS.git"
+ cppdefs: "-DINTERNAL_FILE_NML -Duse_libMPI -Duse_netCDF"
+ branch: "2022.01"
+ cppdefs: "-DHAVE_GETTID -Duse_libMPI -Duse_netCDF"
+ otherFlags: "$(FMSincludes)"
+ - component: "am5_phys"
+ requires: ["FMS"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/am5_phys.git"
+ branch: "2022.01"
+ otherFlags: "$(FMSincludes)"
+ - component: "GFDL_atmos_cubed_sphere"
+ requires: ["FMS", "am5_phys"]
+ repo: "https://github.com/NOAA-GFDL/GFDL_atmos_cubed_sphere.git"
+ cppdefs: "-DSPMD -DCLIMATE_NUDGE -DINTERNAL_FILE_NML"
+ branch: "2022.01"
+ paths: ["GFDL_atmos_cubed_sphere/driver/GFDL",
+ "GFDL_atmos_cubed_sphere/model",
+ "GFDL_atmos_cubed_sphere/driver/SHiELD/cloud_diagnosis.F90",
+ "GFDL_atmos_cubed_sphere/driver/SHiELD/gfdl_cloud_microphys.F90",
+ "GFDL_atmos_cubed_sphere/tools",
+ "GFDL_atmos_cubed_sphere/GFDL_tools"]
+ otherFlags: "$(FMSincludes)"
+ - component: "atmos_drivers"
+ requires: ["FMS", "am5_phys", "GFDL_atmos_cubed_sphere"]
+ repo: "https://github.com/NOAA-GFDL/atmos_drivers.git"
+ cppdefs: "-DSPMD -DCLIMATE_NUDGE"
+ branch: "2022.01"
+ paths: ["atmos_drivers/coupled"]
+ otherFlags: "$(FMSincludes)"
+ - component: "ice_sis"
+ requires: ["FMS", "ice_param", "mom6"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/ice_sis.git"
+ branch: "2021.02"
+ otherFlags: "$(FMSincludes) $(momIncludes)"
+ - component: "ice_param"
+ repo: "https://github.com/NOAA-GFDL/ice_param.git"
+ cppdefs: "-Duse_yaml -Duse_libMPI -Duse_netCDF"
+ branch: "2021.02"
+ requires: ["FMS", "mom6"]
+ otherFlags: "$(FMSincludes) $(momIncludes)"
+ - component: "land_lad2"
+ requires: ["FMS"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/land_lad2.git"
+ branch: "2022.01"
+ branch: "land_lad2_2021.02"
+ doF90Cpp: True
+ cppdefs: "-DINTERNAL_FILE_NML"
+ otherFlags: "$(FMSincludes)"
+ - component: "mom6"
+ requires: ["FMS"]
+ paths: ["mom6/MOM6-examples/src/MOM6/config_src/dynamic", "mom6/MOM6-examples/src/MOM6/config_src/coupled_driver", "mom6/MOM6-examples/src/MOM6/src/*/", "mom6/MOM6-examples/src/MOM6/src/*/*/", "mom6/ocean_BGC/generic_tracers", "mom6/ocean_BGC/mocsy/src"]
+ branch: ["2021.02","dev/gfdl/2018.04.06"]
+ repo: ["https://github.com/NOAA-GFDL/ocean_BGC.git","https://github.com/NOAA-GFDL/MOM6-examples.git"]
+ makeOverrides: 'OPENMP=""'
+ otherFlags: "$(FMSincludes) $(momIncludes)"
+ - component: "FMScoupler"
+ paths: ["FMScoupler/full", "FMScoupler/shared"]
+ repo: "https://github.com/NOAA-GFDL/FMScoupler.git"
+ branch: "2022.01"
+ requires: ["FMS", "atmos_drivers", "am5_phys", "land_lad2", "ice_sis", "ice_param", "mom6"]
+ otherFlags: "$(FMSincludes) $(momIncludes)"
diff --git a/lib/fre/make/gfdlfremake/yamls/platforms.yaml b/lib/fre/make/gfdlfremake/yamls/platforms.yaml
new file mode 100644
index 00000000..02b7d222
--- /dev/null
+++ b/lib/fre/make/gfdlfremake/yamls/platforms.yaml
@@ -0,0 +1,26 @@
+platforms:
+ - name: ncrc5.intel
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: ["$(INTEL)/2022.2.1","fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: ncrc5.intel23
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: ["$(INTEL)/2023.1.0","fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: hpcme.2023
+ compiler: intel
+ RUNenv: [". /spack/share/spack/setup-env.sh", "spack load libyaml", "spack load netcdf-fortran@4.5.4", "spack load hdf5@1.14.0"]
+ modelRoot: /apps
+ fc: mpiifort
+ cc: mpiicc
+ container: True
+ containerBuild: "podman"
+ containerRun: "apptainer"
diff --git a/lib/fre/make/runFremake.py b/lib/fre/make/runFremake.py
new file mode 100644
index 00000000..9e1c1730
--- /dev/null
+++ b/lib/fre/make/runFremake.py
@@ -0,0 +1,224 @@
+#!/usr/bin/python3
+'''
+date 2023
+author(s): Tom Robinson, Dana Singh, Bennett Chang
+fre make is used to create, run and checkout code, and compile a model.
+'''
+
+import os
+import logging
+from multiprocessing.dummy import Pool
+from pathlib import Path
+import click
+import subprocess
+import fre.yamltools.combine_yamls as cy
+from .gfdlfremake import (
+ targetfre, varsfre, yamlfre, checkout,
+ makefilefre, buildDocker, buildBaremetal )
+
+def fremake_run(yamlfile,platform,target,parallel,jobs,no_parallel_checkout,execute,verbose):
+ ''' run fremake via click'''
+ yml = yamlfile
+ name = yamlfile.split(".")[0]
+ nparallel = parallel
+ jobs = str(jobs)
+ pcheck = no_parallel_checkout
+
+ if pcheck:
+ pc = ""
+ else:
+ pc = " &"
+
+ if verbose:
+ logging.basicConfig(level=logging.INFO)
+ else:
+ logging.basicConfig(level=logging.ERROR)
+
+ #### Main
+ srcDir="src"
+ checkoutScriptName = "checkout.sh"
+ baremetalRun = False # This is needed if there are no bare metal runs
+
+ ## Split and store the platforms and targets in a list
+ plist = platform
+ tlist = target
+
+ # Combined compile yaml file
+ combined = Path(f"combined-{name}.yaml")
+
+ ## If combined yaml exists, note message of its existence
+ ## If combined yaml does not exist, combine model, compile, and platform yamls
+ full_combined = cy.combined_compile_existcheck(combined,yml,platform,target)
+
+ ## Get the variables in the model yaml
+ freVars = varsfre.frevars(full_combined)
+
+ ## Open the yaml file and parse as fremakeYaml
+ modelYaml = yamlfre.freyaml(full_combined,freVars)
+ fremakeYaml = modelYaml.getCompileYaml()
+
+ ## Error checking the targets
+ for targetName in tlist:
+ target = targetfre.fretarget(targetName)
+
+ ## Loop through the platforms specified on the command line
+ ## If the platform is a baremetal platform, write the checkout script and run it once
+ ## This should be done separately and serially because bare metal platforms should all be using
+ ## the same source code.
+ for platformName in plist:
+ if modelYaml.platforms.hasPlatform(platformName):
+ pass
+ else:
+ raise ValueError(f'{platformName} does not exist in '
+ f'{modelYaml.combined.get("compile").get("platformYaml")}')
+
+ ( compiler, modules, modulesInit,
+ fc, cc, modelRoot, iscontainer,
+ mkTemplate, containerBuild, ContainerRun,
+ RUNenv ) = modelYaml.platforms.getPlatformFromName(platformName)
+
+ ## Create the checkout script
+ if not iscontainer:
+ ## Create the source directory for the platform
+ srcDir = modelRoot + "/" + fremakeYaml["experiment"] + "/src"
+ if not os.path.exists(srcDir):
+ os.system("mkdir -p " + srcDir)
+ if not os.path.exists(srcDir+"/checkout.sh"):
+ freCheckout = checkout.checkout("checkout.sh",srcDir)
+ freCheckout.writeCheckout(modelYaml.compile.getCompileYaml(),jobs,pc)
+ freCheckout.finish(pc)
+ os.chmod(srcDir+"/checkout.sh", 0o744)
+ print("\nCheckout script created at "+ srcDir + "/checkout.sh \n")
+ ## TODO: Options for running on login cluster?
+ freCheckout.run()
+
+ fremakeBuildList = []
+ ## Loop through platforms and targets
+ for platformName in plist:
+ for targetName in tlist:
+ target = targetfre.fretarget(targetName)
+ if modelYaml.platforms.hasPlatform(platformName):
+ pass
+ else:
+ raise ValueError (platformName + " does not exist in " + modelYaml.platformsfile)
+ ( compiler, modules, modulesInit,
+ fc, cc, modelRoot, iscontainer,
+ mkTemplate, containerBuild, containerRun,
+ RUNenv ) = modelYaml.platforms.getPlatformFromName(platformName)
+
+ ## Make the source directory based on the modelRoot and platform
+ srcDir = modelRoot + "/" + fremakeYaml["experiment"] + "/src"
+
+ ## Check for type of build
+ if not iscontainer:
+ baremetalRun = True
+ ## Make the build directory based on the modelRoot, the platform, and the target
+ bldDir = f'{modelRoot}/{fremakeYaml["experiment"]}/' + \
+ f'{platformName}-{target.gettargetName()}/exec'
+ os.system("mkdir -p " + bldDir)
+
+ ## Create the Makefile
+ freMakefile = makefilefre.makefile(exp = fremakeYaml["experiment"],
+ libs = fremakeYaml["baremetal_linkerflags"],
+ srcDir = srcDir,
+ bldDir = bldDir,
+ mkTemplatePath = mkTemplate)
+
+
+ # Loop through components, send component name/requires/overrides for Makefile
+ for c in fremakeYaml['src']:
+ freMakefile.addComponent(c['component'],c['requires'],c['makeOverrides'])
+ print("\nMakefile created at " + bldDir + "/Makefile" + "\n")
+ freMakefile.writeMakefile()
+
+ ## Create a list of compile scripts to run in parallel
+ fremakeBuild = buildBaremetal.buildBaremetal(exp = fremakeYaml["experiment"],
+ mkTemplatePath = mkTemplate,
+ srcDir = srcDir,
+ bldDir = bldDir,
+ target = target,
+ modules = modules,
+ modulesInit = modulesInit,
+ jobs = jobs)
+
+ for c in fremakeYaml['src']:
+ fremakeBuild.writeBuildComponents(c)
+ fremakeBuild.writeScript()
+ fremakeBuildList.append(fremakeBuild)
+ ## Run the build if --execute option given, otherwise print out compile script path
+ if execute:
+ fremakeBuild.run()
+ else:
+ print("Compile script created at "+ bldDir+"/compile.sh\n\n")
+ else:
+ ###################### container stuff below #######################################
+ ## Run the checkout script
+ # image="hpc-me-intel:2021.1.1"
+ image="ecpe4s/noaa-intel-prototype:2023.09.25"
+ bldDir = modelRoot + "/" + fremakeYaml["experiment"] + "/exec"
+ tmpDir = "tmp/"+platformName
+
+ ## Create the checkout script
+ freCheckout = checkout.checkoutForContainer("checkout.sh", srcDir, tmpDir)
+ freCheckout.writeCheckout(modelYaml.compile.getCompileYaml(),jobs,pc)
+ freCheckout.finish(pc)
+
+ ## Create the makefile
+ ### Should this even be a separate class from "makefile" in makefilefre? ~ ejs
+ freMakefile = makefilefre.makefileContainer(exp = fremakeYaml["experiment"],
+ libs = fremakeYaml["container_addlibs"],
+ srcDir = srcDir,
+ bldDir = bldDir,
+ mkTemplatePath = mkTemplate,
+ tmpDir = tmpDir)
+
+ # Loop through components and send the component name and requires for the Makefile
+ for c in fremakeYaml['src']:
+ freMakefile.addComponent(c['component'],c['requires'],c['makeOverrides'])
+ freMakefile.writeMakefile()
+
+ ## Build the dockerfile
+ dockerBuild = buildDocker.container(base = image,
+ exp = fremakeYaml["experiment"],
+ libs = fremakeYaml["container_addlibs"],
+ RUNenv = RUNenv,
+ target = target)
+
+ dockerBuild.writeDockerfileCheckout("checkout.sh", tmpDir+"/checkout.sh")
+ dockerBuild.writeDockerfileMakefile(freMakefile.getTmpDir() + "/Makefile",
+ freMakefile.getTmpDir() + "/linkline.sh")
+
+ for c in fremakeYaml['src']:
+ dockerBuild.writeDockerfileMkmf(c)
+
+ dockerBuild.writeRunscript(RUNenv,containerRun,tmpDir+"/execrunscript.sh")
+
+ # Create build script for container
+ dockerBuild.createBuildScript(containerBuild, containerRun)
+ print("Container build script created at "+dockerBuild.userScriptPath+"\n\n")
+
+ # Execute if flag is given
+ if execute:
+ subprocess.run(args=[dockerBuild.userScriptPath], check=True)
+
+ #freCheckout.cleanup()
+ #buildDockerfile(fremakeYaml,image)
+
+ if baremetalRun:
+ if __name__ == '__main__':
+ if execute:
+ # Create a multiprocessing Pool
+ pool = Pool(processes=nparallel)
+ # process data_inputs iterable with pool
+ pool.map(buildBaremetal.fremake_parallel,fremakeBuildList)
+
+@click.command()
+def _fremake_run(yamlfile,platform,target,parallel,jobs,no_parallel_checkout,execute,verbose):
+ '''
+ Decorator for calling _fremake_run - allows the decorated version
+ of the function to be separate from the undecorated version
+ '''
+ return fremake_run(yamlfile,platform,target,parallel,jobs,no_parallel_checkout,execute,verbose)
+
+if __name__ == "__main__":
+ fremake_run()
diff --git a/lib/fre/make/tests/AM5_example/am5.yaml b/lib/fre/make/tests/AM5_example/am5.yaml
new file mode 100644
index 00000000..359755ec
--- /dev/null
+++ b/lib/fre/make/tests/AM5_example/am5.yaml
@@ -0,0 +1,104 @@
+# reusable variables
+fre_properties:
+ - &AM5_VERSION "am5f7b12r1"
+ - &FRE_STEM !join [am5/, *AM5_VERSION]
+
+ # amip
+ - &EXP_AMIP_START "1979"
+ - &EXP_AMIP_END "2020"
+ - &ANA_AMIP_START "1980"
+ - &ANA_AMIP_END "2020"
+
+ - &PP_AMIP_CHUNK96 "1yr"
+ - &PP_AMIP_CHUNK384 "1yr"
+ - &PP_XYINTERP96 "180,288"
+ - &PP_XYINTERP384 "720,1152"
+
+ # climo
+ - &EXP_CLIMO_START96 "0001"
+ - &EXP_CLIMO_END96 "0011"
+ - &ANA_CLIMO_START96 "0002"
+ - &ANA_CLIMO_END96 "0011"
+
+ - &EXP_CLIMO_START384 "0001"
+ - &EXP_CLIMO_END384 "0006"
+ - &ANA_CLIMO_START384 "0002"
+ - &ANA_CLIMO_END384 "0006"
+
+ # coupled
+ - &PP_CPLD_CHUNK_A "5yr"
+ - &PP_CPLD_CHUNK_B "20yr"
+
+ # grids
+ - &GRID_SPEC96 "/archive/oar.gfdl.am5/model_gen5/inputs/c96_grid/c96_OM4_025_grid_No_mg_drag_v20160808.tar"
+
+ # compile information
+ - &release "f1a1r1"
+ - &INTEL "intel-classic"
+ - &FMSincludes "-IFMS/fms2_io/include -IFMS/include -IFMS/mpp/include"
+ - &momIncludes "-Imom6/MOM6-examples/src/MOM6/pkg/CVMix-src/include"
+
+build:
+ # compile information
+ compileYaml: "compile.yaml"
+ platformYaml: "yaml_include/platforms.yaml"
+
+shared:
+ # directories shared across tools
+ directories: &shared_directories
+ history_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, history]
+ pp_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, pp]
+ analysis_dir: !join [/nbhome/$USER/, *FRE_STEM, /, *name]
+ ptmp_dir: "/xtmp/$USER/ptmp"
+ fre_analysis_home: "/home/fms/local/opt/fre-analysis/test"
+
+ # shared pp settings
+ postprocess:
+ settings: &shared_settings
+ history_segment: "P1Y"
+ site: "ppan"
+ switches: &shared_switches
+ do_statics: True
+ do_timeavgs: True
+ clean_work: True
+ do_refinediag: False
+ do_atmos_plevel_masking: True
+ do_preanalysis: False
+ do_analysis: True
+
+experiments:
+ - name: "c96L65_am5f7b12r1_amip"
+ pp:
+ - "yaml_include/pp.c96_amip.yaml"
+ - name: "c96L65_am5f7b12r1_pdclim1850F"
+ pp:
+ - "yaml_include/pp.c96_clim.yaml"
+ - name: "c96L65_am5f7b12r1_pdclim2010F"
+ pp:
+ - "yaml_include/pp.c96_clim.yaml"
+ - name: "c96L65_am5f7b12r1_pdclim2010AERF"
+ pp:
+ - "yaml_include/pp.c96_clim.yaml"
+ - name: "c384L65_am5f7b12r1_amip"
+ pp:
+ - "yaml_include/pp.c384_amip.yaml"
+ - name: "c384L65_am5f7b12r1_pdclim2010F"
+ pp:
+ - "yaml_include/pp.c384_clim.yaml"
+ - name: "c384L65_am5f7b12r1_pdclim1850F"
+ pp:
+ - "yaml_include/pp.c384_clim.yaml"
+ - name: "c384L65_am5f7b12r1_pdclim2010AERF"
+ pp:
+ - "yaml_include/pp.c384_clim.yaml"
+ - name: "c384L65_am5f7b12r1_OM4_p25_piControl_noBLING_DynVeg"
+ pp:
+ - "yaml_include/pp.c384_amip.yaml"
+ - "yaml_include/pp.om4.yaml"
+ - name: "c96L65_am5f7b12r1_OM4_p25_piControl_noBLING_DynVeg"
+ pp:
+ - "yaml_include/pp.c96_amip.yaml"
+ - "yaml_include/pp.om4.yaml"
+ - name: "c96L65_am5f7b12r1_amip_cosp"
+ pp:
+ - "yaml_include/pp.c96_amip.yaml"
diff --git a/lib/fre/make/tests/AM5_example/compile.yaml b/lib/fre/make/tests/AM5_example/compile.yaml
new file mode 100644
index 00000000..5f9a361b
--- /dev/null
+++ b/lib/fre/make/tests/AM5_example/compile.yaml
@@ -0,0 +1,67 @@
+compile:
+ experiment: "am5"
+ container_addlibs:
+ baremetal_linkerflags:
+ src:
+ - component: "FMS"
+ repo: "https://github.com/NOAA-GFDL/FMS.git"
+ cppdefs: "-DINTERNAL_FILE_NML -Duse_libMPI -Duse_netCDF"
+ branch: "2022.01"
+ cppdefs: "-DHAVE_GETTID -Duse_libMPI -Duse_netCDF"
+ otherFlags: *FMSincludes
+ - component: "am5_phys"
+ requires: ["FMS"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/am5_phys.git"
+ branch: "2022.01"
+ otherFlags: *FMSincludes
+ - component: "GFDL_atmos_cubed_sphere"
+ requires: ["FMS", "am5_phys"]
+ repo: "https://github.com/NOAA-GFDL/GFDL_atmos_cubed_sphere.git"
+ cppdefs: "-DSPMD -DCLIMATE_NUDGE -DINTERNAL_FILE_NML"
+ branch: "2022.01"
+ paths: ["GFDL_atmos_cubed_sphere/driver/GFDL",
+ "GFDL_atmos_cubed_sphere/model",
+ "GFDL_atmos_cubed_sphere/driver/SHiELD/cloud_diagnosis.F90",
+ "GFDL_atmos_cubed_sphere/driver/SHiELD/gfdl_cloud_microphys.F90",
+ "GFDL_atmos_cubed_sphere/tools",
+ "GFDL_atmos_cubed_sphere/GFDL_tools"]
+ otherFlags: *FMSincludes
+ - component: "atmos_drivers"
+ requires: ["FMS", "am5_phys", "GFDL_atmos_cubed_sphere"]
+ repo: "https://github.com/NOAA-GFDL/atmos_drivers.git"
+ cppdefs: "-DSPMD -DCLIMATE_NUDGE"
+ branch: "2022.01"
+ paths: ["atmos_drivers/coupled"]
+ otherFlags: *FMSincludes
+ - component: "ice_sis"
+ requires: ["FMS", "ice_param", "mom6"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/ice_sis.git"
+ branch: "2021.02"
+ otherFlags: !join [*FMSincludes, " ", *momIncludes]
+ - component: "ice_param"
+ repo: "https://github.com/NOAA-GFDL/ice_param.git"
+ cppdefs: "-Duse_yaml -Duse_libMPI -Duse_netCDF"
+ branch: "2021.02"
+ requires: ["FMS", "mom6"]
+ otherFlags: !join [*FMSincludes," ", *momIncludes]
+ - component: "land_lad2"
+ requires: ["FMS"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/land_lad2.git"
+ branch: "2022.01"
+ branch: "land_lad2_2021.02"
+ doF90Cpp: True
+ cppdefs: "-DINTERNAL_FILE_NML"
+ otherFlags: *FMSincludes
+ - component: "mom6"
+ requires: ["FMS"]
+ paths: ["mom6/MOM6-examples/src/MOM6/config_src/dynamic", "mom6/MOM6-examples/src/MOM6/config_src/coupled_driver", "mom6/MOM6-examples/src/MOM6/src/*/", "mom6/MOM6-examples/src/MOM6/src/*/*/", "mom6/ocean_BGC/generic_tracers", "mom6/ocean_BGC/mocsy/src"]
+ branch: ["2021.02","dev/gfdl/2018.04.06"]
+ repo: ["https://github.com/NOAA-GFDL/ocean_BGC.git","https://github.com/NOAA-GFDL/MOM6-examples.git"]
+ makeOverrides: 'OPENMP=""'
+ otherFlags: !join [*FMSincludes, " ", *momIncludes]
+ - component: "FMScoupler"
+ paths: ["FMScoupler/full", "FMScoupler/shared"]
+ repo: "https://github.com/NOAA-GFDL/FMScoupler.git"
+ branch: "2022.01"
+ requires: ["FMS", "atmos_drivers", "am5_phys", "land_lad2", "ice_sis", "ice_param", "mom6"]
+ otherFlags: !join [*FMSincludes, " ", *momIncludes]
diff --git a/lib/fre/make/tests/AM5_example/yaml_include/platforms.yaml b/lib/fre/make/tests/AM5_example/yaml_include/platforms.yaml
new file mode 100644
index 00000000..60d1aad2
--- /dev/null
+++ b/lib/fre/make/tests/AM5_example/yaml_include/platforms.yaml
@@ -0,0 +1,26 @@
+platforms:
+ - name: ncrc5.intel
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: [ !join [*INTEL, "/2022.2.1"],"fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: !join ["/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/", *INTEL, ".mk"]
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: ncrc5.intel23
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: [!join [*INTEL, "/2023.1.0"],"fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: !join ["/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/", *INTEL, ".mk"]
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: hpcme.2023
+ compiler: intel
+ RUNenv: [". /spack/share/spack/setup-env.sh", "spack load libyaml", "spack load netcdf-fortran@4.5.4", "spack load hdf5@1.14.0"]
+ modelRoot: /apps
+ fc: mpiifort
+ cc: mpiicc
+ container: True
+ containerBuild: "podman"
+ containerRun: "apptainer"
diff --git a/lib/fre/make/tests/AM5_example/yaml_include/pp.c96_amip.yaml b/lib/fre/make/tests/AM5_example/yaml_include/pp.c96_amip.yaml
new file mode 100644
index 00000000..117c66c6
--- /dev/null
+++ b/lib/fre/make/tests/AM5_example/yaml_include/pp.c96_amip.yaml
@@ -0,0 +1,88 @@
+# local reusable variable overrides
+fre_properties:
+ - &custom_interp "180,360"
+
+# directory overrides
+#c96_amip_directories:
+directories:
+ <<: *shared_directories
+ ptmp_dir: "/ptmp/$USER"
+ pp_grid_spec: *GRID_SPEC96
+
+#c96_amip_postprocess:
+postprocess:
+ # pp setting overrides
+ settings:
+ <<: *shared_settings
+ pp_start: *ANA_AMIP_START
+ pp_stop: *ANA_AMIP_END
+ pp_chunk_a: *PP_AMIP_CHUNK96
+ pp_components: "atmos atmos_scalar"
+ switches:
+ <<: *shared_switches
+ do_statics: False
+
+ # main pp instructions
+ components:
+ - type: "atmos_cmip"
+ sources: "atmos_month_cmip atmos_8xdaily_cmip atmos_daily_cmip"
+ sourceGrid: "cubedsphere"
+ xyInterp: *custom_interp
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos"
+ sources: "atmos_month"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_level_cmip"
+ sources: "atmos_level_cmip"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_level"
+ sources: "atmos_month"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_month_aer"
+ sources: "atmos_month_aer"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'atmos'
+ - type: "atmos_diurnal"
+ sources: "atmos_diurnal"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_scalar"
+ sources: "atmos_scalar"
+ - type: "aerosol_cmip"
+ xyInterp: *PP_XYINTERP96
+ sources: "aerosol_month_cmip"
+ sourceGrid: "cubedsphere"
+ interpMethod: "conserve_order1"
+ inputRealm: 'atmos'
+ - type: "land"
+ sources: "land_month"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'land'
+ - type: "land_cmip"
+ sources: "land_month_cmip"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'land'
+ - type: "tracer_level"
+ sources: "atmos_tracer"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'atmos'
diff --git a/lib/fre/make/tests/__init__.py b/lib/fre/make/tests/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/make/tests/compilation/test_fre_make_run_fremake.py b/lib/fre/make/tests/compilation/test_fre_make_run_fremake.py
new file mode 100644
index 00000000..c693a7a5
--- /dev/null
+++ b/lib/fre/make/tests/compilation/test_fre_make_run_fremake.py
@@ -0,0 +1,26 @@
+''' test "fre make run-fremake" calls '''
+
+import os
+from pathlib import Path
+
+import pytest
+
+from fre.make import runFremake
+
+
+# command options
+YAMLFILE = "fre/make/tests/null_example/null_model.yaml"
+PLATFORM = [ "ci.gnu" ]
+CONTAINER_PLATFORM = ["hpcme.2023"]
+TARGET = ["debug"]
+EXPERIMENT = "null_model_full"
+
+# get HOME dir to check output
+HOME_DIR = os.environ["HOME"]
+
+@pytest.mark.skip(reason='failing: fix in development, see PR 275')
+def test_fre_make_run_fremake_null_model_serial_compile():
+ ''' run fre make with run-fremake subcommand and build the null model experiment with gnu'''
+ runFremake.fremake_run(YAMLFILE, PLATFORM, TARGET, False, 1, False, True, False)
+ assert Path(f"{HOME_DIR}/fremake_canopy/test/{EXPERIMENT}/{PLATFORM[0]}-{TARGET[0]}/exec/{EXPERIMENT}.x").exists()
+
diff --git a/lib/fre/make/tests/null_example/compile.yaml b/lib/fre/make/tests/null_example/compile.yaml
new file mode 100644
index 00000000..68c151f8
--- /dev/null
+++ b/lib/fre/make/tests/null_example/compile.yaml
@@ -0,0 +1,48 @@
+compile:
+ experiment: "null_model_full"
+ container_addlibs:
+ baremetal_linkerflags:
+ src:
+ - component: "FMS"
+ repo: "https://github.com/NOAA-GFDL/FMS.git"
+ cppdefs: "-Duse_netCDF -Duse_libMPI -DMAXFIELDS_=200 -DMAXFIELDMETHODS_=200 -DINTERNAL_FILE_NML -DHAVE_GETTID"
+ otherFlags: "-fallow-argument-mismatch" # only needed for gcc
+ branch: *branch
+ - component: "atmos_null"
+ requires: ["FMS"]
+ repo: "https://github.com/NOAA-GFDL/atmos_null.git"
+ branch: *branch
+ cppdefs: "-DINTERNAL_FILE_NML"
+ otherFlags: *FMSincludes
+ - component: "land_null"
+ requires: ["FMS"]
+ repo: "https://github.com/NOAA-GFDL/land_null.git"
+ branch: *branch
+ cppdefs: "-DINTERNAL_FILE_NML"
+ otherFlags: *FMSincludes
+ - component: "ice_param"
+ requires: ["FMS"]
+ repo: "https://github.com/NOAA-GFDL/ice_param.git"
+ branch: *branch
+ cppdefs: "-DINTERNAL_FILE_NML"
+ otherFlags: *FMSincludes
+ - component: "ocean_null"
+ requires: ["FMS"]
+ repo: "https://github.com/NOAA-GFDL/ocean_null.git"
+ branch: *branch
+ otherFlags: *FMSincludes
+ - component: "ice_null"
+ requires: ["FMS", "ice_param", "ocean_null"]
+ repo: "https://github.com/NOAA-GFDL/ice_null.git"
+ branch: *branch
+ cppdefs: "-DINTERNAL_FILE_NML"
+ otherFlags: *FMSincludes
+ - component: "coupler"
+ requires: ["FMS", "ocean_null", "atmos_null", "land_null", "ice_param", "ocean_null", "ice_null"]
+ repo: "https://github.com/NOAA-GFDL/FMScoupler.git"
+ branch: *branch
+ otherFlags: *FMSincludes
+ cppdefs: "-D_USE_LEGACY_LAND_ -Duse_AM3_physics -DINTERNAL_FILE_NML"
+ paths: [ "coupler/full",
+ "coupler/shared" ]
+
diff --git a/lib/fre/make/tests/null_example/null_model.yaml b/lib/fre/make/tests/null_example/null_model.yaml
new file mode 100644
index 00000000..2dcbcad4
--- /dev/null
+++ b/lib/fre/make/tests/null_example/null_model.yaml
@@ -0,0 +1,15 @@
+# reusable variables
+fre_properties:
+ - &VERSION "full"
+ - &FRE_STEM !join [null_model/, *VERSION]
+ # compile information
+ - &branch "main"
+ - &INTEL "intel-classic"
+ - &FMSincludes "-IFMS/include"
+
+build:
+ compileYaml: "compile.yaml"
+ platformYaml: "platforms.yaml"
+
+experiments:
+ - name: "null_model_full"
diff --git a/lib/fre/make/tests/null_example/platforms.yaml b/lib/fre/make/tests/null_example/platforms.yaml
new file mode 100644
index 00000000..d2a3c97f
--- /dev/null
+++ b/lib/fre/make/tests/null_example/platforms.yaml
@@ -0,0 +1,24 @@
+platforms:
+ - name: ncrc5.intel23
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: [!join [*INTEL, "/2023.2.0"],"fre/bronx-21",cray-hdf5/1.12.2.11, cray-netcdf/4.9.0.11]
+ fc: ftn
+ cc: cc
+ mkTemplate: !join ["/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/", *INTEL, ".mk"]
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: hpcme.2023
+ compiler: intel
+ RUNenv: [". /spack/share/spack/setup-env.sh", "spack load libyaml", "spack load netcdf-fortran@4.5.4", "spack load hdf5@1.14.0"]
+ modelRoot: /apps
+ fc: mpiifort
+ cc: mpiicc
+ container: True
+ containerBuild: "podman"
+ containerRun: "apptainer"
+ - name: ci.gnu
+ compiler: gnu
+ fc: mpifort
+ cc: mpicc
+ mkTemplate: /__w/fre-cli/fre-cli/mkmf/templates/linux-ubuntu-xenial-gnu.mk
+ modelRoot: ${HOME}/fremake_canopy/test
diff --git a/lib/fre/make/tests/test_create_makefile.py b/lib/fre/make/tests/test_create_makefile.py
new file mode 100644
index 00000000..1fe2001c
--- /dev/null
+++ b/lib/fre/make/tests/test_create_makefile.py
@@ -0,0 +1,75 @@
+"""
+Test fre make create-makefile
+"""
+import os
+import shutil
+from pathlib import Path
+from fre.make import createMakefile
+
+# SET-UP
+TEST_DIR = Path("fre/make/tests")
+NM_EXAMPLE = Path("null_example")
+YAMLFILE = "null_model.yaml"
+BM_PLATFORM = ["ncrc5.intel23"]
+CONTAINER_PLATFORM = ["hpcme.2023"]
+TARGET = ["debug"]
+EXPERIMENT = "null_model_full"
+
+# Create output location
+OUT = f"{TEST_DIR}/makefile_out"
+if Path(OUT).exists():
+ # remove
+ shutil.rmtree(OUT)
+ # create output directory
+ Path(OUT).mkdir(parents=True,exist_ok=True)
+else:
+ Path(OUT).mkdir(parents=True,exist_ok=True)
+
+# Set output directory as home for fre make output
+#os.environ["HOME"]=str(Path(OUT))
+
+def test_modelyaml_exists():
+ """
+ Check the model yaml exists
+ """
+ assert Path(f"{TEST_DIR}/{NM_EXAMPLE}/{YAMLFILE}").exists()
+
+def test_compileyaml_exists():
+ """
+ Check the compile yaml exists
+ """
+ assert Path(f"{TEST_DIR}/{NM_EXAMPLE}/compile.yaml").exists()
+
+def test_platformyaml_exists():
+ """
+ Check the platform yaml exists
+ """
+ assert Path(f"{TEST_DIR}/{NM_EXAMPLE}/platforms.yaml").exists()
+
+def test_bm_makefile_creation():
+ """
+ Check the makefile is created when a bare-metal platform is used
+ """
+ # Set output directory as home for fre make output
+ def_home = str(os.environ["HOME"])
+ os.environ["HOME"]=OUT#str(Path(OUT))
+
+ bm_plat = BM_PLATFORM[0]
+ targ = TARGET[0]
+ yamlfile_path = f"{TEST_DIR}/{NM_EXAMPLE}/{YAMLFILE}"
+
+ createMakefile.makefile_create(yamlfile_path,BM_PLATFORM,TARGET)
+
+ assert Path(f"{OUT}/fremake_canopy/test/{EXPERIMENT}/{bm_plat}-{targ}/exec/Makefile").exists()
+ os.environ["HOME"] = def_home
+ assert os.environ["HOME"] == def_home
+
+def test_container_makefile_creation():
+ """
+ Check the makefile is created when the container platform is used
+ """
+ container_plat = CONTAINER_PLATFORM[0]
+ yamlfile_path = f"{TEST_DIR}/{NM_EXAMPLE}/{YAMLFILE}"
+ createMakefile.makefile_create(yamlfile_path,CONTAINER_PLATFORM,TARGET)
+
+ assert Path(f"tmp/{container_plat}/Makefile").exists()
diff --git a/lib/fre/pp/README.md b/lib/fre/pp/README.md
new file mode 100644
index 00000000..173c1055
--- /dev/null
+++ b/lib/fre/pp/README.md
@@ -0,0 +1,148 @@
+
+
+
+
+
+
+
+## **Quickstart instructions to postprocess FMS history output on PP/AN or gaea**
+
+1. Checkout postprocessing workflow template
+This will clone the postprocessing repository into `/home/$USER/cylc-src/EXPNAME__PLATFORM__TARGET`.
+```
+module load fre/canopy
+fre pp checkout -e EXPNAME -p PLATFORM -t TARGET
+```
+
+2. Configure pp template with either XML or pp.yaml
+
+```
+fre pp configure-xml -e EXPNAME -p PLATFORM -t TARGET -x XML
+```
+or
+```
+fre pp configure-yaml -e EXPNAME -p PLATFORM -t TARGET -y YAML
+
+```
+
+3. (OPTIONAL BUT RECCOMENDED) Create `history-manifest` for config validation
+
+Create a `history-manifest` of a single tar file archive first for use in the validation.
+This list represents the available source files within the history tar archives, and enables the
+validation procedure to catch a wider variety of potential errors. This can be done like so-
+```
+tar -tf /archive/$USER/path/to/history/files/YYYYMMDD.nc.tar | grep -v tile[2-6] | sort > /home/$USER/cylc-src/EXPNAME__PLATFORM__TARGET/history-manifest
+```
+
+4. Validate the configuration
+```
+fre pp validate -e EXPNAME -p PLATFORM -t TARGET
+```
+
+Warnings related to directories are probably valid and should be fixed in `rose-suite.conf`, or created as necessary via `mkdir`.
+
+If you are running postprocessing gaea, you'll need to change the `SITE` variable in `rose-suite.conf` from `ppan` to `gaea`.
+
+5. Install the workflow
+
+```
+fre pp install -e EXPNAME -p PLATFORM -t TARGET
+```
+
+If you are attempting this on gaea, you'll need to make two one-time changes before installing.
+- Currently, `cylc`, `rose`, and `isodatetime` must be in your PATH for new shells. One approach to do this is
+to symlink the fms-user-installed fre-cli cylc/rose/isodatetime scripts into your local `~/bin` directory,
+and then add that `~/bin` directory to your PATH in your `.bashrc` or `.cshrc`. (If you don't do this, Cylc tasks
+will fail complaining those 3 tools are not available.)
+
+```
+cd ~/bin
+ln -s /ncrc/home2/Flexible.Modeling.System/conda/envs/fre-cli/bin/{cylc,rose,isodatetime} .
+echo 'setenv PATH ${PATH}:~/bin' >> ~/.cshrc
+```
+- Currently, the cylc available on gaea (through `module load cylc` or the `PATH` trick above) does not
+include any global configuration, so you'll need to create a file `~/.cylc/flow/global.cylc` that contains the following.
+If you don't do this, Cylc will use your home directory for the scratch space and rapidly fill your quota.)
+
+```
+[install]
+ [[symlink dirs]]
+ [[[localhost]]]
+ run = /gpfs/f5/scratch/gfdl_f/$USER
+```
+
+6. Run the workflow
+
+```
+fre pp run -e EXPNAME -p PLATFORM -t TARGET
+```
+
+7. Report status of workflow progress
+
+```
+fre pp status -e EXPNAME -p PLATFORM -t TARGET
+```
+
+8. Launch GUI
+
+```
+TODO: fre pp gui?
+
+The full GUI can be launched on jhan or jhanbigmem (an107 or an201).
+
+cylc gui --ip=`hostname -f` --port=`jhp 1` --no-browser
+```
+
+## **PP Wrapper Usage**
+
+Here
+
+### **PP Wrapper Decision Tree**
+![pp_wrapper_decsiontree](https://github.com/NOAA-GFDL/fre-cli/assets/98476720/d3eaa237-1e29-4922-9d83-8d9d11925c54)
+
+### **Tests**
+
+To run `fre pp` tests, return to root directory of the fre-cli repo and call just those tests with
+
+ python -m pytest fre/pp/tests/[test script.py]
+
+Or run all tests with
+
+ python -m pytests fre/pp/tests
diff --git a/lib/fre/pp/__init__.py b/lib/fre/pp/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/pp/checkout_script.py b/lib/fre/pp/checkout_script.py
new file mode 100644
index 00000000..ccd99368
--- /dev/null
+++ b/lib/fre/pp/checkout_script.py
@@ -0,0 +1,98 @@
+'''
+Description: Checkout script which accounts for 4 different scenarios:
+1. branch not given, folder does not exist,
+2. branch given, folder does not exist,
+3. branch not given, folder exists,
+4. branch given and folder exists
+'''
+import os
+import sys
+import subprocess
+
+import click
+
+import fre
+
+FRE_WORKFLOWS_URL = 'https://github.com/NOAA-GFDL/fre-workflows.git'
+
+def checkout_template(experiment = None, platform = None, target = None, branch = None):
+ """
+ Checkout the workflow template files from the repo
+ """
+ ## Chdir back to here before we exit this routine
+ go_back_here = os.getcwd()
+
+ # branch and version parameters
+ default_tag = fre.__version__
+ print(f'(checkout_script) default_tag is {default_tag}')
+
+
+ # check args + set the name of the directory
+ if None in [experiment, platform, target]:
+ raise ValueError( 'one of these are None: experiment / platform / target = \n'
+ f'{experiment} / {platform} / {target}' )
+ name = f"{experiment}__{platform}__{target}"
+
+ # Create the directory if it doesn't exist
+ directory = os.path.expanduser("~/cylc-src")
+ try:
+ os.makedirs(directory, exist_ok = True)
+ except Exception as exc:
+ raise OSError(
+ '(checkoutScript) directory {directory} wasnt able to be created. exit!') from exc
+
+ print(f'(checkout_script) branch is {branch}')
+ checkout_exists = os.path.isdir(f'{directory}/{name}')
+ git_clone_branch_arg = branch if branch is not None else default_tag
+ if branch is not None:
+ print('(checkout_script) WARNING using default_tag as branch argument for git clone!')
+
+
+ if not checkout_exists: # scenarios 1+2, checkout doesn't exist, branch specified (or not)
+ clone_output = subprocess.run( ['git', 'clone','--recursive',
+ f'--branch={git_clone_branch_arg}',
+ FRE_WORKFLOWS_URL, f'{directory}/{name}'],
+ capture_output = True, text = True, check = True)
+ print(f'(checkout_script) output git clone command: {clone_output}')
+
+ else: # the repo checkout does exist, scenarios 3 and 4.
+ os.chdir(f'{directory}/{name}')
+
+ name_path_tag_subproc_out = subprocess.run(["git","describe","--tags"],
+ capture_output = True,
+ text = True, check = True).stdout
+ if branch is not None:
+ name_path_tag = name_path_tag_subproc_out.split('*')
+ name_path_branch = subprocess.run(["git","branch"],
+ capture_output = True,
+ text = True, check = True).stdout.split()[0]
+ if all( [ default_tag not in name_path_tag,
+ name_path_branch != branch ] ):
+ sys.exit(
+ f"Tag and branch of prexisting directory {directory}/{name} does not match "
+ "fre --version or branch requested" )
+ else:
+ name_path_tag = name_path_tag_subproc_out.split()[0]
+ if not default_tag in name_path_tag:
+ sys.exit(
+ f"Tag of prexisting directory {directory}/{name} does not match fre --version")
+
+ # make sure we are back where we should be
+ if os.getcwd() != go_back_here:
+ os.chdir(go_back_here)
+
+ return 0
+
+#############################################
+
+@click.command()
+def _checkout_template(experiment, platform, target, branch = None):
+ '''
+ Wrapper script for calling checkout_template - allows the decorated version
+ of the function to be separate from the undecorated version
+ '''
+ return checkout_template(experiment, platform, target, branch)
+
+
+if __name__ == '__main__':
+ checkout_template()
diff --git a/lib/fre/pp/configure_script_xml.py b/lib/fre/pp/configure_script_xml.py
new file mode 100644
index 00000000..38247ce1
--- /dev/null
+++ b/lib/fre/pp/configure_script_xml.py
@@ -0,0 +1,695 @@
+'''
+ Primary Usage: fre-bronx-to-canopy -x XML -e EXP -p PLATFORM -t TARGET
+
+ The Bronx-to-Canopy XML converter overwrites 3 files:
+ - rose-suite.conf
+ - app/remap-pp-components/rose-app.conf
+ - app/regrid-xy/rose-app.conf
+'''
+
+# std lib
+import re
+import os
+import subprocess
+import logging
+
+# third party
+import click
+import metomi.rose.config
+import metomi.isodatetime.parsers
+
+#############################################
+
+LOGGING_FORMAT = '%(asctime)s %(levelname)s: %(message)s'
+CYLC_PATH = '/home/fms/fre-canopy/system-settings/bin'
+CYLC_REFINED_SCRIPTS = ["check4ptop.pl",
+ "module_init_3_1_6.pl",
+ "plevel_mask.ncl",
+ "refineDiag_atmos.csh",
+ "refine_fields.pl",
+ "surface_albedo.ncl",
+ "tasminmax.ncl",
+ "tracer_refine.ncl",
+ "refineDiag_atmos_cmip6.csh"
+ ]
+CYLC_REFINED_DIR = "\\$CYLC_WORKFLOW_RUN_DIR/etc/refineDiag"
+PREANALYSIS_SCRIPT = "refineDiag_data_stager_globalAve.csh"
+
+
+def freq_from_legacy(legacy_freq):
+ """Return ISO8601 duration given Bronx-style frequencies
+
+ Arguments:
+ 1. legacy_freq[str]: The Bronx frequency
+ """
+ lookup = {
+ 'annual': 'P1Y',
+ 'monthly': 'P1M',
+ 'seasonal': 'P3M',
+ 'daily': 'P1D',
+ '120hr': 'P120D',
+ '12hr': 'PT12H',
+ '8hr': 'PT8H',
+ '6hr': 'PT6H',
+ '4hr': 'PT4H',
+ '3hr': 'PT3H',
+ '2hr': 'PT2H',
+ '1hr': 'PT1H',
+ 'hourly': 'PT1H',
+ '30min': 'PT30M'
+ }
+ return lookup[legacy_freq]
+
+def chunk_from_legacy(legacy_chunk):
+ """Return ISO8601 duration given Bronx-style chunk
+
+ Arguments:
+ 1. legacy_chunk[str]: The Bronx chunk
+ """
+ regex = re.compile( r'(\d+)(\w+)' )
+ match = regex.match( legacy_chunk )
+ if not match:
+ logging.error( "Could not convert Bronx chunk to ISO8601 duration: %s",
+ legacy_chunk )
+ raise ValueError
+
+ time_unit = match.group(2)
+ if time_unit not in ['yr','mo']:
+ logging.error("Unknown time units %s", match.group(2) )
+ raise ValueError
+
+ time_quant = match.group(1)
+ ret_val=f'P{time_quant}'
+ if time_unit == "yr":
+ ret_val+='Y'#return f'P{time_quant}Y'
+ elif time_unit == 'mo':
+ ret_val+='M'#return f'P{time_quant}M'
+ return ret_val
+
+def frelist_xpath(xml, platform, target, experiment,
+# do_analysis, historydir, refinedir, ppdir,
+# do_refinediag, pp_start, pp_stop, validate, verbose, quiet, dual,
+ xpath):
+ """Returns filepaths of FRE XML elements that use X-path notation
+ using Bronx's 'frelist' command via subprocess module
+
+ Arguments:
+ 1. args[str]: Argparse user-input arguments
+ 2. xpath[str]: X-path (XML) notation required by 'frelist'
+ """
+
+ cmd = f"frelist -x {xml} -p {platform} -t {target} {experiment} --evaluate '{xpath}'"
+ logging.info("running cmd:\n %s",cmd) #logging.info(f"running cmd:\n {cmd}")
+ logging.info(">> %s",xpath)
+ process = subprocess.run(cmd,
+ shell=True,
+ check=False, #if True, retrieving std err difficult...
+ capture_output=True,
+ universal_newlines=True)
+
+ logging.info("stdout: \n%s",process.stdout.strip())
+ logging.info("stderr: \n%s",process.stderr.strip())
+ logging.info("returncode: \n%s",process.returncode)
+
+
+ result = process.stdout.strip()
+ if process.returncode > 0:
+ raise subprocess.CalledProcessError(
+ returncode=process.returncode,
+ cmd=cmd,
+ output=process.stdout.strip(),
+ stderr=process.stderr.strip() )
+ return result
+
+def duration_to_seconds(duration):
+ """Returns the conversion of a chunk duration to seconds
+
+ Arguments:
+ 1. duration[str]: The original chunk duration
+ """
+ dur = metomi.isodatetime.parsers.DurationParser().parse(duration)
+ return dur.get_seconds()
+
+
+def main(xml, platform, target, experiment, do_analysis, historydir, refinedir, ppdir,
+ do_refinediag, pp_start, pp_stop, validate, verbose, quiet, dual):
+ """The meat of the converter
+
+ Arguments:
+ 1. args[argparse Namespace]: Arguments given at the command line
+
+ Tasks:
+ 1. Generate key-value pairs for the rose-suite.conf.
+ 2. Generate content for the regrid-xy app
+ 3. Generate content for the remap-pp-components app
+ """
+
+ ##########################################################################
+ # Set up default configurations for regrid-xy and remap-pp-components
+ ##########################################################################
+ rose_remap = metomi.rose.config.ConfigNode()
+ rose_remap.set(keys=['command', 'default'], value='remap-pp-components')
+
+ rose_regrid_xy = metomi.rose.config.ConfigNode()
+ rose_regrid_xy.set(keys=['command', 'default'], value='regrid-xy')
+
+ ##########################################################################
+ # Create the rose-suite config and begin setting up key-value pairs
+ # Note: All strings inside the rose-suite configuration MUST be quoted.
+ # Note: The exception to the 'quote' rule is boolean values.
+ # Note: Addition of quotes will appear as "'" in the code.
+ ##########################################################################
+ rose_suite = metomi.rose.config.ConfigNode()
+ rose_suite.set(keys=['template variables', 'SITE'], value='"ppan"')
+ rose_suite.set(keys=['template variables', 'CLEAN_WORK'], value='True')
+ rose_suite.set(keys=['template variables', 'PTMP_DIR'], value='"/xtmp/$USER/ptmp"')
+ rose_suite.set(keys=['template variables', 'DO_MDTF'], value='False')
+ rose_suite.set(keys=['template variables', 'DO_STATICS'], value='True')
+ rose_suite.set(keys=['template variables', 'DO_TIMEAVGS'], value='True')
+ rose_suite.set(keys=['template variables', 'DO_ANALYSIS_ONLY'], value='False')
+ rose_suite.set(keys=['template variables', 'DO_ATMOS_PLEVEL_MASKING'], value='True')
+ rose_suite.set(keys=['template variables', 'FRE_ANALYSIS_HOME'],
+ value='"/home/fms/local/opt/fre-analysis/test"')
+
+ # not sure about these
+ rose_suite.set(keys=['template variables', 'PP_DEFAULT_XYINTERP'], value='"360,180"')
+ rose_suite.set(keys=['template variables', 'DO_ANALYSIS'], value='True')
+
+ rose_suite.set(keys=['template variables', 'EXPERIMENT'],
+ value=f"'{experiment}'") #value=f"'{}'".format(experiment))
+ rose_suite.set(keys=['template variables', 'PLATFORM'],
+ value=f"'{platform}'")#value="'{}'".format(platform))
+ rose_suite.set(keys=['template variables', 'TARGET'],
+ value=f"'{target}'")#value="'{}'".format(target))
+
+ #regex_fre_property = re.compile(r'\$\((\w+)') #notyetimplemented
+ #all_components = set() #notyetimplemented
+
+ ##########################################################################
+ # Run 'frelist' in the background to fetch the default history directory,
+ # the default history_refineDiag directory, and the default PP directory,
+ # all of which have been set in the XML. If the custom arguments for these
+ # directory variables have not been set by the user at the command line,
+ # the XML's default paths will be inserted into rose-suite.conf.
+ ##########################################################################
+ if historydir is None:
+ logging.info("Running frelist for historydir assignment, this may fail")
+ logging.info("If so, try the frelist call manually and use the historydir argument")
+
+ fetch_history_cmd = f"frelist -x {xml} -p {platform} -t {target} {experiment} -d archive"
+ logging.info(">> %s", fetch_history_cmd)
+
+ fetch_history_process = subprocess.run(fetch_history_cmd,
+ shell=True,
+ check=True,
+ capture_output=True,
+ universal_newlines=True)
+ historydir = fetch_history_process.stdout.strip() + '/history'
+ logging.info(historydir)
+
+
+
+ # Q: should respond to do_refineDiag and refinedir args of this function?
+ historydir_refined = historydir + '_refineDiag'
+
+ if ppdir is None:
+ logging.info("Running frelist for ppdir assignment...")
+ logging.info("If this fails, try the frelist call manually and use the ppdir argument")
+ fetch_pp_cmd = f"frelist -x {xml} -p {platform} -t {target} {experiment} -d postProcess"
+ logging.info(">> %s", fetch_pp_cmd)
+
+ fetch_pp_process = subprocess.run(fetch_pp_cmd,
+ shell=True,
+ check=True,
+ capture_output=True,
+ universal_newlines=True)
+ ppdir = fetch_pp_process.stdout.strip()
+ logging.info(ppdir)
+
+
+ # Q: shouldn't there be a CLI analysis dir arg while we're here?
+ # basically, this is borderline on the same level as the ppDir and historydir fields.
+ #if do_analysis:
+ logging.info("Running frelist for analysis_dir assignment...")
+ fetch_analysis_dir_cmd = f"frelist -x {xml} -p {platform} -t {target} {experiment} -d analysis"
+ logging.info(">> %s", fetch_analysis_dir_cmd)
+
+ fetch_analysis_dir_process = subprocess.run(fetch_analysis_dir_cmd,
+ shell=True,
+ check=True,
+ capture_output=True,
+ universal_newlines=True)
+ analysis_dir = fetch_analysis_dir_process.stdout.strip()
+ logging.info(analysis_dir)
+ #else:
+ # logging.info('not doing analysis.')
+
+
+ grid_spec = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir, do_refinediag,
+ #pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ 'input/dataFile[@label="gridSpec"]')
+ sim_time = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir, do_refinediag,
+ #pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ 'runtime/production/@simTime')
+ sim_units = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir, do_refinediag,
+ #pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ 'runtime/production/@units')
+
+ rose_suite.set(keys=['template variables', 'HISTORY_DIR'],
+ value=f"'{historydir}'") #value="'{}'".format(historydir))
+ # set some dirs to something else to allow bronx dual-pps easily
+ if dual:
+ rose_suite.set(keys=['template variables', 'PP_DIR'],
+ value=f"'{ppdir}_canopy'")
+ rose_suite.set(keys=['template variables', 'ANALYSIS_DIR'],
+ value=f"'{analysis_dir}_canopy'")
+ else:
+ rose_suite.set(keys=['template variables', 'PP_DIR'],
+ value=f"'{ppdir}'")
+ rose_suite.set(keys=['template variables', 'ANALYSIS_DIR'],
+ value=f"'{analysis_dir}'")
+ rose_suite.set(keys=['template variables', 'PP_GRID_SPEC'],
+ value=f"'{grid_spec}'")
+
+ ##########################################################################
+ # Process the refineDiag scripts into the rose-suite configuration from
+ # the
tags in the XML. There is one special
+ # script that contains its own key-value pair: PREANALYSIS. Everything
+ # else gets processed into a list of strings under the REFINEDIAG_SCRIPT
+ # rose-suite setting. Also, if a script referenced can also be found in
+ # Canopy's centralized refineDiag repository, located in the
+ # $CYLC_WORKFLOW_DIR/etc/refineDiag, then THOSE references will be used.
+ # Otherwise, the reference will be whatever path the XML finds.
+ ##########################################################################
+ #preanalysis_path_xml = None #notyetimplemented
+ #preanalysis_path_cylc = "'{}/{}'".format(CYLC_REFINED_DIR, #notyetimplemented
+ # PREANALYSIS_SCRIPT)
+
+ # get the refinediag scripts
+ refine_diag_cmd = f"frelist -x {xml} -p {platform} -t {target} {experiment} " \
+ "--evaluate postProcess/refineDiag/@script"
+ refine_diag_process = subprocess.run(refine_diag_cmd,
+ shell=True,
+ check=True,
+ capture_output=True,
+ universal_newlines=True)
+ refine_diag_scripts = refine_diag_process.stdout.strip('\n')
+
+ # If one of the refinediag scripts contains "vitals", assume it
+ # won't generate output, so relabel it as a preAnalysis script.
+ # There is only one preAnalysis slot currently, so if there are multiple
+ # refineDiag scripts that match "vitals" throw away the rest.
+ list_refinediags = []
+ str_preanalysis = None
+ for x in refine_diag_scripts.split():
+ if "vitals" in x:
+ if str_preanalysis is None:
+ str_preanalysis = x
+ else:
+ list_refinediags.append(x)
+
+ if list_refinediags:
+ # turn refinediag off by default in favor of the built-in plevel masking
+ rose_suite.set(keys=['template variables', 'DO_REFINEDIAG'], value='False')
+ if dual:
+ rose_suite.set(keys=['template variables', '#HISTORY_DIR_REFINED'],
+ value=f"'{historydir_refined}_canopy'")
+ else:
+ rose_suite.set(keys=['template variables', '#HISTORY_DIR_REFINED'],
+ value=f"'{historydir_refined}'")
+
+ rose_suite.set(keys=['template variables', '#REFINEDIAG_SCRIPTS'],
+ value=f"'{' '.join(list_refinediags)}'")
+
+ logging.info( "refineDiag scripts: %s", ' '.join(list_refinediags) )
+ logging.info( "NOTE: Now turned off by default; please re-enable in config file if needed" )
+ else:
+ rose_suite.set(keys=['template variables', 'DO_REFINEDIAG'], value='False')
+ logging.info("No refineDiag scripts written. ")
+
+ if str_preanalysis is not None:
+ rose_suite.set(keys=['template variables', 'DO_PREANALYSIS'], value='True')
+ rose_suite.set(keys=['template variables', 'PREANALYSIS_SCRIPT'],
+ value=f"'{str_preanalysis}'" )
+ logging.info( "Preanalysis script: %s", str_preanalysis )
+ else:
+ rose_suite.set(keys=['template variables', 'DO_PREANALYSIS'], value='False')
+ logging.info( "No preAnalysis scripts written." )
+
+ # Grab all of the necessary PP component items/elements from the XML
+ comps = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ 'postProcess/component/@type').split()
+ rose_suite.set(keys=['template variables', 'PP_COMPONENTS'],
+ value=f"'{' '.join(sorted(comps))}'" )
+
+ segment_time = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ 'runtime/production/segment/@simTime')
+ segment_units = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ 'runtime/production/segment/@units')
+
+ if segment_units == 'years':
+ segment = f'P{segment_time}Y'
+ elif segment_units == 'months':
+ segment = f'P{segment_time}M'
+ else:
+ logging.error("Unknown segment units: %s", segment_units)
+ raise ValueError
+
+ # P12M is identical to P1Y but the latter looks nicer
+ if segment == 'P12M':
+ segment = 'P1Y'
+ rose_suite.set(keys=['template variables', 'HISTORY_SEGMENT'], value=f"'{segment}'" )
+
+ # Get the namelist current_date as the likely PP_START (unless "start" is used in the PP tags)
+ # frelist --namelist may be better, but sometimes may not work
+ current_date_str = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ 'input/namelist')
+ match = re.search(r'current_date\s*=\s*(\d+),(\d+),(\d+)', current_date_str)
+ if match:
+ try:
+ current_date = metomi.isodatetime.data.TimePoint(
+ year=match.group(1), month_of_year=match.group(2), day_of_month=match.group(3) )
+ except:
+ logging.warning("Could not parse date from namelist current_date")
+ current_date = None
+ else:
+ current_date = None
+ logging.warning("Could not find current_date in namelists")
+ logging.info("current_date (from namelists): %s", current_date)
+
+ # Take a good guess for the PP_START and PP_STOP
+ # PP_START could be the coupler_nml/current_date
+ # PP_STOP could be the PP_START plus the simulation length
+ if sim_units == "years":
+ oneless = int(sim_time) - 1
+ duration = f"P{oneless}Y"
+ elif sim_units == "months":
+ duration = f"P{sim_time}M"
+ else:
+ raise ValueError(f"Was hoping sim_units would be years or months; got {sim_units}")
+ dur = metomi.isodatetime.parsers.DurationParser().parse(duration)
+ pp_stop = current_date + dur
+ rose_suite.set(keys=['template variables', 'PP_START'], value=f'"{current_date}"')
+ rose_suite.set(keys=['template variables', 'PP_STOP'], value=f'"{pp_stop}"')
+
+ # Loop over all of the PP components, fetching the sources, xyInterp,
+ # and sourceGrid.
+ chunks = set()
+ comp_count = 0
+ for comp in comps:
+ comp_count += 1
+ pp_comp_xpath_header = f'postProcess/component[@type="{comp}"]'
+ logging.info( "Component loop: %s out of %s", comp_count, len(comps))
+
+ # get the comp attributes
+ comp_source = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ f'{pp_comp_xpath_header}/@source' )
+ xy_interp = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ f'{pp_comp_xpath_header}/@xyInterp' )
+ source_grid = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ f'{pp_comp_xpath_header}/@sourceGrid' )
+ interp_method = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ f'{pp_comp_xpath_header}/@interpMethod')
+
+ # split some of the stuffs
+ if xy_interp != "":
+ interp_split = xy_interp.split(',')
+ output_grid_lon = interp_split[1]
+ output_grid_lat = interp_split[0]
+ if source_grid != "":
+ sourcegrid_split = source_grid.split('-')
+ input_grid = sourcegrid_split[1]
+ input_realm = sourcegrid_split[0]
+
+ # determine the interp method
+ if xy_interp:
+ if interp_method == "":
+ if input_grid == "cubedsphere":
+ interp_method = 'conserve_order2'
+ elif input_grid == 'tripolar':
+ interp_method = 'conserve_order1'
+ else:
+ raise Exception(f"Expected cubedsphere or tripolar, not {source_grid}")
+
+ # determine the grid label
+ if xy_interp:
+ grid = f"regrid-xy/{output_grid_lon}_{output_grid_lat}.{interp_method}"
+ grid_tail = f"{output_grid_lon}_{output_grid_lat}.{interp_method}"
+ else:
+ grid = "native"
+
+ sources = set()
+ if comp_source.endswith('_refined'):
+ logging.info(
+ "NOTE: Skipping history file %s, refineDiag is turned off by default.", comp_source)
+ else:
+ sources.add(comp_source)
+ timeseries_count = 0
+
+ # Get the number of TS nodes
+ results = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ f'{pp_comp_xpath_header}/timeSeries/@freq' ).split()
+ timeseries_count = len(results)
+
+ # Loop over the TS nodes and write out the frequency, chunklength, and
+ # grid to the remap-pp-components Rose app configuration
+ for i in range(1, timeseries_count + 1):
+ #label = "{}.{}".format(comp, str(i)) #notyetimplemented
+
+ source = frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ f'{pp_comp_xpath_header}/timeSeries[{i}]/@source' )
+ if source.endswith('_refined'):
+ logging.info(
+ "NOTE: Skipping history file %s, refineDiag is turned off by default.", source)
+ else:
+ sources.add(source)
+
+ #freq = freq_from_legacy(frelist_xpath(args,
+ # '{}/timeSeries[{}]/@freq' \
+ # .format(pp_comp_xpath_header, i)))
+ chunk = chunk_from_legacy(
+ frelist_xpath(xml, platform, target, experiment,
+ #do_analysis,
+ #historydir, refinedir, ppdir,
+ #do_refinediag, pp_start, pp_stop, validate,
+ #verbose, quiet, dual,
+ f'{pp_comp_xpath_header}/timeSeries[{i}]/@chunkLength'
+ ) )
+ chunks.add(chunk)
+ #rose_remap.set(keys=[label, 'freq'], value=freq)
+ #rose_remap.set(keys=[label, 'chunk'], value=chunk)
+
+ rose_remap.set(keys=[comp, 'sources'], value=' '.join(sources))
+ rose_remap.set(keys=[comp, 'grid'], value=grid)
+
+ if grid == "native":
+ pass
+ else:
+ # Write out values to the 'regrid-xy' Rose app
+ rose_regrid_xy.set(keys=[comp, 'sources'], value=' '.join(sources))
+ rose_regrid_xy.set(keys=[comp, 'inputGrid'], value=input_grid)
+ rose_regrid_xy.set(keys=[comp, 'inputRealm'], value=input_realm)
+ rose_regrid_xy.set(keys=[comp, 'interpMethod'], value=interp_method)
+ rose_regrid_xy.set(keys=[comp, 'outputGridType'], value=grid_tail)
+ rose_regrid_xy.set(keys=[comp, 'outputGridLon'], value=output_grid_lon)
+ rose_regrid_xy.set(keys=[comp, 'outputGridLat'], value=output_grid_lat)
+
+ # Process all of the found PP chunks into the rose-suite configuration
+ if verbose:
+ print("")
+ logging.info("Setting PP chunks...")
+
+ sorted_chunks = list(chunks)
+ sorted_chunks.sort(key=duration_to_seconds, reverse=False)
+
+ if len(chunks) == 0:
+ raise ValueError('no chunks found! exit.')
+
+ logging.info(" Chunks found: %s", ', '.join(sorted_chunks))
+ if len(chunks) == 1:
+ rose_suite.set(['template variables', 'PP_CHUNK_A'],
+ f"'{sorted_chunks[0]}'")
+ else:
+ rose_suite.set(['template variables', 'PP_CHUNK_A'],
+ f"'{sorted_chunks[0]}'")
+ rose_suite.set(['template variables', 'PP_CHUNK_B'],
+ f"'{sorted_chunks[1]}'")
+ logging.info(" Chunks used: %s", ', '.join(sorted_chunks[0:2]) )
+
+ # Write out the final configurations.
+ if verbose:
+ print("")
+ logging.info("Writing output files...")
+
+ dumper = metomi.rose.config.ConfigDumper()
+ rose_outfiles=[ ( rose_suite, 'rose-suite.conf' ),
+ ( rose_remap, 'app/remap-pp-components/rose-app.conf' ),
+ ( rose_regrid_xy, 'app/regrid-xy/rose-app.conf' ) ]
+ for outfile in rose_outfiles:
+ logging.info(" %s", outfile[1])
+ dumper(outfile[0], outfile[1])
+
+ #outfile = "app/remap-pp-components/rose-app.conf"
+ #logging.info(" %s", outfile)
+ #dumper(rose_remap, outfile)
+ #
+ #outfile = "app/regrid-xy/rose-app.conf"
+ #logging.info(" %s", outfile)
+ #dumper(rose_regrid_xy, outfile)
+
+def _convert(xml, platform, target, experiment, do_analysis=False, historydir=None,
+ refinedir=None, ppdir=None, do_refinediag=False, pp_start=None,
+ pp_stop=None, validate=False, verbose=False, quiet=False, dual=False):
+ """
+ Converts a Bronx XML to a Canopy rose-suite.conf
+ """
+
+ # Logging settings. The default is throwing only warning messages
+ if verbose:
+ logging.basicConfig(level=logging.INFO, format=LOGGING_FORMAT)
+ elif quiet:
+ logging.basicConfig(level=logging.ERROR, format=LOGGING_FORMAT)
+ else:
+ logging.basicConfig(level=logging.WARNING, format=LOGGING_FORMAT)
+
+ # Set the name of the directory
+ name = f"{experiment}__{platform}__{target}"
+
+ # Create the directory if it doesn't exist
+ cylc_dir = os.path.expanduser("~/cylc-src")
+ new_dir = os.path.join(cylc_dir, name)
+ os.makedirs(new_dir, exist_ok=True)
+
+ # Change the current working directory
+ xml = os.path.abspath(xml)
+ os.chdir(new_dir)
+
+ ##########################################################################
+ # NOTE: Ideally we would check that "frelist" is in the PATH here,
+ # but in practice the frelist will just fail later if it's not loaded.
+ # If that error scenario is too confusing, let's add a check back here.
+ cylc_loaded = False
+ if validate:
+ if CYLC_PATH in os.getenv('PATH'):
+ cylc_loaded = True
+ else:
+ raise EnvironmentError("Cannot run the validator tool because " \
+ "the Cylc module isn't loaded. Please " \
+ "run 'module load cylc/test' and try again.")
+
+ # Alert the user if only 1 or zero PP years are given as an option, and
+ # notify them that a default of '0000' for those years will be set in the
+ # rose-suite configuration
+ if any( [ pp_stop is None, pp_start is None ] ):
+ if pp_start is None:
+ logging.warning("PP start year was not specified.")
+ if pp_stop is None:
+ logging.warning("PP stop year was not specified.")
+ logging.warning("After the converter has run, please edit the " \
+ "default '0000' values within your rose-suite.conf.")
+
+ # These series of conditionals takes into account input from the user
+ # (for the PP_START and PP_STOP year) that is not 4 digits or other
+ # nonsensical years. The rose-suite config requires 4 digits for years
+ # and if the year is under '1000' (but > 0), then leading zeros must be used.
+ if all( [ pp_start is not None, pp_stop is not None ] ):
+ def format_req_pp_year(pp_year):
+ if len(pp_year) < 4 and int(pp_year) > 0:
+ pp_year = '0' * (4 - len(pp_year)) + pp_year
+ return pp_year
+ pp_start = format_req_pp_year(pp_start)
+ pp_stop = format_req_pp_year(pp_stop)
+
+ if int(pp_start) >= int(pp_stop):
+ logging.warning("Your PP_START date is equal to or later than " \
+ "your PP_STOP date. Please revise these values in " \
+ "your configuration after the converter has run.")
+ if any( [ len(pp_start) > 4, len(pp_stop) > 4,
+ int(pp_start) <= 0, int(pp_stop) <= 0 ] ):
+ logging.warning("At least one of your PP_start or PP_stop years " \
+ "does not make sense. Please revise this value in " \
+ "your configuration after the converter has run.")
+
+ main( xml, platform, target, experiment, do_analysis, historydir, refinedir, ppdir,
+ do_refinediag, pp_start, pp_stop, validate, verbose, quiet, dual )
+
+ if verbose:
+ print("")
+ logging.info("XML conversion complete!")
+
+ # Run the Cylc validator tool on the current directory if conditions are met.
+ # Note: the user must be running the converter in the parent Cylc Workflow
+ # Directory if the validator is run.
+ if cylc_loaded:
+ if verbose:
+ print("")
+ logging.info("Running the Cylc validator tool...")
+ try:
+ subprocess.run("cylc validate .", shell=True, check=True)
+ except subprocess.CalledProcessError:
+ logging.error("Errant values in rose-suite.conf or other Cylc errors. " \
+ "Please check your configuration and run the validator " \
+ "again separately.")
+ finally:
+ logging.info("Validation step complete!")
+
+##############################################
+
+@click.command()
+def convert():
+ '''
+ Wrapper for convert call - allows users to call without command-line args
+ '''
+ _convert()
+
+if __name__ == '__main__':
+ convert()
diff --git a/lib/fre/pp/configure_script_yaml.py b/lib/fre/pp/configure_script_yaml.py
new file mode 100644
index 00000000..383d5acf
--- /dev/null
+++ b/lib/fre/pp/configure_script_yaml.py
@@ -0,0 +1,211 @@
+"""
+ Script creates rose-apps and rose-suite
+ files for the workflow from the pp yaml.
+"""
+
+## TO-DO:
+# - condition where there are multiple pp yamls
+# - validation here or in combined-yamls tools
+
+import os
+import json
+import shutil
+import click
+
+from jsonschema import validate
+import yaml
+import metomi.rose.config
+
+# Relative import
+#f = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
+#sys.path.append(f)
+import fre.yamltools.combine_yamls as cy
+
+####################
+def yaml_load(yamlfile):
+ """
+ Load the given yaml and validate.
+ """
+ # Load the main yaml and validate
+ with open(yamlfile,'r') as f:
+ y=yaml.safe_load(f)
+
+ return y
+
+######VALIDATE#####
+def validate_yaml(yamlfile):
+ """
+ Using the schema.json file, the yaml format is validated.
+ """
+ # Load the yaml
+ yml = yaml_load(yamlfile)
+
+ package_dir = os.path.dirname(os.path.abspath(__file__))
+ schema_path = os.path.join(package_dir, 'schema.json')
+ # Load the json schema: .load() (vs .loads()) reads and parses the json in one
+ with open(schema_path,'r') as s:
+ schema = json.load(s)
+
+ # Validate yaml
+ # If the yaml is not valid, the schema validation will raise errors and exit
+ if validate(instance=yml,schema=schema) is None:
+ print("COMBINED YAML VALID \n")
+
+####################
+def rose_init(experiment,platform,target):
+ """
+ Initialize the rose suite and app configurations.
+ """
+ # initialize rose suite config
+ rose_suite = metomi.rose.config.ConfigNode()
+ # disagreeable; these should be optional
+ rose_suite.set(keys=['template variables', 'DO_ANALYSIS_ONLY'], value='False')
+ rose_suite.set(keys=['template variables', 'DO_MDTF'], value='False')
+ rose_suite.set(keys=['template variables', 'PP_DEFAULT_XYINTERP'], value='0,0')
+
+ # set some rose suite vars
+ rose_suite.set(keys=['template variables', 'EXPERIMENT'], value=f'"{experiment}"')
+ rose_suite.set(keys=['template variables', 'PLATFORM'], value=f'"{platform}"')
+ rose_suite.set(keys=['template variables', 'TARGET'], value=f'"{target}"')
+
+ # initialize rose regrid config
+ rose_regrid = metomi.rose.config.ConfigNode()
+ rose_regrid.set(keys=['command', 'default'], value='regrid-xy')
+
+ # initialize rose remap config
+ rose_remap = metomi.rose.config.ConfigNode()
+ rose_remap.set(keys=['command', 'default'], value='remap-pp-components')
+
+ return(rose_suite,rose_regrid,rose_remap)
+
+####################
+def quote_rose_values(value):
+ """
+ rose-suite.conf template variables must be quoted unless they are
+ boolean, in which case do not quote them.
+ """
+ if isinstance(value, bool):
+ return f"{value}"
+ else:
+ return "'" + value + "'"
+
+####################
+def set_rose_suite(yamlfile,rose_suite):
+ """
+ Set items in the rose suite configuration.
+ """
+ pp=yamlfile.get("postprocess")
+ dirs=yamlfile.get("directories")
+
+ # set rose-suite items
+ if pp is not None:
+ for i in pp.values():
+ if not isinstance(i,list):
+ for key,value in i.items():
+ # rose-suite.conf is somewhat finicky with quoting
+ # cylc validate will reveal any complaints
+ rose_suite.set( keys = ['template variables', key.upper()],
+ value = quote_rose_values(value) )
+ if dirs is not None:
+ for key,value in dirs.items():
+ rose_suite.set(keys=['template variables', key.upper()], value=quote_rose_values(value))
+
+####################
+def set_rose_apps(yamlfile,rose_regrid,rose_remap):
+ """
+ Set items in the regrid and remap rose app configurations.
+ """
+ components = yamlfile.get("postprocess").get("components")
+ for i in components:
+ comp = i.get('type')
+ sources = i.get('sources')
+ interp_method = i.get('interpMethod')
+
+ # set remap items
+ rose_remap.set(keys=[f'{comp}', 'sources'], value=f'{sources}')
+ # if xyInterp doesnt exist, grid is native
+ if i.get("xyInterp") is None:
+ rose_remap.set(keys=[f'{comp}', 'grid'], value='native')
+
+ # if xyInterp exists, component can be regridded
+ else:
+ interp_split = i.get('xyInterp').split(',')
+ rose_remap.set(keys=[f'{comp}', 'grid'],
+ value=f'regrid-xy/{interp_split[0]}_{interp_split[1]}.{interp_method}')
+
+ # set regrid items
+ if i.get("xyInterp") is not None:
+ rose_regrid.set(keys=[f'{comp}', 'sources'], value=f'{sources}')
+ rose_regrid.set(keys=[f'{comp}', 'inputRealm'], value=f'{i.get("inputRealm")}')
+ rose_regrid.set(keys=[f'{comp}', 'inputGrid'], value=f'{i.get("sourceGrid")}')
+ rose_regrid.set(keys=[f'{comp}', 'interpMethod'], value=f'{interp_method}')
+ interp_split = i.get('xyInterp').split(',')
+ rose_regrid.set(keys=[f'{comp}', 'outputGridLon'], value=f'{interp_split[1]}')
+ rose_regrid.set(keys=[f'{comp}', 'outputGridLat'], value=f'{interp_split[0]}')
+ rose_regrid.set(keys=[f'{comp}', 'outputGridType'],
+ value=f'{interp_split[0]}_{interp_split[1]}.{interp_method}')
+
+####################
+def yamlInfo(yamlfile,experiment,platform,target):
+ """
+ Using a valid pp.yaml, the rose-app and rose-suite
+ configuration files are created in the cylc-src
+ directory. The pp.yaml is also copied to the
+ cylc-src directory.
+ """
+ e = experiment
+ p = platform
+ t = target
+ yml = yamlfile
+
+ # Initialize the rose configurations
+ rose_suite,rose_regrid,rose_remap = rose_init(e,p,t)
+
+ # Combine model, experiment, and analysis yamls
+ comb = cy.init_pp_yaml(yml,e,p,t)
+ full_combined = cy.get_combined_ppyaml(comb)
+
+ # Validate yaml
+ validate_yaml(full_combined)
+
+ # Load the combined yaml
+ comb_pp_yaml = yaml_load(full_combined)
+
+ ## PARSE COMBINED YAML TO CREATE CONFIGS
+ # Set rose-suite items
+ set_rose_suite(comb_pp_yaml,rose_suite)
+
+ # Set regrid and remap rose app items
+ set_rose_apps(comb_pp_yaml,rose_regrid,rose_remap)
+
+ # write output files
+ print("Writing output files...")
+ cylc_dir = os.path.join(os.path.expanduser("~/cylc-src"), f"{e}__{p}__{t}")
+ outfile = os.path.join(cylc_dir, f"{e}.yaml")
+ shutil.copyfile(full_combined, outfile)
+ print(" " + outfile)
+
+ dumper = metomi.rose.config.ConfigDumper()
+ outfile = os.path.join(cylc_dir, "rose-suite.conf")
+ dumper(rose_suite, outfile)
+ print(" " + outfile)
+
+ outfile = os.path.join(cylc_dir, "app", "regrid-xy", "rose-app.conf")
+ dumper(rose_regrid, outfile)
+ print(" " + outfile)
+
+ outfile = os.path.join(cylc_dir, "app", "remap-pp-components", "rose-app.conf")
+ dumper(rose_remap, outfile)
+ print(" " + outfile)
+
+@click.command()
+def _yamlInfo(yamlfile,experiment,platform,target):
+ '''
+ Wrapper script for calling yamlInfo - allows the decorated version
+ of the function to be separate from the undecorated version
+ '''
+ return yamlInfo(yamlfile,experiment,platform,target)
+
+# Use parseyaml function to parse created edits.yaml
+if __name__ == '__main__':
+ yamlInfo()
diff --git a/lib/fre/pp/edits-template.yaml b/lib/fre/pp/edits-template.yaml
new file mode 100644
index 00000000..9bedfe71
--- /dev/null
+++ b/lib/fre/pp/edits-template.yaml
@@ -0,0 +1,271 @@
+## User defined edits
+
+# Paths
+configuration paths:
+ # Path to where rose-suite experiment configuration should be in pp repo
+ rose-suite: "opt/"
+ # Path to remap-pp-components rose-app.conf in pp repo
+ rose-remap: "app/remap-pp-components/rose-app.conf"
+ # Path to regrid rose-app.conf in pp repo
+ rose-regrid: "app/regrid-xy/rose-app.conf"
+
+# Define widely used variables
+# Length of analysis of experiment, string in double quotes
+ana_amip_len: &ANA_AMIP_LEN ""
+# String in double quotes
+pp_amip_chunk: &PP_AMIP_CHUNK96 ""
+# Year to start analysis on experiment, string in double quotes
+ana_amip_start: &ANA_AMIP_START ""
+# Year to stop analysis on experiment, string in double quotes
+ana_amip_stop: &ANA_AMIP_STOP ""
+# Default xy Interp, string in double quotes
+defaultyInterp: &defaultxyInterp ""
+
+# Information for rose-suite experiment configuration
+rose-suite:
+ settings:
+ # ISO8601 duration of the length of the history segment
+ history_segment:
+ # ISO8601 date to begin postprocessing
+ pp_start:
+ # ISO8601 date to stop postprocessing
+ pp_stop:
+ # Length of experiment
+ len:
+ # Specific site used for workflow; must be 'generic', 'gfdl-ws' or 'ppan'
+ site:
+ # Location for shared analysis scripts
+ fre_analysis_home:
+ # ISO8601 duration of the desired post-processed output
+ pp_chunk_a:
+ # ISO8601 duration of a second desired postprocessed output
+ pp_chunk_b:
+ # Components to be post-processed
+ pp_components:
+ # Default regridded resolution
+ pp_default_xyinterp: *defaultxyInterp
+ # Filepath to an FMS gridSpec netCDF file or a tarfile
+ pp_grid_spec:
+
+ switches:
+ # Switch to remove intermediate data files when they are no longer needed
+ clean_work:
+ # Switch to run MDTF on generated pp output
+ do_mdtf:
+ # Switch to turn on/off statics processing
+ do_statics:
+ # Switch to turn on/off time-average file generation
+ do_timeavgs:
+ # Switch to run refine-diag script(s) on history file to generate additional diagnostics
+ do_refinediag:
+ # Switch to mask atmos pressure-level output above/below surface pressure/atmos top
+ do_atmos_plevel_masking:
+ # Switch to run a pre-analysis script on history files
+ do_preanalysis:
+ do_analysis:
+ do_analysis_only:
+
+ directories:
+ # Directory to the primary location for history files
+ history_dir:
+ # Directory to write pp products to
+ pp_dir:
+ # Directory to use for history file cache
+ ptmp_dir:
+ # Filepath to refineDiag scripts
+ refinediag_scripts:
+ # Filepath to the user script
+ preanalysis_script:
+ # Directory to the secondary location for history files
+ history_dir_refined:
+ # Directory to write analysis output
+ analysis:
+
+# Information needed to populate the rose-apps for regrid-xy and remap-pp-components
+# These rose apps will be retired eventually
+components:
+ # Name of component
+ - type:
+ # Analysis start
+ start: *ANA_AMIP_START
+ sources:
+ sourceGrid:
+ # Default regridded grid
+ xyInterp: *defaultxyInterp
+ # Should be 'conserve_order1', 'conserve_order2' or 'bilinear'
+ interpMethod:
+ timeSeries:
+ - freq:
+ source:
+ chunkLength: *PP_AMIP_CHUNK96
+ - freq:
+ source:
+ chunkLength: *PP_AMIP_CHUNK96
+ variables:
+ - freq:
+ source:
+ chunkLength: *PP_AMIP_CHUNK96
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ timeAverage:
+ - source:
+ interval: *PP_AMIP_CHUNK96
+ - source:
+ interval: *ANA_AMIP_LEN
+ - source:
+ interval: *PP_AMIP_CHUNK96
+ calcInterval: *PP_AMIP_CHUNK96
+ - type:
+ zInterp:
+ start: *ANA_AMIP_START
+ sources:
+ sourceGrid:
+ xyInterp: *defaultxyInterp
+ interpMethod:
+ cmip:
+ timeSeries:
+ - freq:
+ source:
+ chunkLength: *PP_AMIP_CHUNK96
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ timeAverage:
+ - source:
+ interval: *PP_AMIP_CHUNK96
+ - source:
+ interval: *ANA_AMIP_LEN
+ - source:
+ interval: *PP_AMIP_CHUNK96
+ calcInterval: *PP_AMIP_CHUNK96
+ - source:
+ interval: *ANA_AMIP_LEN
+ - type:
+ start: *ANA_AMIP_START
+ sources:
+ sourceGrid:
+ xyInterp: *defaultxyInterp
+ interpMethod:
+ cmip:
+ timeSeries:
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ - type:
+ start: *ANA_AMIP_START
+ sources:
+ sourceGrid:
+ xyInterp: *defaultxyInterp
+ interpMethod:
+ cmip:
+ timeSeries:
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ timeAverage:
+ - source:
+ interval: *PP_AMIP_CHUNK96
+ - source:
+ interval: *ANA_AMIP_LEN
+ - type:
+ start: *ANA_AMIP_START
+ sources:
+ sourceGrid:
+ xyInterp: *defaultxyInterp
+ interpMethod:
+ cmip:
+ timeAverage:
+ - source:
+ interval: *PP_AMIP_CHUNK96
+ - source:
+ interval: *ANA_AMIP_LEN
+ timeSeries:
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ - type:
+ start: *ANA_AMIP_START
+ sources:
+ sourceGrid:
+ xyInterp: *defaultxyInterp
+ interpMethod:
+ timeSeries:
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ - type:
+ start: *ANA_AMIP_START
+ sources:
+ xyInterp:
+ interpMethod:
+ cmip:
+ timeSeries:
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ - freq:
+ source:
+ chunkLength: *PP_AMIP_CHUNK96
+ - type:
+ start: *ANA_AMIP_START
+ sources:
+ sourceGrid:
+ xyInterp: *defaultxyInterp
+ interpMethod:
+ cmip:
+ timeSeries:
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ timeAverage:
+ - source:
+ interval: *PP_AMIP_CHUNK96
+ - source:
+ interval: *ANA_AMIP_LEN
+ - type:
+ start: *ANA_AMIP_START
+ xyInterp: *defaultxyInterp
+ sources:
+ sourceGrid:
+ interpMethod:
+ cmip:
+ timeSeries:
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ source:
+ - type:
+ start: *ANA_AMIP_START
+ sources:
+ sourceGrid:
+ xyInterp: *defaultxyInterp
+ interpMethod:
+ cmip:
+ timeSeries:
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+ timeAverage:
+ - source:
+ interval: *PP_AMIP_CHUNK96
+ - source:
+ interval: *ANA_AMIP_LEN
+ - type:
+ start: *ANA_AMIP_START
+ sources:
+ sourceGrid:
+ xyInterp: *defaultxyInterp
+ interpMethod:
+ cmip:
+ timeSeries:
+ - freq:
+ chunkLength: *PP_AMIP_CHUNK96
+
+# TMP DIRECTORY CREATION
+# This edit is not needed when working on pp/an. If not working on pp/an, a tmpDir must be defined.
+tmpdir:
+ tmpdirpath:
+
+# Optional edit: Add --symlink-dirs option to install cylc-run directory to location with more space available; default installation location for cylc-run directory is $HOME;
+# Utilize this edit if you do not have enough space in home directory
+install-exp-script:
+ - path:
+ - install-option:
+ install:
diff --git a/lib/fre/pp/frepp.py b/lib/fre/pp/frepp.py
new file mode 100644
index 00000000..d21c5d23
--- /dev/null
+++ b/lib/fre/pp/frepp.py
@@ -0,0 +1,231 @@
+''' fre pp '''
+
+import click
+
+from fre.pp import checkout_script
+from fre.pp import configure_script_yaml
+from fre.pp import configure_script_xml
+#from fre.pp import validate
+from fre.pp import install
+from fre.pp import run
+from fre.pp import trigger
+from fre.pp import status
+from fre.pp import wrapper
+
+@click.group(help=click.style(" - access fre pp subcommands", fg=(57,139,210)))
+def pp_cli():
+ ''' entry point to fre pp click commands '''
+
+
+# fre pp status
+@pp_cli.command()
+@click.option("-e", "--experiment", type=str,
+ help="Experiment name",
+ required=True)
+@click.option("-p", "--platform", type=str,
+ help="Platform name",
+ required=True)
+@click.option("-t", "--target", type=str,
+ help="Target name",
+ required=True)
+@click.pass_context
+def status(context, experiment, platform, target):
+ # pylint: disable=unused-argument
+ """ - Report status of PP configuration"""
+ context.forward(status.status_subtool)
+
+# fre pp run
+@pp_cli.command()
+@click.option("-e", "--experiment", type=str,
+ help="Experiment name",
+ required=True)
+@click.option("-p", "--platform", type=str,
+ help="Platform name",
+ required=True)
+@click.option("-t", "--target", type=str,
+ help="Target name",
+ required=True)
+@click.pass_context
+def run(context, experiment, platform, target):
+ # pylint: disable=unused-argument
+ """ - Run PP configuration"""
+ context.forward(run.pp_run_subtool)
+
+# fre pp validate
+@pp_cli.command()
+@click.option("-e", "--experiment", type=str,
+ help="Experiment name",
+ required=True)
+@click.option("-p", "--platform", type=str,
+ help="Platform name",
+ required=True)
+@click.option("-t", "--target", type=str,
+ help="Target name",
+ required=True)
+@click.pass_context
+def validate(context, experiment, platform, target):
+ # pylint: disable=unused-argument
+ """ - Validate PP configuration"""
+ context.forward(_validate_subtool)
+
+# fre pp install
+@pp_cli.command()
+@click.option("-e", "--experiment", type=str,
+ help="Experiment name",
+ required=True)
+@click.option("-p", "--platform", type=str,
+ help="Platform name",
+ required=True)
+@click.option("-t", "--target", type=str,
+ help="Target name",
+ required=True)
+@click.pass_context
+def install(context, experiment, platform, target):
+ # pylint: disable=unused-argument
+ """ - Install PP configuration"""
+ context.forward(install.install_subtool)
+
+@pp_cli.command()
+@click.option("-y", "--yamlfile", type=str,
+ help="YAML file to be used for parsing",
+ required=True)
+@click.option("-e", "--experiment", type=str,
+ help="Experiment name",
+ required=True)
+@click.option("-p", "--platform", type=str,
+ help="Platform name",
+ required=True)
+@click.option("-t", "--target", type=str,
+ help="Target name",
+ required=True)
+@click.pass_context
+def configure_yaml(context,yamlfile,experiment,platform,target):
+ # pylint: disable=unused-argument
+ """ - Execute fre pp configure """
+ context.forward(configure_script_yaml._yamlInfo)
+
+@pp_cli.command()
+@click.option("-e", "--experiment", type=str,
+ help="Experiment name",
+ required=True)
+@click.option("-p", "--platform", type=str,
+ help="Platform name",
+ required=True)
+@click.option("-t", "--target", type=str,
+ help="Target name",
+ required=True)
+@click.option("-b", "--branch",
+ required=False,
+ help="fre-workflows branch/tag to clone; default is $(fre --version)")
+@click.pass_context
+def checkout(context, experiment, platform, target, branch=None):
+ # pylint: disable=unused-argument
+ """ - Execute fre pp checkout """
+ context.forward(checkout_script._checkout_template)
+
+@pp_cli.command()
+@click.option('-x', '--xml',
+ required=True,
+ help="Required. The Bronx XML")
+@click.option('-p', '--platform',
+ required=True,
+ help="Required. The Bronx XML Platform")
+@click.option('-t', '--target',
+ required=True,
+ help="Required. The Bronx XML Target")
+@click.option('-e', '--experiment',
+ required=True,
+ help="Required. The Bronx XML Experiment")
+@click.option('--do_analysis',
+ is_flag=True,
+ default=False,
+ help="Optional. Runs the analysis scripts.")
+@click.option('--historydir',
+ help="Optional. History directory to reference. " \
+ "If not specified, the XML's default will be used.")
+@click.option('--refinedir',
+ help="Optional. History refineDiag directory to reference. " \
+ "If not specified, the XML's default will be used.")
+@click.option('--ppdir',
+ help="Optional. Postprocessing directory to reference. " \
+ "If not specified, the XML's default will be used.")
+@click.option('--do_refinediag',
+ is_flag=True,
+ default=False,
+ help="Optional. Process refineDiag scripts")
+@click.option('--pp_start',
+ help="Optional. Starting year of postprocessing. " \
+ "If not specified, a default value of '0000' " \
+ "will be set and must be changed in rose-suite.conf")
+@click.option('--pp_stop',
+ help="Optional. Ending year of postprocessing. " \
+ "If not specified, a default value of '0000' " \
+ "will be set and must be changed in rose-suite.conf")
+@click.option('--validate',
+ is_flag=True,
+ help="Optional. Run the Cylc validator " \
+ "immediately after conversion")
+@click.option('-v', '--verbose',
+ is_flag=True,
+ help="Optional. Display detailed output")
+@click.option('-q', '--quiet',
+ is_flag=True,
+ help="Optional. Display only serious messages and/or errors")
+@click.option('--dual',
+ is_flag=True,
+ help="Optional. Append '_canopy' to pp, analysis, and refinediag dirs")
+@click.pass_context
+def configure_xml(context, xml, platform, target, experiment, do_analysis, historydir, refinedir,
+ ppdir, do_refinediag, pp_start, pp_stop, validate, verbose, quiet, dual):
+ # pylint: disable=unused-argument
+ """ - Converts a Bronx XML to a Canopy rose-suite.conf """
+ context.forward(configure_script_xml.convert)
+
+#fre pp wrapper
+@pp_cli.command()
+@click.option("-e", "--experiment", type=str,
+ help="Experiment name",
+ required=True)
+@click.option("-p", "--platform", type=str,
+ help="Platform name",
+ required=True)
+@click.option("-T", "--target", type=str,
+ help="Target name",
+ required=True)
+@click.option("-c", "--config-file", type=str,
+ help="Path to a configuration file in either XML or YAML",
+ required=True)
+@click.option("-t", "--time",
+ required=False,
+ help="Time whose history files are ready")
+@click.option("-b", "--branch",
+ required=False,
+ help="fre-workflows branch/tag to clone; default is $(fre --version)")
+@click.pass_context
+def wrapper(context, experiment, platform, target, config_file, time=None, branch=None):
+ # pylint: disable=unused-argument
+ """ - Execute fre pp steps in order """
+ context.forward(wrapper.runFre2pp)
+
+@pp_cli.command()
+@click.option("-e", "--experiment", type=str,
+ help="Experiment name",
+ required=True)
+@click.option("-p", "--platform", type=str,
+ help="Platform name",
+ required=True)
+@click.option("-T", "--target", type=str,
+ help="Target name",
+ required=True)
+@click.option("-t", "--time",
+ required=True,
+ help="Time whose history files are ready")
+@click.pass_context
+def trigger(context, experiment, platform, target, time):
+ # pylint: disable=unused-argument
+ """ - Start postprocessing for a particular time """
+ context.forward(trigger._trigger)
+
+if __name__ == "__main__":
+ ''' entry point for click to fre pp commands '''
+ pp_cli()
diff --git a/lib/fre/pp/install.py b/lib/fre/pp/install.py
new file mode 100644
index 00000000..cb2ed021
--- /dev/null
+++ b/lib/fre/pp/install.py
@@ -0,0 +1,41 @@
+''' fre pp install '''
+
+from pathlib import Path
+import os
+import subprocess
+import click
+
+def install_subtool(experiment, platform, target):
+ """
+ Install the Cylc workflow definition located in
+ ~/cylc-src/____
+ to
+ ~/cylc-run/____
+ """
+
+ name = experiment + '__' + platform + '__' + target
+ # if the cylc-run directory already exists,
+ # then check whether the cylc expanded definition (cylc config)
+ # is identical. If the same, good. If not, bad.
+ source_dir = Path(os.path.expanduser("~/cylc-src"), name)
+ install_dir = Path(os.path.expanduser("~/cylc-run"), name)
+ if os.path.isdir(install_dir):
+ installed_def = subprocess.run(["cylc", "config", name],capture_output=True).stdout
+ go_back_here = os.getcwd()
+ os.chdir(source_dir)
+ source_def = subprocess.run(['cylc', 'config', '.'], capture_output=True).stdout
+ if installed_def == source_def:
+ print(f"NOTE: Workflow '{install_dir}' already installed, and the definition is unchanged")
+ else:
+ print(f"ERROR: Workflow '{install_dir}' already installed, and the definition has changed!")
+ print(f"ERROR: Please remove installed workflow with 'cylc clean {name}' or move the workflow run directory '{install_dir}'")
+ exit(1)
+ else:
+ print(f"NOTE: About to install workflow into ~/cylc-run/{name}")
+ cmd = f"cylc install --no-run-name {name}"
+ subprocess.run(cmd, shell=True, check=True)
+
+@click.command()
+def _install_subtool(experiment, platform, target):
+ ''' entry point to install for click '''
+ return _install_subtool(experiment, platform, target)
diff --git a/lib/fre/pp/run.py b/lib/fre/pp/run.py
new file mode 100644
index 00000000..d50c9f01
--- /dev/null
+++ b/lib/fre/pp/run.py
@@ -0,0 +1,19 @@
+''' fre pp run '''
+
+import subprocess
+import click
+
+def pp_run_subtool(experiment, platform, target):
+ """
+ Start or restart the Cylc workflow identified by:
+ ____
+ """
+
+ name = experiment + '__' + platform + '__' + target
+ cmd = f"cylc play {name}"
+ subprocess.run(cmd, shell=True, check=True)
+
+@click.command()
+def _pp_run_subtool(experiment, platform, target):
+ ''' entry point to run for click '''
+ return _pp_run_subtool(experiment, platform, target)
diff --git a/lib/fre/pp/schema.json b/lib/fre/pp/schema.json
new file mode 100644
index 00000000..dfb9cff5
--- /dev/null
+++ b/lib/fre/pp/schema.json
@@ -0,0 +1,75 @@
+{
+ "$schema": "http://json-schema.org/draft-06/schema#",
+ "title": "Schema for PP Yaml",
+ "type": "object",
+ "properties": {
+ "name": {"type": "string"},
+ "platform": {"type": "string"},
+ "target": {"type": "string"},
+ "directories": {
+ "description": "FRE shared directories",
+ "type": "object",
+ "items":{"$ref": "#/$defs/dirs" }
+ },
+ "postprocess": {
+ "description": "FRE post-processing information",
+ "type": "object",
+ "items":{"$ref": "#/$defs/pp" }
+ }
+ },
+ "$defs": {
+ "dirs": {
+ "history_dir": {"type":"string"},
+ "pp_dir": {"type":"string"},
+ "ptmp_dir": {"type":"string"},
+ "refinediag_scripts":{"type":["string","null"]},
+ "preanalysis_script":{"type":["string","null"]},
+ "history_refined":{"type":["string","null"]},
+ "analysis_dir":{"type":["string","null"]},
+ "pp_grid_spec": {"type":"string"},
+ "fre_analysis_home": {"type":["string","null"]}
+ },
+ "pp": {
+ "type": "object",
+ "properties": {
+ "settings": {
+ "type:": "object",
+ "properties": {
+ "history_segment": {"type":"string"},
+ "site": {"type":"string"},
+ "pp_chunk_a": {"type":"string"},
+ "pp_chunk_b": {"type":"string"},
+ "pp_start": {"type":"string"},
+ "pp_stop": {"type":"string"},
+ "pp_components": {"type":"string"}
+ }
+ },
+ "switches": {
+ "type": "object",
+ "properties": {
+ "clean_work": {"type":"boolean"},
+ "do_mdtf": {"type":"boolean"},
+ "do_statics": {"type":"boolean"},
+ "do_timeavgs": {"type":"boolean"},
+ "do_refinediag": {"type":"boolean"},
+ "do_atmos_plevel_masking": {"type":"boolean"},
+ "do_preanalysis": {"type":"boolean"},
+ "do_analysis": {"type":"boolean"},
+ "do_analysis_only": {"type":"boolean"}
+ }
+ },
+ "components": {
+ "type": "array",
+ "properties": {
+ "type": {"type":"string"},
+ "sources": {"type":"string"},
+ "sourceGrid": {"type":"string"},
+ "xyInterp": {"type":"string"},
+ "interpMethod": {"type":"string"},
+ "inputRealm": {"type":"string"}
+ }
+ }
+ }
+ }
+ }
+}
diff --git a/lib/fre/pp/status.py b/lib/fre/pp/status.py
new file mode 100644
index 00000000..0631be7a
--- /dev/null
+++ b/lib/fre/pp/status.py
@@ -0,0 +1,20 @@
+''' fre pp status '''
+
+import subprocess
+import click
+
+def status_subtool(experiment, platform, target):
+ """
+ Report workflow state for the Cylc workflow
+ ____
+ """
+
+ name = experiment + '__' + platform + '__' + target
+ cmd = f"cylc workflow-state {name}"
+ subprocess.run(cmd, shell=True, check=True, timeout=30)
+
+
+@click.command()
+def _status_subtool(experiment, platform, target):
+ ''' entry point to status for click '''
+ return _status_subtool(experiment, platform, target)
diff --git a/lib/fre/pp/tests/AM5_example/am5.yaml b/lib/fre/pp/tests/AM5_example/am5.yaml
new file mode 100644
index 00000000..4f3e5a4d
--- /dev/null
+++ b/lib/fre/pp/tests/AM5_example/am5.yaml
@@ -0,0 +1,115 @@
+# reusable variables
+fre_properties:
+ - &AM5_VERSION "am5f7b12r1"
+ - &FRE_STEM !join [am5/, *AM5_VERSION]
+
+ # amip
+ - &EXP_AMIP_START "19790101T0000Z"
+ - &EXP_AMIP_END "20200101T0000Z"
+ - &ANA_AMIP_START "19800101T0000Z"
+ - &ANA_AMIP_END "20200101T0000Z"
+
+ - &PP_AMIP_CHUNK96 "P1Y"
+ - &PP_AMIP_CHUNK384 "P1Y"
+ - &PP_XYINTERP96 "180,288"
+ - &PP_XYINTERP384 "720,1152"
+
+ # climo
+ - &EXP_CLIMO_START96 "0001"
+ - &EXP_CLIMO_END96 "0011"
+ - &ANA_CLIMO_START96 "0002"
+ - &ANA_CLIMO_END96 "0011"
+
+ - &EXP_CLIMO_START384 "0001"
+ - &EXP_CLIMO_END384 "0006"
+ - &ANA_CLIMO_START384 "0002"
+ - &ANA_CLIMO_END384 "0006"
+
+ # coupled
+ - &PP_CPLD_CHUNK_A "P5Y"
+ - &PP_CPLD_CHUNK_B "P20Y"
+
+ # grids
+ - &GRID_SPEC96 "/archive/oar.gfdl.am5/model_gen5/inputs/c96_grid/c96_OM4_025_grid_No_mg_drag_v20160808.tar"
+
+ # compile information
+ - &release "f1a1r1"
+ - &INTEL "intel-classic"
+ - &FMSincludes "-IFMS/fms2_io/include -IFMS/include -IFMS/mpp/include"
+ - &momIncludes "-Imom6/MOM6-examples/src/MOM6/pkg/CVMix-src/include"
+
+shared:
+ # compile information
+ compile:
+ compileYaml: &compile_yaml "compile.yaml"
+ platformYaml: "yaml_include/platforms.yaml"
+
+ # directories shared across tools
+ directories: &shared_directories
+ history_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, history]
+ pp_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, pp]
+ analysis_dir: !join [/nbhome/$USER/, *FRE_STEM, /, *name]
+ ptmp_dir: "/xtmp/$USER/ptmp"
+ fre_analysis_home: "/home/fms/local/opt/fre-analysis/test"
+
+ # shared pp settings
+ postprocess:
+ settings: &shared_settings
+ history_segment: "P1Y"
+ site: "ppan"
+ switches: &shared_switches
+ do_statics: True
+ do_timeavgs: True
+ clean_work: True
+ do_refinediag: False
+ do_atmos_plevel_masking: True
+ do_preanalysis: False
+ do_analysis: True
+
+experiments:
+ - name: "c96L65_am5f7b12r1_amip"
+ pp:
+ - "yaml_include/pp.c96_amip.yaml"
+ compile: *compile_yaml
+ - name: "c96L65_am5f7b12r1_pdclim1850F"
+ pp:
+ - "yaml_include/pp.c96_clim.yaml"
+ compile: *compile_yaml
+ - name: "c96L65_am5f7b12r1_pdclim2010F"
+ pp:
+ - "yaml_include/pp.c96_clim.yaml"
+ compile: *compile_yaml
+ - name: "c96L65_am5f7b12r1_pdclim2010AERF"
+ pp:
+ - "yaml_include/pp.c96_clim.yaml"
+ compile: *compile_yaml
+ - name: "c384L65_am5f7b12r1_amip"
+ pp:
+ - "yaml_include/pp.c384_amip.yaml"
+ compile: *compile_yaml
+ - name: "c384L65_am5f7b12r1_pdclim2010F"
+ pp:
+ - "yaml_include/pp.c384_clim.yaml"
+ compile: *compile_yaml
+ - name: "c384L65_am5f7b12r1_pdclim1850F"
+ pp:
+ - "yaml_include/pp.c384_clim.yaml"
+ compile: *compile_yaml
+ - name: "c384L65_am5f7b12r1_pdclim2010AERF"
+ pp:
+ - "yaml_include/pp.c384_clim.yaml"
+ compile: *compile_yaml
+ - name: "c384L65_am5f7b12r1_OM4_p25_piControl_noBLING_DynVeg"
+ pp:
+ - "yaml_include/pp.c384_amip.yaml"
+ - "yaml_include/pp.om4.yaml"
+ compile: *compile_yaml
+ - name: "c96L65_am5f7b12r1_OM4_p25_piControl_noBLING_DynVeg"
+ pp:
+ - "yaml_include/pp.c96_amip.yaml"
+ - "yaml_include/pp.om4.yaml"
+ compile: *compile_yaml
+ - name: "c96L65_am5f7b12r1_amip_cosp"
+ pp:
+ - "yaml_include/pp.c96_amip.yaml"
+ compile: *compile_yaml
diff --git a/lib/fre/pp/tests/AM5_example/compile.yaml b/lib/fre/pp/tests/AM5_example/compile.yaml
new file mode 100644
index 00000000..5f9a361b
--- /dev/null
+++ b/lib/fre/pp/tests/AM5_example/compile.yaml
@@ -0,0 +1,67 @@
+compile:
+ experiment: "am5"
+ container_addlibs:
+ baremetal_linkerflags:
+ src:
+ - component: "FMS"
+ repo: "https://github.com/NOAA-GFDL/FMS.git"
+ cppdefs: "-DINTERNAL_FILE_NML -Duse_libMPI -Duse_netCDF"
+ branch: "2022.01"
+ cppdefs: "-DHAVE_GETTID -Duse_libMPI -Duse_netCDF"
+ otherFlags: *FMSincludes
+ - component: "am5_phys"
+ requires: ["FMS"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/am5_phys.git"
+ branch: "2022.01"
+ otherFlags: *FMSincludes
+ - component: "GFDL_atmos_cubed_sphere"
+ requires: ["FMS", "am5_phys"]
+ repo: "https://github.com/NOAA-GFDL/GFDL_atmos_cubed_sphere.git"
+ cppdefs: "-DSPMD -DCLIMATE_NUDGE -DINTERNAL_FILE_NML"
+ branch: "2022.01"
+ paths: ["GFDL_atmos_cubed_sphere/driver/GFDL",
+ "GFDL_atmos_cubed_sphere/model",
+ "GFDL_atmos_cubed_sphere/driver/SHiELD/cloud_diagnosis.F90",
+ "GFDL_atmos_cubed_sphere/driver/SHiELD/gfdl_cloud_microphys.F90",
+ "GFDL_atmos_cubed_sphere/tools",
+ "GFDL_atmos_cubed_sphere/GFDL_tools"]
+ otherFlags: *FMSincludes
+ - component: "atmos_drivers"
+ requires: ["FMS", "am5_phys", "GFDL_atmos_cubed_sphere"]
+ repo: "https://github.com/NOAA-GFDL/atmos_drivers.git"
+ cppdefs: "-DSPMD -DCLIMATE_NUDGE"
+ branch: "2022.01"
+ paths: ["atmos_drivers/coupled"]
+ otherFlags: *FMSincludes
+ - component: "ice_sis"
+ requires: ["FMS", "ice_param", "mom6"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/ice_sis.git"
+ branch: "2021.02"
+ otherFlags: !join [*FMSincludes, " ", *momIncludes]
+ - component: "ice_param"
+ repo: "https://github.com/NOAA-GFDL/ice_param.git"
+ cppdefs: "-Duse_yaml -Duse_libMPI -Duse_netCDF"
+ branch: "2021.02"
+ requires: ["FMS", "mom6"]
+ otherFlags: !join [*FMSincludes," ", *momIncludes]
+ - component: "land_lad2"
+ requires: ["FMS"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/land_lad2.git"
+ branch: "2022.01"
+ branch: "land_lad2_2021.02"
+ doF90Cpp: True
+ cppdefs: "-DINTERNAL_FILE_NML"
+ otherFlags: *FMSincludes
+ - component: "mom6"
+ requires: ["FMS"]
+ paths: ["mom6/MOM6-examples/src/MOM6/config_src/dynamic", "mom6/MOM6-examples/src/MOM6/config_src/coupled_driver", "mom6/MOM6-examples/src/MOM6/src/*/", "mom6/MOM6-examples/src/MOM6/src/*/*/", "mom6/ocean_BGC/generic_tracers", "mom6/ocean_BGC/mocsy/src"]
+ branch: ["2021.02","dev/gfdl/2018.04.06"]
+ repo: ["https://github.com/NOAA-GFDL/ocean_BGC.git","https://github.com/NOAA-GFDL/MOM6-examples.git"]
+ makeOverrides: 'OPENMP=""'
+ otherFlags: !join [*FMSincludes, " ", *momIncludes]
+ - component: "FMScoupler"
+ paths: ["FMScoupler/full", "FMScoupler/shared"]
+ repo: "https://github.com/NOAA-GFDL/FMScoupler.git"
+ branch: "2022.01"
+ requires: ["FMS", "atmos_drivers", "am5_phys", "land_lad2", "ice_sis", "ice_param", "mom6"]
+ otherFlags: !join [*FMSincludes, " ", *momIncludes]
diff --git a/lib/fre/pp/tests/AM5_example/yaml_include/platforms.yaml b/lib/fre/pp/tests/AM5_example/yaml_include/platforms.yaml
new file mode 100644
index 00000000..7e1b9f49
--- /dev/null
+++ b/lib/fre/pp/tests/AM5_example/yaml_include/platforms.yaml
@@ -0,0 +1,26 @@
+platforms:
+ - name: ncrc5.intel
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: [ !join [*INTEL, "/2022.2.1"],"fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: ncrc5.intel23
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: [!join [*INTEL, "/2023.1.0"],"fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: hpcme.2023
+ compiler: intel
+ RUNenv: [". /spack/share/spack/setup-env.sh", "spack load libyaml", "spack load netcdf-fortran@4.5.4", "spack load hdf5@1.14.0"]
+ modelRoot: /apps
+ fc: mpiifort
+ cc: mpiicc
+ container: True
+ containerBuild: "podman"
+ containerRun: "apptainer"
diff --git a/lib/fre/pp/tests/AM5_example/yaml_include/pp.c96_amip.yaml b/lib/fre/pp/tests/AM5_example/yaml_include/pp.c96_amip.yaml
new file mode 100644
index 00000000..117c66c6
--- /dev/null
+++ b/lib/fre/pp/tests/AM5_example/yaml_include/pp.c96_amip.yaml
@@ -0,0 +1,88 @@
+# local reusable variable overrides
+fre_properties:
+ - &custom_interp "180,360"
+
+# directory overrides
+#c96_amip_directories:
+directories:
+ <<: *shared_directories
+ ptmp_dir: "/ptmp/$USER"
+ pp_grid_spec: *GRID_SPEC96
+
+#c96_amip_postprocess:
+postprocess:
+ # pp setting overrides
+ settings:
+ <<: *shared_settings
+ pp_start: *ANA_AMIP_START
+ pp_stop: *ANA_AMIP_END
+ pp_chunk_a: *PP_AMIP_CHUNK96
+ pp_components: "atmos atmos_scalar"
+ switches:
+ <<: *shared_switches
+ do_statics: False
+
+ # main pp instructions
+ components:
+ - type: "atmos_cmip"
+ sources: "atmos_month_cmip atmos_8xdaily_cmip atmos_daily_cmip"
+ sourceGrid: "cubedsphere"
+ xyInterp: *custom_interp
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos"
+ sources: "atmos_month"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_level_cmip"
+ sources: "atmos_level_cmip"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_level"
+ sources: "atmos_month"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_month_aer"
+ sources: "atmos_month_aer"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'atmos'
+ - type: "atmos_diurnal"
+ sources: "atmos_diurnal"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_scalar"
+ sources: "atmos_scalar"
+ - type: "aerosol_cmip"
+ xyInterp: *PP_XYINTERP96
+ sources: "aerosol_month_cmip"
+ sourceGrid: "cubedsphere"
+ interpMethod: "conserve_order1"
+ inputRealm: 'atmos'
+ - type: "land"
+ sources: "land_month"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'land'
+ - type: "land_cmip"
+ sources: "land_month_cmip"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'land'
+ - type: "tracer_level"
+ sources: "atmos_tracer"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'atmos'
diff --git a/lib/fre/pp/tests/__init__.py b/lib/fre/pp/tests/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/pp/tests/test_configure_script_yaml.py b/lib/fre/pp/tests/test_configure_script_yaml.py
new file mode 100644
index 00000000..eaf1fc2e
--- /dev/null
+++ b/lib/fre/pp/tests/test_configure_script_yaml.py
@@ -0,0 +1,46 @@
+"""
+Test configure_script_yaml
+"""
+import os
+from pathlib import Path
+from fre.pp import configure_script_yaml as csy
+
+# Set what would be click options
+EXPERIMENT = "c96L65_am5f7b12r1_amip"
+PLATFORM = "gfdl.ncrc5-intel22-classic"
+TARGET = "prod-openmp"
+
+# Set example yaml paths, input directory
+test_dir = Path("fre/pp/tests")
+test_yaml = Path("AM5_example/am5.yaml")
+
+def test_combinedyaml_exists():
+ """
+ Make sure combined yaml file exists
+ """
+ assert Path(f"{test_dir}/{test_yaml}").exists()
+
+def test_configure_script():
+ """
+ Tests success of confgure yaml script
+ Creates rose-suite, regrid rose-app, remap rose-app
+ TO-DO: will break this up for better tests
+ """
+ # Set home for ~/cylc-src location in script
+ os.environ["HOME"]=str(Path(f"{test_dir}/configure_yaml_out"))
+
+ # Set output directory
+ out_dir = Path(f"{os.getenv('HOME')}/cylc-src/{EXPERIMENT}__{PLATFORM}__{TARGET}")
+ Path(out_dir).mkdir(parents=True,exist_ok=True)
+
+ # Define combined yaml
+ model_yaml = str(Path(f"{test_dir}/{test_yaml}"))
+
+ # Invoke configure_yaml_script.py
+ csy.yamlInfo(model_yaml,EXPERIMENT,PLATFORM,TARGET)
+
+ # Check for configuration creation and final combined yaml
+ assert all([Path(f"{out_dir}/{EXPERIMENT}.yaml").exists(),
+ Path(f"{out_dir}/rose-suite.conf").exists(),
+ Path(f"{out_dir}/app/regrid-xy/rose-app.conf").exists(),
+ Path(f"{out_dir}/app/remap-pp-components/rose-app.conf").exists()])
diff --git a/lib/fre/pp/tests/test_rose_quoting.py b/lib/fre/pp/tests/test_rose_quoting.py
new file mode 100644
index 00000000..bd5353b1
--- /dev/null
+++ b/lib/fre/pp/tests/test_rose_quoting.py
@@ -0,0 +1,7 @@
+from fre.pp.configure_script_yaml import quote_rose_values
+
+def test_boolean():
+ assert quote_rose_values(True) == 'True'
+
+def test_string():
+ assert quote_rose_values('foo') == "'foo'"
diff --git a/lib/fre/pp/trigger.py b/lib/fre/pp/trigger.py
new file mode 100644
index 00000000..e6ebd01b
--- /dev/null
+++ b/lib/fre/pp/trigger.py
@@ -0,0 +1,19 @@
+''' fre pp trigger '''
+
+import subprocess
+import click
+
+def trigger(experiment, platform, target, time):
+ """
+ Trigger the pp-starter task for the time indicated
+ """
+
+ name = experiment + '__' + platform + '__' + target
+ cmd = f"cylc trigger {name}//{time}/pp-starter"
+ subprocess.run(cmd, shell=True, check=True, timeout=30)
+
+
+@click.command()
+def _trigger(experiment, platform, target, time):
+ ''' entry point to trigger for click '''
+ return trigger(experiment, platform, target, time)
diff --git a/lib/fre/pp/validate.py b/lib/fre/pp/validate.py
new file mode 100644
index 00000000..fd54c52c
--- /dev/null
+++ b/lib/fre/pp/validate.py
@@ -0,0 +1,30 @@
+''' fre pp validate '''
+
+import os
+import subprocess
+import click
+
+def validate_subtool(experiment, platform, target):
+ """
+ Validate the Cylc workflow definition located in
+ ~/cylc-src/____
+ """
+ go_back_here = os.getcwd()
+ directory = os.path.expanduser('~/cylc-src/' + experiment + '__' + platform + '__' + target)
+
+ # Change the current working directory
+ os.chdir(directory)
+
+ # Run the Rose validation macros
+ cmd = "rose macro --validate"
+ subprocess.run(cmd, shell=True, check=True)
+
+ # Validate the Cylc configuration
+ cmd = "cylc validate ."
+ subprocess.run(cmd, shell=True, check=True)
+ os.chdir(go_back_here)
+
+@click.command()
+def _validate_subtool(experiment, platform, target):
+ ''' entry point to validate for click '''
+ return validate_subtool(experiment, platform, target)
diff --git a/lib/fre/pp/wrapper.py b/lib/fre/pp/wrapper.py
new file mode 100644
index 00000000..c9d6793b
--- /dev/null
+++ b/lib/fre/pp/wrapper.py
@@ -0,0 +1,49 @@
+"""
+frepp.py, a replacement for the frepp bash script located at:
+https://gitlab.gfdl.noaa.gov/fre2/system-settings/-/blob/main/bin/frepp
+Author: Carolyn.Whitlock
+"""
+
+#todo:
+# add relative path import to rest of pp tools
+# add command-line args using same format as fre.py
+# include arg for pp start / stop
+# test yaml path
+# error handling
+
+import os
+import time
+import click
+
+# Import from the local packages
+from .checkout_script import checkout_template
+from .configure_script_yaml import yamlInfo
+from .install import install_subtool
+from .run import pp_run_subtool
+from .trigger import trigger
+from .status import status_subtool
+
+@click.command()
+def runFre2pp(experiment, platform, target, config_file, branch, time):
+ '''
+ Wrapper script for calling a FRE2 pp experiment with the canopy-style
+ infrastructure and fre-cli
+ '''
+
+ config_file = os.path.abspath(config_file)
+
+ checkout_template(experiment, platform, target, branch)
+
+ yamlInfo(config_file, experiment, platform, target)
+
+ install_subtool(experiment, platform, target)
+
+ pp_run_subtool(experiment, platform, target)
+
+ if time:
+ trigger(experiment, platform, target, time)
+
+ status_subtool(experiment, platform, target)
+
+if __name__ == '__main__':
+ runFre2pp()
diff --git a/lib/fre/pytest.ini b/lib/fre/pytest.ini
new file mode 100644
index 00000000..eb793231
--- /dev/null
+++ b/lib/fre/pytest.ini
@@ -0,0 +1,15 @@
+[pytest]
+testpaths =
+ fre/tests
+ fre/catalog/tests
+# fre/check/tests
+ fre/cmor/tests
+# fre/list/tests
+ fre/make/tests
+ fre/pp/tests
+# fre/run/tests
+# fre/test/tests
+ fre/yamltools/tests
+# fre/app/tests
+ fre/app/generate_time_averages/tests
+ fre/app/regrid_xy/tests
diff --git a/lib/fre/run/__init__.py b/lib/fre/run/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/run/frerun.py b/lib/fre/run/frerun.py
new file mode 100644
index 00000000..c2c611c3
--- /dev/null
+++ b/lib/fre/run/frerun.py
@@ -0,0 +1,21 @@
+'''
+entry point for fre run subcommands
+'''
+
+import click
+from .frerunexample import run_test_function
+
+@click.group(help=click.style(" - access fre run subcommands", fg=(164,29,132)))
+def run_cli():
+ ''' entry point to fre run click commands '''
+
+@run_cli.command()
+@click.option('--uppercase', '-u', is_flag=True, help = 'Print statement in uppercase.')
+@click.pass_context
+def function(context, uppercase):
+ # pylint: disable=unused-argument
+ """ - Execute fre run test """
+ context.forward(run_test_function)
+
+if __name__ == "__main__":
+ run_cli()
diff --git a/lib/fre/run/frerunexample.py b/lib/fre/run/frerunexample.py
new file mode 100644
index 00000000..a4335144
--- /dev/null
+++ b/lib/fre/run/frerunexample.py
@@ -0,0 +1,18 @@
+"""
+experimentation file for integrating one file's functions into main prototype fre file
+authored by Bennett.Chang@noaa.gov | bcc2761
+NOAA | GFDL
+"""
+
+import click
+
+@click.command()
+def run_test_function(uppercase=None):
+ """Execute fre run run_test_function"""
+ statement = "testingtestingtestingtesting"
+ if uppercase:
+ statement = statement.upper()
+ click.echo(statement)
+
+if __name__ == '__main__':
+ run_test_function()
diff --git a/lib/fre/test/__init__.py b/lib/fre/test/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/test/fretest.py b/lib/fre/test/fretest.py
new file mode 100644
index 00000000..172ac855
--- /dev/null
+++ b/lib/fre/test/fretest.py
@@ -0,0 +1,21 @@
+'''
+entry point for fre test subcommands
+'''
+
+import click
+from .fretestexample import test_test_function
+
+@click.group(help=click.style(" - access fre test subcommands", fg=(92,164,29)))
+def test_cli():
+ ''' entry point to fre test click commands '''
+
+@test_cli.command()
+@click.option('--uppercase', '-u', is_flag=True, help = 'Print statement in uppercase.')
+@click.pass_context
+def function(context, uppercase):
+ # pylint: disable=unused-argument
+ """ - Execute fre test test """
+ context.forward(test_test_function)
+
+if __name__ == "__main__":
+ test_cli()
diff --git a/lib/fre/test/fretestexample.py b/lib/fre/test/fretestexample.py
new file mode 100644
index 00000000..86d0f6ce
--- /dev/null
+++ b/lib/fre/test/fretestexample.py
@@ -0,0 +1,18 @@
+"""
+experimentation file for integrating one file's functions into main prototype fre file
+authored by Bennett.Chang@noaa.gov | bcc2761
+NOAA | GFDL
+"""
+
+import click
+
+@click.command()
+def test_test_function(uppercase=None):
+ """Execute fre list test_test_function"""
+ statement = "testingtestingtestingtesting"
+ if uppercase:
+ statement = statement.upper()
+ click.echo(statement)
+
+if __name__ == '__main__':
+ test_test_function()
diff --git a/lib/fre/tests/README.md b/lib/fre/tests/README.md
new file mode 100644
index 00000000..31b2c2c6
--- /dev/null
+++ b/lib/fre/tests/README.md
@@ -0,0 +1,20 @@
+note- this directory is for unit tests assessing `fre` functionality, and NOT for code corresponding to the command `fre test`
+
+to run the tests as intended, simply call `pytest` from the root directory of this repository.
+
+From the root directory of this repository, if you want to...
+
+Invoke all tests (intended use):
+ `pytest fre/tests/`
+
+Invoke all tests in a single file:
+ `pytest fre/tests/test_fre_cli.py`
+
+Invoke a single test in a file:
+ `pytest fre/tests/test_fre_cli.py::test_cli_fre_option_dne`
+
+
+Note that pytest will not print stdout from individual tests. If you want to...
+
+Print stdout from an invoked test (debugging with print statements):
+ `pytest fre/tests/test_fre_cli.py::test_cli_fre_option_dne -s`
diff --git a/lib/fre/tests/__init__.py b/lib/fre/tests/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/tests/test_files/CMOR_input_example.json b/lib/fre/tests/test_files/CMOR_input_example.json
new file mode 100644
index 00000000..3ce8a985
--- /dev/null
+++ b/lib/fre/tests/test_files/CMOR_input_example.json
@@ -0,0 +1,74 @@
+{
+ "#note": "explanation of what source_type is goes here",
+ "source_type": "AOGCM ISM AER",
+
+ "#note": "CMIP6 valid experiment_ids are found in CMIP6_CV.json",
+ "experiment_id": "piControl-withism",
+ "activity_id": "ISMIP6",
+ "sub_experiment_id": "none",
+
+ "realization_index": "3",
+ "initialization_index": "1",
+ "physics_index": "1",
+ "forcing_index": "1",
+
+ "#note": "Text stored in attribute variant_info (recommended, not required description of run variant)",
+ "run_variant": "3rd realization",
+
+ "parent_experiment_id": "no parent",
+ "parent_activity_id": "no parent",
+ "parent_source_id": "PCMDI-test-1-0",
+ "parent_variant_label": "r3i1p1f1",
+
+ "parent_time_units": "days since 1850-01-01",
+ "branch_method": "standard",
+ "branch_time_in_child": 59400.0,
+ "branch_time_in_parent": 59400.0,
+
+ "#note": "institution_id must be registered at https://github.com/WCRP-CMIP/CMIP6_CVs/issues/new ",
+ "institution_id": "PCMDI",
+
+ "#note": "source_id (model name) must be registered at https://github.com/WCRP-CMIP/CMIP6_CVs/issues/new ",
+ "source_id": "PCMDI-test-1-0",
+
+ "calendar": "360_day",
+
+ "grid": "native atmosphere regular grid (3x4 latxlon)",
+ "grid_label": "gn",
+ "nominal_resolution": "10000 km",
+
+ "license": "CMIP6 model data produced by Lawrence Livermore PCMDI is licensed under a Creative Commons Attribution 4.0 International License (https://creativecommons.org/licenses/by/4.0/). Consult https://pcmdi.llnl.gov/CMIP6/TermsOfUse for terms of use governing CMIP6 output, including citation requirements and proper acknowledgment. Further information about this data, including some limitations, can be found via the further_info_url (recorded as a global attribute in this file) and at https:///pcmdi.llnl.gov/. The data producers and data providers make no warranty, either express or implied, including, but not limited to, warranties of merchantability and fitness for a particular purpose. All liabilities arising from the supply of the information (including any liability arising in negligence) are excluded to the fullest extent permitted by law.",
+
+ "#output": "Root directory for output (can be either a relative or full path)",
+ "outpath": "CMIP6",
+
+ "#note": " **** The following descriptors are optional and may be set to an empty string ",
+
+ "contact ": "Python Coder (coder@a.b.c.com)",
+ "history": "Output from archivcl_A1.nce/giccm_03_std_2xCO2_2256.",
+ "comment": "",
+ "references": "Model described by Koder and Tolkien (J. Geophys. Res., 2001, 576-591). Also see http://www.GICC.su/giccm/doc/index.html. The ssp245 simulation is described in Dorkey et al. '(Clim. Dyn., 2003, 323-357.)'",
+
+ "#note": " **** The following will be obtained from the CV and do not need to be defined here",
+
+ "sub_experiment": "none",
+ "institution": "",
+ "source": "PCMDI-test 1.0 (1989)",
+
+ "#note": " **** The following are set correctly for CMIP6 and should not normally need editing",
+
+ "_controlled_vocabulary_file": "CMIP6_CV.json",
+ "_AXIS_ENTRY_FILE": "CMIP6_coordinate.json",
+ "_FORMULA_VAR_FILE": "CMIP6_formula_terms.json",
+ "_cmip6_option": "CMIP6",
+
+ "mip_era": "CMIP6",
+ "parent_mip_era": "CMIP6",
+
+ "tracking_prefix": "hdl:21.14100",
+ "_history_template": "%s ;rewrote data to be consistent with for variable found in table .",
+
+ "#output_path_template": "Template for output path directory using tables keys or global attributes, these should follow the relevant data reference syntax",
+ "output_path_template": "<_member_id>",
+ "output_file_template": "<_member_id>",
+}
diff --git a/lib/fre/tests/test_files/CMORbite_var_list.json b/lib/fre/tests/test_files/CMORbite_var_list.json
new file mode 100644
index 00000000..5b75e54f
--- /dev/null
+++ b/lib/fre/tests/test_files/CMORbite_var_list.json
@@ -0,0 +1,11 @@
+{
+ "lai": "lai",
+ "t_ref": "t_ref",
+ "cl": "cl",
+ "mc": "mc",
+ "ta": "ta",
+ "sos": "sos",
+ "so": "so",
+ "ch4global": "ch4global",
+ "gppLut": "gppLut"
+}
diff --git a/lib/fre/tests/test_files/ocean_sos_var_file/ocean_monthly_1x1deg.199301-199712.sos.nc b/lib/fre/tests/test_files/ocean_sos_var_file/ocean_monthly_1x1deg.199301-199712.sos.nc
new file mode 100644
index 00000000..77c1ec7f
Binary files /dev/null and b/lib/fre/tests/test_files/ocean_sos_var_file/ocean_monthly_1x1deg.199301-199712.sos.nc differ
diff --git a/lib/fre/tests/test_files/varlist b/lib/fre/tests/test_files/varlist
new file mode 100644
index 00000000..6ab9e86a
--- /dev/null
+++ b/lib/fre/tests/test_files/varlist
@@ -0,0 +1,3 @@
+{
+ "sos": "sos"
+}
diff --git a/lib/fre/tests/test_files/varlist_local_target_vars_differ b/lib/fre/tests/test_files/varlist_local_target_vars_differ
new file mode 100644
index 00000000..402be0e6
--- /dev/null
+++ b/lib/fre/tests/test_files/varlist_local_target_vars_differ
@@ -0,0 +1,3 @@
+{
+ "sosV2": "sos"
+}
diff --git a/lib/fre/tests/test_fre_app_cli.py b/lib/fre/tests/test_fre_app_cli.py
new file mode 100644
index 00000000..ca14df51
--- /dev/null
+++ b/lib/fre/tests/test_fre_app_cli.py
@@ -0,0 +1,117 @@
+""" test "fre app" calls """
+
+import os
+import subprocess
+from pathlib import Path
+
+import click
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+# fre app
+def test_cli_fre_app(capfd):
+ """ fre app """
+ result = runner.invoke(fre.fre, args=["app"])
+ assert result.exit_code == 0
+ _out, _err = capfd.readouterr()
+
+def test_cli_fre_app_help(capfd):
+ """ fre app --help """
+ result = runner.invoke(fre.fre, args=["app", "--help"])
+ assert result.exit_code == 0
+ _out, _err = capfd.readouterr()
+
+def test_cli_fre_app_opt_dne(capfd):
+ """ fre app optionDNE """
+ result = runner.invoke(fre.fre, args=["app", "optionDNE"])
+ assert result.exit_code == 2
+ _out, _err = capfd.readouterr()
+
+# fre app gen-time-averages
+def test_cli_fre_app_gen_time_averages(capfd):
+ """ fre cmor run """
+ result = runner.invoke(fre.fre, args=["app", "gen-time-averages"])
+ assert result.exit_code == 2
+ _out, _err = capfd.readouterr()
+
+def test_cli_fre_app_gen_time_averages_help(capfd):
+ """ fre cmor run --help """
+ result = runner.invoke(fre.fre, args=["app", "gen-time-averages", "--help"])
+ assert result.exit_code == 0
+ _out, _err = capfd.readouterr()
+
+def test_cli_fre_app_gen_time_averages_opt_dne(capfd):
+ """ fre cmor run optionDNE """
+ result = runner.invoke(fre.fre, args=["app", "gen-time-averages", "optionDNE"])
+ assert result.exit_code == 2
+ _out, _err = capfd.readouterr()
+
+# fre app regrid
+def test_cli_fre_app_regrid(capfd):
+ """ fre cmor run """
+ result = runner.invoke(fre.fre, args=["app", "regrid"])
+ assert result.exit_code == 2
+ _out, _err = capfd.readouterr()
+
+def test_cli_fre_app_regrid_help(capfd):
+ """ fre cmor run --help """
+ result = runner.invoke(fre.fre, args=["app", "regrid", "--help"])
+ assert result.exit_code == 0
+ _out, _err = capfd.readouterr()
+
+def test_cli_fre_app_regrid_opt_dne(capfd):
+ """ fre cmor run optionDNE """
+ result = runner.invoke(fre.fre, args=["app", "regrid", "optionDNE"])
+ assert result.exit_code == 2
+ _out, _err = capfd.readouterr()
+
+def test_cli_fre_app_regrid_test_case_1(capfd):
+ """ fre cmor run --help """
+
+ import fre.app.regrid_xy.tests.test_regrid_xy as t_rgxy
+ assert t_rgxy is not None
+
+ # input files for this test are locked up in here as well
+ if not Path( t_rgxy.TEST_DIR+'/in-dir' ).exists():
+ assert Path(t_rgxy.TAR_IN_DIR).exists()
+ ex = [ "tar", "-C", t_rgxy.TEST_DIR, "-zxvf", t_rgxy.TAR_IN_DIR ]
+ sp = subprocess.run( ex )
+ assert all ( [ sp.returncode == 0,
+ Path(IN_DIR).exists() ] )
+
+ # for the time being, still a little dependent on rose for configuration value passing
+ if Path(os.getcwd()+'/rose-app-run.conf').exists():
+ Path(os.getcwd()+'/rose-app-run.conf').unlink()
+
+ with open(os.getcwd()+'/rose-app-run.conf','a',encoding='utf-8') as rose_app_run_config:
+ rose_app_run_config.write( '[command]\n' )
+ rose_app_run_config.write( 'default=regrid-xy\n' )
+ rose_app_run_config.write( '\n' )
+ rose_app_run_config.write( f'[{t_rgxy.COMPONENT}]\n' )
+ rose_app_run_config.write( f'sources={t_rgxy.SOURCE}\n' )
+ rose_app_run_config.write( f'inputGrid={t_rgxy.INPUT_GRID}\n' )
+ rose_app_run_config.write( f'inputRealm={t_rgxy.INPUT_REALM}\n' )
+ rose_app_run_config.write( f'interpMethod={t_rgxy.INTERP_METHOD}\n' )
+ rose_app_run_config.write( f'outputGridLon={t_rgxy.NLON}\n' )
+ rose_app_run_config.write( f'outputGridLat={t_rgxy.NLAT}\n' )
+ rose_app_run_config.write( '\n' )
+ assert Path('./rose-app-run.conf').exists()
+
+ args_list = ["app", "regrid",
+ "--input_dir", f"{t_rgxy.WORK_YYYYMMDD_DIR}",
+ "--output_dir", f"{t_rgxy.TEST_OUT_DIR}",
+ "--begin", f"{t_rgxy.YYYYMMDD}T000000",
+ "--tmp_dir", f"{t_rgxy.TEST_DIR}",
+ "--remap_dir", f"{t_rgxy.REMAP_DIR}",
+ "--source", f"{t_rgxy.SOURCE}",
+ "--grid_spec", f"{t_rgxy.GOLD_GRID_SPEC_NO_TAR}",
+ "--def_xy_interp", f'"{t_rgxy.NLON},{t_rgxy.NLAT}"' ]
+ click.echo(f'args_list = \n {args_list}')
+ click.echo('fre ' + ' '.join(args_list))
+
+ result = runner.invoke(fre.fre, args=args_list )
+ assert result.exit_code == 0
+ _out, _err = capfd.readouterr()
diff --git a/lib/fre/tests/test_fre_catalog_cli.py b/lib/fre/tests/test_fre_catalog_cli.py
new file mode 100644
index 00000000..79a1d35a
--- /dev/null
+++ b/lib/fre/tests/test_fre_catalog_cli.py
@@ -0,0 +1,50 @@
+''' test "fre catalog" calls '''
+
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+def test_cli_fre_catalog():
+ ''' fre catalog '''
+ result = runner.invoke(fre.fre, args=["catalog"])
+ assert result.exit_code == 0
+
+def test_cli_fre_catalog_help():
+ ''' fre catalog --help '''
+ result = runner.invoke(fre.fre, args=["catalog", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_catalog_opt_dne():
+ ''' fre catalog optionDNE '''
+ result = runner.invoke(fre.fre, args=["catalog", "optionDNE"])
+ assert result.exit_code == 2
+
+def test_cli_fre_catalog_builder():
+ ''' fre catalog builder '''
+ result = runner.invoke(fre.fre, args=["catalog", "builder"])
+ stdout_str = 'Missing: input_path or output_path. ' + \
+ 'Pass it in the config yaml or as command-line option'
+ assert all( [
+ result.exit_code == 1,
+ stdout_str in result.stdout.split('\n')
+ ]
+ )
+
+def test_cli_fre_catalog_builder_help():
+ ''' fre catalog builder --help '''
+ result = runner.invoke(fre.fre, args=["catalog", "builder", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_catalog_merge():
+ result = runner.invoke(fre.fre, args=["catalog", "merge"])
+ expected_stdout = "Error: Missing option '--input'."
+ assert all( [
+ result.exit_code == 2,
+ expected_stdout in result.stdout.split('\n')
+ ] )
+
+def test_cli_fre_catalog_merge_help():
+ result = runner.invoke(fre.fre, args=["catalog", "merge", "--help"])
+ assert result.exit_code == 0
diff --git a/lib/fre/tests/test_fre_check_cli.py b/lib/fre/tests/test_fre_check_cli.py
new file mode 100644
index 00000000..80657b65
--- /dev/null
+++ b/lib/fre/tests/test_fre_check_cli.py
@@ -0,0 +1,22 @@
+''' test "fre check" calls '''
+
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+def test_cli_fre_check():
+ ''' fre check '''
+ result = runner.invoke(fre.fre, args=["check"])
+ assert result.exit_code == 0
+
+def test_cli_fre_check_help():
+ ''' fre check --help '''
+ result = runner.invoke(fre.fre, args=["check", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_check_opt_dne():
+ ''' fre check optionDNE '''
+ result = runner.invoke(fre.fre, args=["check", "optionDNE"])
+ assert result.exit_code == 2
diff --git a/lib/fre/tests/test_fre_cli.py b/lib/fre/tests/test_fre_cli.py
new file mode 100644
index 00000000..5c5bdb12
--- /dev/null
+++ b/lib/fre/tests/test_fre_cli.py
@@ -0,0 +1,22 @@
+''' test "fre" calls '''
+
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+def test_cli_fre():
+ ''' fre '''
+ result = runner.invoke(fre.fre)
+ assert result.exit_code == 0
+
+def test_cli_fre_help():
+ ''' fre --help '''
+ result = runner.invoke(fre.fre, args='--help')
+ assert result.exit_code == 0
+
+def test_cli_fre_option_dne():
+ ''' fre optionDNE '''
+ result = runner.invoke(fre.fre, args='optionDNE')
+ assert result.exit_code == 2
diff --git a/lib/fre/tests/test_fre_cmor_cli.py b/lib/fre/tests/test_fre_cmor_cli.py
new file mode 100644
index 00000000..b2a09b62
--- /dev/null
+++ b/lib/fre/tests/test_fre_cmor_cli.py
@@ -0,0 +1,134 @@
+''' test "fre cmor" calls '''
+
+from datetime import date
+from pathlib import Path
+
+import click
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+# fre cmor
+def test_cli_fre_cmor():
+ ''' fre cmor '''
+ result = runner.invoke(fre.fre, args=["cmor"])
+ assert result.exit_code == 0
+
+def test_cli_fre_cmor_help():
+ ''' fre cmor --help '''
+ result = runner.invoke(fre.fre, args=["cmor", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_cmor_opt_dne():
+ ''' fre cmor optionDNE '''
+ result = runner.invoke(fre.fre, args=["cmor", "optionDNE"])
+ assert result.exit_code == 2
+
+# fre cmor run
+def test_cli_fre_cmor_run():
+ ''' fre cmor run '''
+ result = runner.invoke(fre.fre, args=["cmor", "run"])
+ assert result.exit_code == 2
+
+def test_cli_fre_cmor_run_help():
+ ''' fre cmor run --help '''
+ result = runner.invoke(fre.fre, args=["cmor", "run", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_cmor_run_opt_dne():
+ ''' fre cmor run optionDNE '''
+ result = runner.invoke(fre.fre, args=["cmor", "run", "optionDNE"])
+ assert result.exit_code == 2
+
+
+# these unit tests should be more about the cli, rather than the workload
+YYYYMMDD=date.today().strftime('%Y%m%d')
+def test_cli_fre_cmor_run_case1():
+ ''' fre cmor run, test-use case '''
+
+ # where are we? we're running pytest from the base directory of this repo
+ rootdir = 'fre/tests/test_files'
+
+ # explicit inputs to tool
+ indir = f'{rootdir}/ocean_sos_var_file'
+ varlist = f'{rootdir}/varlist'
+ table_config = f'{rootdir}/cmip6-cmor-tables/Tables/CMIP6_Omon.json'
+ exp_config = f'{rootdir}/CMOR_input_example.json'
+ outdir = f'{rootdir}/outdir'
+
+ # determined by cmor_run_subtool
+ cmor_creates_dir = \
+ 'CMIP6/CMIP6/ISMIP6/PCMDI/PCMDI-test-1-0/piControl-withism/r3i1p1f1/Omon/sos/gn'
+ full_outputdir = \
+ f"{outdir}/{cmor_creates_dir}/v{YYYYMMDD}" # yay no more 'fre' where it shouldnt be
+ full_outputfile = \
+ f"{full_outputdir}/sos_Omon_PCMDI-test-1-0_piControl-withism_r3i1p1f1_gn_199307-199807.nc"
+
+ # FYI
+ filename = 'ocean_monthly_1x1deg.199301-199712.sos.nc' # unneeded, this is mostly for reference
+ full_inputfile=f"{indir}/{filename}"
+
+ # clean up, lest we fool outselves
+ if Path(full_outputfile).exists():
+ Path(full_outputfile).unlink()
+
+ #click.echo('')
+ result = runner.invoke(fre.fre, args = ["cmor", "run",
+ "--indir", indir,
+ "--varlist", varlist,
+ "--table_config", table_config,
+ "--exp_config", exp_config,
+ "--outdir", outdir])
+ click.echo(f'stdout = \n {result.stdout}')
+ #click.echo(f'stderr = \n {result.stderr}') #not captured sep.
+ assert all ( [ result.exit_code == 0,
+ Path(full_outputfile).exists(),
+ Path(full_inputfile).exists() ] )
+
+
+
+
+
+def test_cli_fre_cmor_run_case2():
+ ''' fre cmor run, test-use case '''
+
+ # where are we? we're running pytest from the base directory of this repo
+ rootdir = 'fre/tests/test_files'
+
+ # explicit inputs to tool
+ indir = f'{rootdir}/ocean_sos_var_file'
+ varlist = f'{rootdir}/varlist_local_target_vars_differ'
+ table_config = f'{rootdir}/cmip6-cmor-tables/Tables/CMIP6_Omon.json'
+ exp_config = f'{rootdir}/CMOR_input_example.json'
+ outdir = f'{rootdir}/outdir'
+
+ # determined by cmor_run_subtool
+ cmor_creates_dir = \
+ 'CMIP6/CMIP6/ISMIP6/PCMDI/PCMDI-test-1-0/piControl-withism/r3i1p1f1/Omon/sos/gn'
+ full_outputdir = \
+ f"{outdir}/{cmor_creates_dir}/v{YYYYMMDD}" # yay no more 'fre' where it shouldnt be
+ full_outputfile = \
+ f"{full_outputdir}/sos_Omon_PCMDI-test-1-0_piControl-withism_r3i1p1f1_gn_199307-199807.nc"
+
+ # FYI
+ filename = 'ocean_monthly_1x1deg.199301-199712.sosV2.nc' # unneeded, this is mostly for reference
+ full_inputfile=f"{indir}/{filename}"
+
+ # clean up, lest we fool outselves
+ if Path(full_outputfile).exists():
+ Path(full_outputfile).unlink()
+
+ #click.echo('')
+ result = runner.invoke(fre.fre, args = ["cmor", "run",
+ "--indir", indir,
+ "--varlist", varlist,
+ "--table_config", table_config,
+ "--exp_config", exp_config,
+ "--outdir", outdir])
+ click.echo(f'stdout = \n {result.stdout}')
+ #click.echo(f'stderr = \n {result.stderr}') #not captured sep.
+ assert all ( [ result.exit_code == 0,
+ Path(full_outputfile).exists(),
+ Path(full_inputfile).exists() ] )
diff --git a/lib/fre/tests/test_fre_list_cli.py b/lib/fre/tests/test_fre_list_cli.py
new file mode 100644
index 00000000..80c5af2b
--- /dev/null
+++ b/lib/fre/tests/test_fre_list_cli.py
@@ -0,0 +1,22 @@
+''' test "fre list" calls '''
+
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+def test_cli_fre_list():
+ ''' fre list '''
+ result = runner.invoke(fre.fre, args=["list"])
+ assert result.exit_code == 0
+
+def test_cli_fre_list_help():
+ ''' fre list --help '''
+ result = runner.invoke(fre.fre, args=["list", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_list_opt_dne():
+ ''' fre list optionDNE '''
+ result = runner.invoke(fre.fre, args=["list", "optionDNE"])
+ assert result.exit_code == 2
diff --git a/lib/fre/tests/test_fre_make_cli.py b/lib/fre/tests/test_fre_make_cli.py
new file mode 100644
index 00000000..279760b3
--- /dev/null
+++ b/lib/fre/tests/test_fre_make_cli.py
@@ -0,0 +1,62 @@
+''' test "fre make" calls '''
+
+from click.testing import CliRunner
+from pathlib import Path
+import os
+from fre import fre
+
+runner = CliRunner()
+
+def test_cli_fre_make():
+ ''' fre make '''
+ result = runner.invoke(fre.fre, args=["make"])
+ assert result.exit_code == 0
+
+def test_cli_fre_make_help():
+ ''' fre make --help '''
+ result = runner.invoke(fre.fre, args=["make", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_make_opt_dne():
+ ''' fre make optionDNE '''
+ result = runner.invoke(fre.fre, args=["make", "optionDNE"])
+ assert result.exit_code == 2
+
+def test_cli_fre_make_create_checkout_baremetal():
+ ''' fre make create-checkout -y am5.yaml -p ncrc5.intel23 -t debug'''
+ # Set paths and click options
+ test_dir = Path("fre/tests")
+ yamlfile = Path("fre/make/tests/AM5_example/")
+ platform = "ncrc5.intel23"
+ target = "debug"
+
+ # Create output path to test that files exist
+ out_path=f"{test_dir}/fremake_out"
+ Path(out_path).mkdir(parents=True,exist_ok=True)
+
+ # Set HOME for modelRoot location (output location) in fre make
+ os.environ["HOME"]=str(Path(out_path))
+
+ # run create-checkout
+ result = runner.invoke(fre.fre, args=["make", "create-checkout", "-y", f"{yamlfile}/am5.yaml", "-p", platform, "-t", target])
+
+ # Check for successful command, creation of checkout script, and that script is executable (os.access - checks is file has specific access mode, os.X_OK - checks executable permission)
+ assert all ([result.exit_code == 0,
+ Path(f"{out_path}/fremake_canopy/test/am5/src/checkout.sh").exists(),
+ os.access(Path(f"{out_path}/fremake_canopy/test/am5/src/checkout.sh"), os.X_OK)])
+
+def test_cli_fre_make_create_checkout_container():
+ ''' fre make create-checkout -y am5.yaml -p hpcme.2023 -t debug'''
+ # Set paths and click options
+ test_dir = Path("fre/tests")
+ yamlfile = Path("fre/make/tests/AM5_example/")
+ platform = "hpcme.2023"
+ target = "debug"
+
+ # run create-checkout
+ result = runner.invoke(fre.fre, args=["make", "create-checkout", "-y", f"{yamlfile}/am5.yaml", "-p", platform, "-t", target])
+
+ # Check for successful command, creation of checkout script, and that script is executable (os.access - checks is file has specific access mode, os.X_OK - checks executable permission)
+ assert all ([result.exit_code == 0,
+ Path(f"tmp/{platform}/checkout.sh").exists(),
+ os.access(Path(f"tmp/{platform}/checkout.sh"), os.X_OK) == False ])
diff --git a/lib/fre/tests/test_fre_pp_cli.py b/lib/fre/tests/test_fre_pp_cli.py
new file mode 100644
index 00000000..e60d012d
--- /dev/null
+++ b/lib/fre/tests/test_fre_pp_cli.py
@@ -0,0 +1,180 @@
+''' test "fre pp" calls '''
+
+import os
+import shutil
+from pathlib import Path
+
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+
+#-- fre pp
+def test_cli_fre_pp():
+ ''' fre pp '''
+ result = runner.invoke(fre.fre, args=["pp"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_help():
+ ''' fre pp --help '''
+ result = runner.invoke(fre.fre, args=["pp", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_opt_dne():
+ ''' fre pp optionDNE '''
+ result = runner.invoke(fre.fre, args=["pp", "optionDNE"])
+ assert result.exit_code == 2
+
+#-- fre pp checkout
+def test_cli_fre_pp_checkout():
+ ''' fre pp checkout '''
+ result = runner.invoke(fre.fre, args=["pp", "checkout"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_checkout_help():
+ ''' fre pp checkout --help '''
+ result = runner.invoke(fre.fre, args=["pp", "checkout", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_checkout_opt_dne():
+ ''' fre pp checkout optionDNE '''
+ result = runner.invoke(fre.fre, args=["pp", "checkout", "optionDNE"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_checkout_case():
+ ''' fre pp checkout -e FOO -p BAR -t BAZ'''
+ directory = os.path.expanduser("~/cylc-src")+'/FOO__BAR__BAZ'
+ if Path(directory).exists():
+ shutil.rmtree(directory)
+ result = runner.invoke(fre.fre, args=["pp", "checkout",
+ "-e", "FOO",
+ "-p", "BAR",
+ "-t", "BAZ"] )
+ assert all( [ result.exit_code == 0,
+ Path(directory).exists()] )
+
+#-- fre pp configure-xml
+def test_cli_fre_pp_configure_xml():
+ ''' fre pp configure-xml '''
+ result = runner.invoke(fre.fre, args=["pp", "configure-xml"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_configure_xml_help():
+ ''' fre pp configure-xml --help '''
+ result = runner.invoke(fre.fre, args=["pp", "configure-xml", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_configure_xml_opt_dne():
+ ''' fre pp configure-xml optionDNE '''
+ result = runner.invoke(fre.fre, args=["pp", "configure-xml", "optionDNE"])
+ assert result.exit_code == 2
+
+#-- fre pp configure-yaml
+def test_cli_fre_pp_configure_yaml():
+ ''' fre pp configure-yaml '''
+ result = runner.invoke(fre.fre, args=["pp", "configure-yaml"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_configure_yaml_help():
+ ''' fre pp configure-yaml --help '''
+ result = runner.invoke(fre.fre, args=["pp", "configure-yaml", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_configure_yaml_opt_dne():
+ ''' fre pp configure-yaml optionDNE '''
+ result = runner.invoke(fre.fre, args=["pp", "configure-yaml", "optionDNE"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_configure_yaml_fail1():
+ ''' fre pp configure-yaml '''
+ result = runner.invoke(fre.fre, args = [ "pp", "configure-yaml",
+ "-e", "FOO",
+ "-p", "BAR",
+ "-t", "BAZ",
+ "-y", "BOO" ] )
+ assert all( [ result.exit_code == 1,
+ isinstance(result.exception, FileNotFoundError )
+ ] )
+
+
+#-- fre pp install
+def test_cli_fre_pp_install():
+ ''' fre pp install '''
+ result = runner.invoke(fre.fre, args=["pp", "install"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_install_help():
+ ''' fre pp install --help '''
+ result = runner.invoke(fre.fre, args=["pp", "install", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_install_opt_dne():
+ ''' fre pp install optionDNE '''
+ result = runner.invoke(fre.fre, args=["pp", "install", "optionDNE"])
+ assert result.exit_code == 2
+
+#-- fre pp run
+def test_cli_fre_pp_run():
+ ''' fre pp run '''
+ result = runner.invoke(fre.fre, args=["pp", "run"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_run_help():
+ ''' fre pp run --help '''
+ result = runner.invoke(fre.fre, args=["pp", "run", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_run_opt_dne():
+ ''' fre pp run optionDNE '''
+ result = runner.invoke(fre.fre, args=["pp", "run", "optionDNE"])
+ assert result.exit_code == 2
+
+#-- fre pp status
+def test_cli_fre_pp_status():
+ ''' fre pp status '''
+ result = runner.invoke(fre.fre, args=["pp", "status"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_status_help():
+ ''' fre pp status --help '''
+ result = runner.invoke(fre.fre, args=["pp", "status", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_status_opt_dne():
+ ''' fre pp status optionDNE '''
+ result = runner.invoke(fre.fre, args=["pp", "status", "optionDNE"])
+ assert result.exit_code == 2
+
+#-- fre pp validate
+def test_cli_fre_pp_validate():
+ ''' fre pp validate '''
+ result = runner.invoke(fre.fre, args=["pp", "validate"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_validate_help():
+ ''' fre pp validate --help '''
+ result = runner.invoke(fre.fre, args=["pp", "validate", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_validate_opt_dne():
+ ''' fre pp validate optionDNE '''
+ result = runner.invoke(fre.fre, args=["pp", "validate", "optionDNE"])
+ assert result.exit_code == 2
+
+#-- fre pp wrapper
+def test_cli_fre_pp_wrapper():
+ ''' fre pp wrapper '''
+ result = runner.invoke(fre.fre, args=["pp", "wrapper"])
+ assert result.exit_code == 2
+
+def test_cli_fre_pp_wrapper_help():
+ ''' fre pp wrapper --help '''
+ result = runner.invoke(fre.fre, args=["pp", "wrapper", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_pp_wrapper_opt_dne():
+ ''' fre pp wrapper optionDNE '''
+ result = runner.invoke(fre.fre, args=["pp", "wrapper", "optionDNE"])
+ assert result.exit_code == 2
diff --git a/lib/fre/tests/test_fre_run_cli.py b/lib/fre/tests/test_fre_run_cli.py
new file mode 100644
index 00000000..7af6f96f
--- /dev/null
+++ b/lib/fre/tests/test_fre_run_cli.py
@@ -0,0 +1,22 @@
+''' test "fre run" calls '''
+
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+def test_cli_fre_run():
+ ''' fre run '''
+ result = runner.invoke(fre.fre, args=["run"])
+ assert result.exit_code == 0
+
+def test_cli_fre_run_help():
+ ''' fre run --help '''
+ result = runner.invoke(fre.fre, args=["run", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_run_opt_dne():
+ ''' fre run optionDNE '''
+ result = runner.invoke(fre.fre, args=["run", "optionDNE"])
+ assert result.exit_code == 2
diff --git a/lib/fre/tests/test_fre_test_cli.py b/lib/fre/tests/test_fre_test_cli.py
new file mode 100644
index 00000000..faee25d8
--- /dev/null
+++ b/lib/fre/tests/test_fre_test_cli.py
@@ -0,0 +1,22 @@
+''' test "fre test" calls '''
+
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+def test_cli_fre_test():
+ ''' fre test '''
+ result = runner.invoke(fre.fre, args=["test"])
+ assert result.exit_code == 0
+
+def test_cli_fre_test_help():
+ ''' fre test --help '''
+ result = runner.invoke(fre.fre, args=["test", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_test_opt_dne():
+ ''' fre test optionDNE '''
+ result = runner.invoke(fre.fre, args=["test", "optionDNE"])
+ assert result.exit_code == 2
diff --git a/lib/fre/tests/test_fre_yamltools_cli.py b/lib/fre/tests/test_fre_yamltools_cli.py
new file mode 100644
index 00000000..af9481cd
--- /dev/null
+++ b/lib/fre/tests/test_fre_yamltools_cli.py
@@ -0,0 +1,22 @@
+''' test "fre yamltools" calls '''
+
+from click.testing import CliRunner
+
+from fre import fre
+
+runner = CliRunner()
+
+def test_cli_fre_yamltools():
+ ''' fre yamltools '''
+ result = runner.invoke(fre.fre, args=["yamltools"])
+ assert result.exit_code == 0
+
+def test_cli_fre_yamltools_help():
+ ''' fre yamltools --help '''
+ result = runner.invoke(fre.fre, args=["yamltools", "--help"])
+ assert result.exit_code == 0
+
+def test_cli_fre_yamltools_opt_dne():
+ ''' fre yamltools optionDNE '''
+ result = runner.invoke(fre.fre, args=["yamltools", "optionDNE"])
+ assert result.exit_code == 2
diff --git a/lib/fre/yamltools/README.md b/lib/fre/yamltools/README.md
new file mode 100644
index 00000000..f6d896f3
--- /dev/null
+++ b/lib/fre/yamltools/README.md
@@ -0,0 +1,24 @@
+## FRE yamltools
+`fre yamltools` provides subtools that help to manage and perform operations on yaml files.
+
+## Subtools
+- `fre yamltools combine-yamls [options]`
+ - Purpose:
+ - Creates combined yaml file in which the [model].yaml, compile.yaml, and platforms.yaml are merged if `--use compile` is specified
+ - Creates combined yaml file in which the [model].yaml, [experiment].yaml, and [analysis].yaml are merged if `--use pp` is specified
+ - Options:
+ - `-y, --yamlfile [experiment yaml] (required)`
+ - `-p, --platform [platform] (required)`
+ - `-t, --target [target] (required)`
+ - `-e, --experiment [experiment name]`
+ - `--use [compile|pp] (required)`
+
+### **Tests**
+
+To run `fre yamltools` test scripts, return to root directory of the fre-cli repo and call those tests with
+
+ python -m pytest fre/yamltools/tests/[test script.py]
+
+Or run all tests with
+
+ python -m pytest fre/yamltools/tests
diff --git a/lib/fre/yamltools/__init__.py b/lib/fre/yamltools/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/yamltools/combine_yamls.py b/lib/fre/yamltools/combine_yamls.py
new file mode 100644
index 00000000..006bc0d6
--- /dev/null
+++ b/lib/fre/yamltools/combine_yamls.py
@@ -0,0 +1,385 @@
+"""
+Script combines the model yaml with the compile, platform, and experiment yamls.
+"""
+
+## TO-DO:
+# - figure out way to safe_load (yaml_loader=yaml.SafeLoader?)
+# - condition where there are multiple pp and analysis yamls
+
+import os
+import shutil
+
+from pathlib import Path
+import click
+import yaml
+
+def join_constructor(loader, node):
+ """
+ Allows FRE properties defined
+ in main yaml to be concatenated.
+ """
+ seq = loader.construct_sequence(node)
+ return ''.join([str(i) for i in seq])
+
+def yaml_load(yamlfile):
+ """
+ Load the yamlfile
+ """
+ with open(yamlfile, 'r') as yf:
+ y = yaml.load(yf,Loader=yaml.Loader)
+
+ return y
+
+def get_compile_paths(mainyaml_dir,comb):
+ """
+ Extract compile and platform paths from model yaml
+ """
+ comb_model=yaml_load(comb)
+
+ # set platform yaml filepath
+ if comb_model["build"]["platformYaml"] is not None:
+ if Path(os.path.join(mainyaml_dir,comb_model["build"]["platformYaml"])).exists():
+ py=comb_model["build"]["platformYaml"]
+ py_path=Path(os.path.join(mainyaml_dir,py))
+ else:
+ raise ValueError("Incorrect platform yaml path given; does not exist.")
+ else:
+ raise ValueError("No platform yaml path given!")
+ #py_path=None
+
+ # set compile yaml filepath
+ if comb_model["build"]["compileYaml"] is not None:
+ if Path(os.path.join(mainyaml_dir,comb_model["build"]["compileYaml"])).exists():
+ cy=comb_model["build"]["compileYaml"]
+ cy_path=Path(os.path.join(mainyaml_dir,cy))
+ else:
+ raise ValueError("Incorrect compile yaml path given; does not exist.")
+ else:
+ raise ValueError("No compile yaml path given!")
+ #cy_path=None
+
+ return (py_path,cy_path)
+
+
+def experiment_check(mainyaml_dir,comb,experiment):
+ """
+ Check that the experiment given is an experiment listed in the model yaml.
+ Extract experiment specific information and file paths.
+ Arguments:
+ mainyaml_dir : model yaml file
+ comb : combined yaml file name
+ experiment : experiment name
+ """
+ comb_model=yaml_load(comb)
+
+ # Check if exp name given is actually valid experiment listed in combined yaml
+ exp_list = []
+ for i in comb_model.get("experiments"):
+ exp_list.append(i.get("name"))
+
+ if experiment not in exp_list:
+ raise NameError(f"{experiment} is not in the list of experiments")
+
+ # Extract compile yaml path for exp. provided
+ # if experiment matches name in list of experiments in yaml, extract file path
+ for i in comb_model.get("experiments"):
+ if experiment == i.get("name"):
+ expyaml=i.get("pp")
+ analysisyaml=i.get("analysis")
+
+ if expyaml is not None:
+ ey_path=[]
+ for e in expyaml:
+ if Path(os.path.join(mainyaml_dir,e)).exists():
+ ey=Path(os.path.join(mainyaml_dir,e))
+ ey_path.append(ey)
+ else:
+ raise ValueError(f"Incorrect experiment yaml path given ({e}); does not exist.")
+ else:
+ raise ValueError("No experiment yaml path given!")
+
+ if analysisyaml is not None:
+ ay_path=[]
+ for a in analysisyaml:
+ # prepend the directory containing the yaml
+ a = Path(mainyaml_dir, a)
+ if Path(a).exists():
+ ay=Path(os.path.join(mainyaml_dir,a))
+ ay_path.append(ay)
+ else:
+ raise ValueError("Incorrect analysis yaml ath given; does not exist.")
+ else:
+ ay_path=None
+
+ return (ey_path,ay_path)
+
+## COMPILE CLASS ##
+class init_compile_yaml():
+ def __init__(self,yamlfile,platform,target):
+ """
+ Process to combine yamls applicable to compilation
+ """
+ self.yml = yamlfile
+ self.name = yamlfile.split(".")[0]
+ self.namenopath = self.name.split("/")[-1].split(".")[0]
+ self.platform = platform
+ self.target = target
+
+ # Register tag handler
+ yaml.add_constructor('!join', join_constructor)
+
+ # Path to the main model yaml
+ self.mainyaml_dir = os.path.dirname(self.yml)
+
+ # Name of the combined yaml
+ self.combined= f"combined-{self.namenopath}.yaml" if len(self.mainyaml_dir) == 0 else f"{self.mainyaml_dir}/combined-{self.namenopath}.yaml"
+
+ print("Combining yaml files: ")
+
+ def combine_model(self):
+ """
+ Create the combined.yaml and merge it with the model yaml
+ """
+ # copy model yaml info into combined yaml
+ with open(self.combined,'w+',encoding='UTF-8') as f1:
+ f1.write(f'name: &name "{self.name}"\n')
+ f1.write(f'platform: &platform "{self.platform}"\n')
+ f1.write(f'target: &target "{self.target}"\n\n')
+ try:
+ with open(self.yml,'r',encoding='UTF-8') as f2:
+ f1.write("### MODEL YAML SETTINGS ###\n")
+ shutil.copyfileobj(f2,f1)
+ except Exception as exc:
+ raise FileNotFoundError(f'{self.yml} not found') from exc
+ print(f" model yaml: {self.yml}")
+
+ def combine_compile(self):
+ """
+ Combine compile yaml with the defined combined.yaml
+ """
+ # Get compile info
+ (py_path,cy_path) = get_compile_paths(self.mainyaml_dir,self.combined)
+
+ # copy compile yaml info into combined yaml
+ if cy_path is not None:
+ with open(self.combined,'a',encoding='UTF-8') as f1:
+ with open(cy_path,'r',encoding='UTF-8') as f2:
+ f1.write("\n### COMPILE INFO ###\n")
+ shutil.copyfileobj(f2,f1)
+ print(f" compile yaml: {cy_path}")
+
+ def combine_platforms(self):
+ """
+ Combine platforms yaml with the defined combined.yaml
+ """
+ # Get compile info
+ (py_path,cy_path) = get_compile_paths(self.mainyaml_dir,self.combined)
+
+ # combine platform yaml
+ if py_path is not None:
+ with open(self.combined,'a',encoding='UTF-8') as f1:
+ with open(py_path,'r',encoding='UTF-8') as f2:
+ f1.write("\n### PLATFORM INFO ###\n")
+ shutil.copyfileobj(f2,f1)
+ print(f" platforms yaml: {py_path}")
+
+ def clean_yaml(self):
+ """
+ Clean the yaml; remove unnecessary sections in
+ final combined yaml.
+ """
+ # Load the fully combined yaml
+ full_yaml = yaml_load(self.combined)
+
+ # Clean the yaml
+ # If keys exists, delete:
+ keys_clean=["fre_properties", "shared", "experiments"]
+ for kc in keys_clean:
+ if kc in full_yaml.keys():
+ del full_yaml[kc]
+
+ with open(self.combined,'w',encoding='UTF-8') as f:
+ yaml.safe_dump(full_yaml,f,default_flow_style=False,sort_keys=False)
+
+ print(f"Combined yaml located here: {os.path.dirname(self.combined)}/{self.combined}")
+ return self.combined
+
+## PP CLASS ##
+class init_pp_yaml():
+ def __init__(self,yamlfile,experiment,platform,target):
+ """
+ Process to combine the applicable yamls for post-processing
+ """
+ self.yml = yamlfile
+ self.name = experiment
+ self.platform = platform
+ self.target = target
+
+ # Regsiter tag handler
+ yaml.add_constructor('!join', join_constructor)
+
+ # Path to the main model yaml
+ self.mainyaml_dir = os.path.dirname(self.yml)
+
+ # Name of the combined yaml
+ self.combined=f"combined-{self.name}.yaml"
+
+ print("Combining yaml files: ")
+
+ def combine_model(self):
+ """
+ Create the combined.yaml and merge it with the model yaml
+ """
+ # copy model yaml info into combined yaml
+ with open(self.combined,'w+',encoding='UTF-8') as f1:
+ f1.write(f'name: &name "{self.name}"\n')
+ f1.write(f'platform: &platform "{self.platform}"\n')
+ f1.write(f'target: &target "{self.target}"\n\n')
+ try:
+ with open(self.yml,'r',encoding='UTF-8') as f2:
+ f1.write("### MODEL YAML SETTINGS ###\n")
+ shutil.copyfileobj(f2,f1)
+ except Exception as exc:
+ raise FileNotFoundError(f'{self.yml} not found') from exc
+ print(f" model yaml: {self.yml}")
+
+ def combine_experiment(self):
+ """
+ Combine experiment yamls with the defined combined.yaml
+ """
+ # Experiment Check
+ (ey_path,ay_path) = experiment_check(self.mainyaml_dir,self.combined,self.name)
+
+ ## COMBINE EXPERIMENT YAML INFO
+ if ey_path is not None:
+ for i in ey_path:
+ #expyaml_path = os.path.join(mainyaml_dir, i)
+ with open(self.combined,'a',encoding='UTF-8') as f1:
+ with open(i,'r',encoding='UTF-8') as f2:
+ #copy expyaml into combined
+ shutil.copyfileobj(f2,f1)
+ print(f" experiment yaml: {i}")
+
+ def combine_analysis(self):
+ """
+ Combine analysis yamls with the defined combined.yaml
+ """
+ # Experiment Check
+ (ey_path,ay_path) = experiment_check(self.mainyaml_dir,self.combined,self.name)
+
+ ## COMBINE EXPERIMENT YAML INFO
+ if ay_path is not None:
+ for i in ay_path:
+ #analysisyaml_path = os.path.join(mainyaml_dir, i)
+ with open(self.combined,'a',encoding='UTF-8') as f1:
+ with open(i,'r',encoding='UTF-8') as f2:
+ #f1.write(f"\n### {i.upper()} settings ###\n")
+ #copy expyaml into combined
+ shutil.copyfileobj(f2,f1)
+ print(f" analysis yaml: {i}")
+
+ def clean_yaml(self):
+ """
+ Clean the yaml; remove unnecessary sections in
+ final combined yaml.
+ """
+ # Load the fully combined yaml
+ full_yaml = yaml_load(self.combined)
+
+ # Clean the yaml
+ # If keys exists, delete:
+ keys_clean=["fre_properties", "shared", "experiments"]
+ for kc in keys_clean:
+ if kc in full_yaml.keys():
+ del full_yaml[kc]
+
+ with open(self.combined,'w') as f:
+ yaml.safe_dump(full_yaml,f,default_flow_style=False,sort_keys=False)
+
+ print(f"Combined yaml located here: {os.path.dirname(self.combined)}/{self.combined}")
+ return self.combined
+
+## Functions to combine the yaml files ##
+def get_combined_compileyaml(comb):
+ """
+ Combine the model, compile, and platform yamls
+ Arguments:
+ comb : combined yaml object
+ """
+ # Merge model into combined file
+ comb_model = comb.combine_model()
+ # Merge compile.yaml into combined file
+ comb_compile = comb.combine_compile()
+ # Merge platforms.yaml into combined file
+ full_combined = comb.combine_platforms()
+ # Clean the yaml
+ full_combined = comb.clean_yaml()
+
+ return full_combined
+
+def combined_compile_existcheck(combined,yml,platform,target):
+ """
+ Checks for if combined compile yaml exists already.
+ If not, combine model, compile, and platform yamls.
+ """
+ cd = Path.cwd()
+ combined_path=os.path.join(cd,combined)
+
+ # Combine model, compile, and platform yamls
+ # If fre yammltools combine-yamls tools was used, the combined yaml should exist
+ if Path(combined_path).exists():
+ full_combined = combined_path
+ print("\nNOTE: Yamls previously merged.")
+ else:
+ comb = init_compile_yaml(yml,platform,target)
+ full_combined = get_combined_compileyaml(comb)
+
+ return full_combined
+
+def get_combined_ppyaml(comb):
+ """
+ Combine the model, experiment, and analysis yamls
+ Arguments:
+ comb : combined yaml object
+ """
+ # Merge model into combined file
+ comb_model = comb.combine_model()
+ # Merge pp experiment yamls into combined file
+ comb_exp = comb.combine_experiment()
+ # Merge pp analysis yamls, if defined, into combined file
+ comb_analysis = comb.combine_analysis()
+ # Clean the yaml
+ full_combined = comb.clean_yaml()
+
+ return full_combined
+
+###########################################################################################
+def consolidate_yamls(yamlfile,experiment,platform,target,use):
+ # Regsiter tag handler
+ yaml.add_constructor('!join', join_constructor)
+
+ # Path to the main model yaml
+ mainyaml_dir = os.path.dirname(yamlfile)
+
+ if use == "compile":
+ combined = init_compile_yaml(yamlfile, platform, target)
+ # Create combined compile yaml
+ get_combined_compileyaml(combined)
+ elif use =="pp":
+ combined = init_pp_yaml(yamlfile,experiment,platform,target)
+ # Create combined pp yaml
+ get_combined_ppyaml(combined)
+ else:
+ raise ValueError("'use' value is not valid; must be 'compile' or 'pp'")
+
+@click.command()
+def _consolidate_yamls(yamlfile,experiment,platform,target,use):
+ '''
+ Wrapper script for calling yaml_combine - allows the decorated version
+ of the function to be separate from the undecorated version
+ '''
+ return consolidate_yamls(yamlfile,experiment,platform,target,use)
+
+# Use parseyaml function to parse created edits.yaml
+if __name__ == '__main__':
+ consolidate_yamls()
diff --git a/lib/fre/yamltools/freyamltools.py b/lib/fre/yamltools/freyamltools.py
new file mode 100644
index 00000000..55817472
--- /dev/null
+++ b/lib/fre/yamltools/freyamltools.py
@@ -0,0 +1,43 @@
+''' fre yamltools '''
+
+import click
+from .combine_yamls import _consolidate_yamls
+
+@click.group(help=click.style(" - access fre yamltools subcommands", fg=(202,177,95)))
+def yamltools_cli():
+ ''' entry point to fre yamltools click commands '''
+
+@yamltools_cli.command()
+@click.option("-y",
+ "--yamlfile",
+ type=str,
+ help="YAML file to be used for parsing",
+ required=True)
+@click.option("-e",
+ "--experiment",
+ type=str,
+ help="Experiment name")
+@click.option("-p",
+ "--platform",
+ type=str,
+ help="Platform name",
+ required=True)
+@click.option("-t",
+ "--target",
+ type=str,
+ help="Target name",
+ required=True)
+@click.option("--use",
+ type=click.Choice(['compile','pp']),
+ help="Process user is combining yamls for. Can pass 'compile' or 'pp'",
+ required=True)
+@click.pass_context
+def combine_yamls(context,yamlfile,experiment,platform,target,use):
+ """
+ - Combine the model yaml with the compile, platform,
+ experiment, and analysis yamls
+ """
+ context.forward(_consolidate_yamls)
+
+if __name__ == "__main__":
+ yamltools_cli()
diff --git a/lib/fre/yamltools/tests/AM5_example/am5.yaml b/lib/fre/yamltools/tests/AM5_example/am5.yaml
new file mode 100644
index 00000000..c57c997c
--- /dev/null
+++ b/lib/fre/yamltools/tests/AM5_example/am5.yaml
@@ -0,0 +1,72 @@
+# reusable variables
+fre_properties:
+ - &AM5_VERSION "am5f7b12r1"
+ - &FRE_STEM !join [am5/, *AM5_VERSION]
+
+ # amip
+ - &EXP_AMIP_START "1979"
+ - &EXP_AMIP_END "2020"
+ - &ANA_AMIP_START "1980"
+ - &ANA_AMIP_END "2020"
+
+ - &PP_AMIP_CHUNK96 "P1Y"
+ - &PP_AMIP_CHUNK384 "P1Y"
+ - &PP_XYINTERP96 "180,288"
+ - &PP_XYINTERP384 "720,1152"
+
+ # climo
+ - &EXP_CLIMO_START96 "0001"
+ - &EXP_CLIMO_END96 "0011"
+ - &ANA_CLIMO_START96 "0002"
+ - &ANA_CLIMO_END96 "0011"
+
+ - &EXP_CLIMO_START384 "0001"
+ - &EXP_CLIMO_END384 "0006"
+ - &ANA_CLIMO_START384 "0002"
+ - &ANA_CLIMO_END384 "0006"
+
+ # coupled
+ - &PP_CPLD_CHUNK_A "P5Y"
+ - &PP_CPLD_CHUNK_B "P20Y"
+
+ # grids
+ - &GRID_SPEC96 "/archive/oar.gfdl.am5/model_gen5/inputs/c96_grid/c96_OM4_025_grid_No_mg_drag_v20160808.tar"
+
+ # compile information
+ - &release "f1a1r1"
+ - &INTEL "intel-classic"
+ - &FMSincludes "-IFMS/fms2_io/include -IFMS/include -IFMS/mpp/include"
+ - &momIncludes "-Imom6/MOM6-examples/src/MOM6/pkg/CVMix-src/include"
+
+build:
+ # compile information
+ compileYaml: "compile_yamls/compile.yaml"
+ platformYaml: "compile_yamls/platforms.yaml"
+
+shared:
+ # directories shared across tools
+ directories: &shared_directories
+ history_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, history]
+ pp_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, pp]
+ analysis_dir: !join [/nbhome/$USER/, *FRE_STEM, /, *name]
+ ptmp_dir: "/xtmp/$USER/ptmp"
+ fre_analysis_home: "/home/fms/local/opt/fre-analysis/test"
+
+ # shared pp settings
+ postprocess:
+ settings: &shared_settings
+ history_segment: "P1Y"
+ site: "ppan"
+ switches: &shared_switches
+ do_statics: True
+ do_timeavgs: True
+ clean_work: True
+ do_refinediag: False
+ do_atmos_plevel_masking: True
+ do_preanalysis: False
+ do_analysis: True
+
+experiments:
+ - name: "c96L65_am5f7b12r1_amip"
+ pp:
+ - "pp_yamls/pp.c96_amip.yaml"
diff --git a/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile.yaml b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile.yaml
new file mode 100644
index 00000000..5f9a361b
--- /dev/null
+++ b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile.yaml
@@ -0,0 +1,67 @@
+compile:
+ experiment: "am5"
+ container_addlibs:
+ baremetal_linkerflags:
+ src:
+ - component: "FMS"
+ repo: "https://github.com/NOAA-GFDL/FMS.git"
+ cppdefs: "-DINTERNAL_FILE_NML -Duse_libMPI -Duse_netCDF"
+ branch: "2022.01"
+ cppdefs: "-DHAVE_GETTID -Duse_libMPI -Duse_netCDF"
+ otherFlags: *FMSincludes
+ - component: "am5_phys"
+ requires: ["FMS"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/am5_phys.git"
+ branch: "2022.01"
+ otherFlags: *FMSincludes
+ - component: "GFDL_atmos_cubed_sphere"
+ requires: ["FMS", "am5_phys"]
+ repo: "https://github.com/NOAA-GFDL/GFDL_atmos_cubed_sphere.git"
+ cppdefs: "-DSPMD -DCLIMATE_NUDGE -DINTERNAL_FILE_NML"
+ branch: "2022.01"
+ paths: ["GFDL_atmos_cubed_sphere/driver/GFDL",
+ "GFDL_atmos_cubed_sphere/model",
+ "GFDL_atmos_cubed_sphere/driver/SHiELD/cloud_diagnosis.F90",
+ "GFDL_atmos_cubed_sphere/driver/SHiELD/gfdl_cloud_microphys.F90",
+ "GFDL_atmos_cubed_sphere/tools",
+ "GFDL_atmos_cubed_sphere/GFDL_tools"]
+ otherFlags: *FMSincludes
+ - component: "atmos_drivers"
+ requires: ["FMS", "am5_phys", "GFDL_atmos_cubed_sphere"]
+ repo: "https://github.com/NOAA-GFDL/atmos_drivers.git"
+ cppdefs: "-DSPMD -DCLIMATE_NUDGE"
+ branch: "2022.01"
+ paths: ["atmos_drivers/coupled"]
+ otherFlags: *FMSincludes
+ - component: "ice_sis"
+ requires: ["FMS", "ice_param", "mom6"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/ice_sis.git"
+ branch: "2021.02"
+ otherFlags: !join [*FMSincludes, " ", *momIncludes]
+ - component: "ice_param"
+ repo: "https://github.com/NOAA-GFDL/ice_param.git"
+ cppdefs: "-Duse_yaml -Duse_libMPI -Duse_netCDF"
+ branch: "2021.02"
+ requires: ["FMS", "mom6"]
+ otherFlags: !join [*FMSincludes," ", *momIncludes]
+ - component: "land_lad2"
+ requires: ["FMS"]
+ repo: "https://gitlab.gfdl.noaa.gov/FMS/land_lad2.git"
+ branch: "2022.01"
+ branch: "land_lad2_2021.02"
+ doF90Cpp: True
+ cppdefs: "-DINTERNAL_FILE_NML"
+ otherFlags: *FMSincludes
+ - component: "mom6"
+ requires: ["FMS"]
+ paths: ["mom6/MOM6-examples/src/MOM6/config_src/dynamic", "mom6/MOM6-examples/src/MOM6/config_src/coupled_driver", "mom6/MOM6-examples/src/MOM6/src/*/", "mom6/MOM6-examples/src/MOM6/src/*/*/", "mom6/ocean_BGC/generic_tracers", "mom6/ocean_BGC/mocsy/src"]
+ branch: ["2021.02","dev/gfdl/2018.04.06"]
+ repo: ["https://github.com/NOAA-GFDL/ocean_BGC.git","https://github.com/NOAA-GFDL/MOM6-examples.git"]
+ makeOverrides: 'OPENMP=""'
+ otherFlags: !join [*FMSincludes, " ", *momIncludes]
+ - component: "FMScoupler"
+ paths: ["FMScoupler/full", "FMScoupler/shared"]
+ repo: "https://github.com/NOAA-GFDL/FMScoupler.git"
+ branch: "2022.01"
+ requires: ["FMS", "atmos_drivers", "am5_phys", "land_lad2", "ice_sis", "ice_param", "mom6"]
+ otherFlags: !join [*FMSincludes, " ", *momIncludes]
diff --git a/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/am5-wrong_compilefile.yaml b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/am5-wrong_compilefile.yaml
new file mode 100644
index 00000000..86bbb57a
--- /dev/null
+++ b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/am5-wrong_compilefile.yaml
@@ -0,0 +1,72 @@
+# reusable variables
+fre_properties:
+ - &AM5_VERSION "am5f7b12r1"
+ - &FRE_STEM !join [am5/, *AM5_VERSION]
+
+ # amip
+ - &EXP_AMIP_START "1979"
+ - &EXP_AMIP_END "2020"
+ - &ANA_AMIP_START "1980"
+ - &ANA_AMIP_END "2020"
+
+ - &PP_AMIP_CHUNK96 "P1Y"
+ - &PP_AMIP_CHUNK384 "P1Y"
+ - &PP_XYINTERP96 "180,288"
+ - &PP_XYINTERP384 "720,1152"
+
+ # climo
+ - &EXP_CLIMO_START96 "0001"
+ - &EXP_CLIMO_END96 "0011"
+ - &ANA_CLIMO_START96 "0002"
+ - &ANA_CLIMO_END96 "0011"
+
+ - &EXP_CLIMO_START384 "0001"
+ - &EXP_CLIMO_END384 "0006"
+ - &ANA_CLIMO_START384 "0002"
+ - &ANA_CLIMO_END384 "0006"
+
+ # coupled
+ - &PP_CPLD_CHUNK_A "P5Y"
+ - &PP_CPLD_CHUNK_B "P20Y"
+
+ # grids
+ - &GRID_SPEC96 "/archive/oar.gfdl.am5/model_gen5/inputs/c96_grid/c96_OM4_025_grid_No_mg_drag_v20160808.tar"
+
+ # compile information
+ - &release "f1a1r1"
+ - &INTEL "intel-classic"
+ - &FMSincludes "-IFMS/fms2_io/include -IFMS/include -IFMS/mpp/include"
+ - &momIncludes "-Imom6/MOM6-examples/src/MOM6/pkg/CVMix-src/include"
+
+build:
+ # compile information
+ compileYaml: "compiile.yaml"
+ platformYaml: "wrong_platforms.yaml"
+
+shared:
+ # directories shared across tools
+ directories: &shared_directories
+ history_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, history]
+ pp_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, pp]
+ analysis_dir: !join [/nbhome/$USER/, *FRE_STEM, /, *name]
+ ptmp_dir: "/xtmp/$USER/ptmp"
+ fre_analysis_home: "/home/fms/local/opt/fre-analysis/test"
+
+ # shared pp settings
+ postprocess:
+ settings: &shared_settings
+ history_segment: "P1Y"
+ site: "ppan"
+ switches: &shared_switches
+ do_statics: True
+ do_timeavgs: True
+ clean_work: True
+ do_refinediag: False
+ do_atmos_plevel_masking: True
+ do_preanalysis: False
+ do_analysis: True
+
+experiments:
+ - name: "c96L65_am5f7b12r1_amip"
+ pp:
+ -
diff --git a/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/am5-wrong_datatype.yaml b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/am5-wrong_datatype.yaml
new file mode 100644
index 00000000..e65f3bd2
--- /dev/null
+++ b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/am5-wrong_datatype.yaml
@@ -0,0 +1,72 @@
+# reusable variables
+fre_properties:
+ - &AM5_VERSION "am5f7b12r1"
+ - &FRE_STEM !join [am5/, *AM5_VERSION]
+
+ # amip
+ - &EXP_AMIP_START "1979"
+ - &EXP_AMIP_END "2020"
+ - &ANA_AMIP_START "1980"
+ - &ANA_AMIP_END "2020"
+
+ - &PP_AMIP_CHUNK96 "P1Y"
+ - &PP_AMIP_CHUNK384 "P1Y"
+ - &PP_XYINTERP96 "180,288"
+ - &PP_XYINTERP384 "720,1152"
+
+ # climo
+ - &EXP_CLIMO_START96 "0001"
+ - &EXP_CLIMO_END96 "0011"
+ - &ANA_CLIMO_START96 "0002"
+ - &ANA_CLIMO_END96 "0011"
+
+ - &EXP_CLIMO_START384 "0001"
+ - &EXP_CLIMO_END384 "0006"
+ - &ANA_CLIMO_START384 "0002"
+ - &ANA_CLIMO_END384 "0006"
+
+ # coupled
+ - &PP_CPLD_CHUNK_A "P5Y"
+ - &PP_CPLD_CHUNK_B "P20Y"
+
+ # grids
+ - &GRID_SPEC96 "/archive/oar.gfdl.am5/model_gen5/inputs/c96_grid/c96_OM4_025_grid_No_mg_drag_v20160808.tar"
+
+ # compile information
+ - &release "f1a1r1"
+ - &INTEL "intel-classic"
+ - &FMSincludes "-IFMS/fms2_io/include -IFMS/include -IFMS/mpp/include"
+ - &momIncludes "-Imom6/MOM6-examples/src/MOM6/pkg/CVMix-src/include"
+
+build:
+ # compile information
+ compileYaml: "wrong_compile.yaml"
+ platformYaml: "wrong_platforms.yaml"
+
+shared:
+ # directories shared across tools
+ directories: &shared_directories
+ history_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, history]
+ pp_dir: !join [/archive/$USER/, *FRE_STEM, /, *name, /, *platform, -, *target, /, pp]
+ analysis_dir: !join [/nbhome/$USER/, *FRE_STEM, /, *name]
+ ptmp_dir: "/xtmp/$USER/ptmp"
+ fre_analysis_home: "/home/fms/local/opt/fre-analysis/test"
+
+ # shared pp settings
+ postprocess:
+ settings: &shared_settings
+ history_segment: "P1Y"
+ site: "ppan"
+ switches: &shared_switches
+ do_statics: True
+ do_timeavgs: True
+ clean_work: True
+ do_refinediag: False
+ do_atmos_plevel_masking: True
+ do_preanalysis: False
+ do_analysis: True
+
+experiments:
+ - name: "c96L65_am5f7b12r1_amip"
+ pp:
+ -
diff --git a/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/wrong_compile.yaml b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/wrong_compile.yaml
new file mode 100644
index 00000000..c122764a
--- /dev/null
+++ b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/wrong_compile.yaml
@@ -0,0 +1,11 @@
+compile:
+ experiment: "am5"
+ container_addlibs:
+ baremetal_linkerflags:
+ src:
+ - component:
+ repo: "https://github.com/NOAA-GFDL/FMS.git"
+ cppdefs: "-DINTERNAL_FILE_NML -Duse_libMPI -Duse_netCDF"
+ branch: 2022.01
+ cppdefs: "-DHAVE_GETTID -Duse_libMPI -Duse_netCDF"
+ otherFlags: *FMSincludes
diff --git a/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/wrong_platforms.yaml b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/wrong_platforms.yaml
new file mode 100644
index 00000000..7e1b9f49
--- /dev/null
+++ b/lib/fre/yamltools/tests/AM5_example/compile_yamls/compile_fail/wrong_platforms.yaml
@@ -0,0 +1,26 @@
+platforms:
+ - name: ncrc5.intel
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: [ !join [*INTEL, "/2022.2.1"],"fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: ncrc5.intel23
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: [!join [*INTEL, "/2023.1.0"],"fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: hpcme.2023
+ compiler: intel
+ RUNenv: [". /spack/share/spack/setup-env.sh", "spack load libyaml", "spack load netcdf-fortran@4.5.4", "spack load hdf5@1.14.0"]
+ modelRoot: /apps
+ fc: mpiifort
+ cc: mpiicc
+ container: True
+ containerBuild: "podman"
+ containerRun: "apptainer"
diff --git a/lib/fre/yamltools/tests/AM5_example/compile_yamls/platforms.yaml b/lib/fre/yamltools/tests/AM5_example/compile_yamls/platforms.yaml
new file mode 100644
index 00000000..7e1b9f49
--- /dev/null
+++ b/lib/fre/yamltools/tests/AM5_example/compile_yamls/platforms.yaml
@@ -0,0 +1,26 @@
+platforms:
+ - name: ncrc5.intel
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: [ !join [*INTEL, "/2022.2.1"],"fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: ncrc5.intel23
+ compiler: intel
+ modulesInit: [" module use -a /ncrc/home2/fms/local/modulefiles \n","source $MODULESHOME/init/sh \n"]
+ modules: [!join [*INTEL, "/2023.1.0"],"fre/bronx-20",cray-hdf5/1.12.2.3, cray-netcdf/4.9.0.3]
+ fc: ftn
+ cc: cc
+ mkTemplate: "/ncrc/home2/fms/local/opt/fre-commands/bronx-20/site/ncrc5/$(INTEL).mk"
+ modelRoot: ${HOME}/fremake_canopy/test
+ - name: hpcme.2023
+ compiler: intel
+ RUNenv: [". /spack/share/spack/setup-env.sh", "spack load libyaml", "spack load netcdf-fortran@4.5.4", "spack load hdf5@1.14.0"]
+ modelRoot: /apps
+ fc: mpiifort
+ cc: mpiicc
+ container: True
+ containerBuild: "podman"
+ containerRun: "apptainer"
diff --git a/lib/fre/yamltools/tests/AM5_example/pp_yamls/pp.c96_amip.yaml b/lib/fre/yamltools/tests/AM5_example/pp_yamls/pp.c96_amip.yaml
new file mode 100644
index 00000000..117c66c6
--- /dev/null
+++ b/lib/fre/yamltools/tests/AM5_example/pp_yamls/pp.c96_amip.yaml
@@ -0,0 +1,88 @@
+# local reusable variable overrides
+fre_properties:
+ - &custom_interp "180,360"
+
+# directory overrides
+#c96_amip_directories:
+directories:
+ <<: *shared_directories
+ ptmp_dir: "/ptmp/$USER"
+ pp_grid_spec: *GRID_SPEC96
+
+#c96_amip_postprocess:
+postprocess:
+ # pp setting overrides
+ settings:
+ <<: *shared_settings
+ pp_start: *ANA_AMIP_START
+ pp_stop: *ANA_AMIP_END
+ pp_chunk_a: *PP_AMIP_CHUNK96
+ pp_components: "atmos atmos_scalar"
+ switches:
+ <<: *shared_switches
+ do_statics: False
+
+ # main pp instructions
+ components:
+ - type: "atmos_cmip"
+ sources: "atmos_month_cmip atmos_8xdaily_cmip atmos_daily_cmip"
+ sourceGrid: "cubedsphere"
+ xyInterp: *custom_interp
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos"
+ sources: "atmos_month"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_level_cmip"
+ sources: "atmos_level_cmip"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_level"
+ sources: "atmos_month"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_month_aer"
+ sources: "atmos_month_aer"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'atmos'
+ - type: "atmos_diurnal"
+ sources: "atmos_diurnal"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order2"
+ inputRealm: 'atmos'
+ - type: "atmos_scalar"
+ sources: "atmos_scalar"
+ - type: "aerosol_cmip"
+ xyInterp: *PP_XYINTERP96
+ sources: "aerosol_month_cmip"
+ sourceGrid: "cubedsphere"
+ interpMethod: "conserve_order1"
+ inputRealm: 'atmos'
+ - type: "land"
+ sources: "land_month"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'land'
+ - type: "land_cmip"
+ sources: "land_month_cmip"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'land'
+ - type: "tracer_level"
+ sources: "atmos_tracer"
+ sourceGrid: "cubedsphere"
+ xyInterp: *PP_XYINTERP96
+ interpMethod: "conserve_order1"
+ inputRealm: 'atmos'
diff --git a/lib/fre/yamltools/tests/AM5_example/pp_yamls/schema.json b/lib/fre/yamltools/tests/AM5_example/pp_yamls/schema.json
new file mode 100644
index 00000000..dfb9cff5
--- /dev/null
+++ b/lib/fre/yamltools/tests/AM5_example/pp_yamls/schema.json
@@ -0,0 +1,75 @@
+{
+ "$schema": "http://json-schema.org/draft-06/schema#",
+ "title": "Schema for PP Yaml",
+ "type": "object",
+ "properties": {
+ "name": {"type": "string"},
+ "platform": {"type": "string"},
+ "target": {"type": "string"},
+ "directories": {
+ "description": "FRE shared directories",
+ "type": "object",
+ "items":{"$ref": "#/$defs/dirs" }
+ },
+ "postprocess": {
+ "description": "FRE post-processing information",
+ "type": "object",
+ "items":{"$ref": "#/$defs/pp" }
+ }
+ },
+ "$defs": {
+ "dirs": {
+ "history_dir": {"type":"string"},
+ "pp_dir": {"type":"string"},
+ "ptmp_dir": {"type":"string"},
+ "refinediag_scripts":{"type":["string","null"]},
+ "preanalysis_script":{"type":["string","null"]},
+ "history_refined":{"type":["string","null"]},
+ "analysis_dir":{"type":["string","null"]},
+ "pp_grid_spec": {"type":"string"},
+ "fre_analysis_home": {"type":["string","null"]}
+ },
+ "pp": {
+ "type": "object",
+ "properties": {
+ "settings": {
+ "type:": "object",
+ "properties": {
+ "history_segment": {"type":"string"},
+ "site": {"type":"string"},
+ "pp_chunk_a": {"type":"string"},
+ "pp_chunk_b": {"type":"string"},
+ "pp_start": {"type":"string"},
+ "pp_stop": {"type":"string"},
+ "pp_components": {"type":"string"}
+ }
+ },
+ "switches": {
+ "type": "object",
+ "properties": {
+ "clean_work": {"type":"boolean"},
+ "do_mdtf": {"type":"boolean"},
+ "do_statics": {"type":"boolean"},
+ "do_timeavgs": {"type":"boolean"},
+ "do_refinediag": {"type":"boolean"},
+ "do_atmos_plevel_masking": {"type":"boolean"},
+ "do_preanalysis": {"type":"boolean"},
+ "do_analysis": {"type":"boolean"},
+ "do_analysis_only": {"type":"boolean"}
+ }
+ },
+ "components": {
+ "type": "array",
+ "properties": {
+ "type": {"type":"string"},
+ "sources": {"type":"string"},
+ "sourceGrid": {"type":"string"},
+ "xyInterp": {"type":"string"},
+ "interpMethod": {"type":"string"},
+ "inputRealm": {"type":"string"}
+ }
+ }
+ }
+ }
+ }
+}
diff --git a/lib/fre/yamltools/tests/__init__.py b/lib/fre/yamltools/tests/__init__.py
new file mode 100644
index 00000000..e69de29b
diff --git a/lib/fre/yamltools/tests/test_combine_yamls.py b/lib/fre/yamltools/tests/test_combine_yamls.py
new file mode 100644
index 00000000..7df6eb36
--- /dev/null
+++ b/lib/fre/yamltools/tests/test_combine_yamls.py
@@ -0,0 +1,202 @@
+"""
+tests routines in fre.yamltools.combine_yamls
+"""
+import os
+from pathlib import Path
+import pytest
+import shutil
+import json
+import yaml
+from jsonschema import validate
+from fre.yamltools import combine_yamls as cy
+
+
+## SET-UP
+# Set example yaml paths, input directory, output directory
+#CWD = Path.cwd()
+TEST_DIR = Path("fre/yamltools/tests")
+IN_DIR = Path(f"{TEST_DIR}/AM5_example")
+SCHEMA_DIR = Path("fre/gfdl_msd_schemas/FRE")
+
+# Create output directories
+COMP_OUT_DIR = Path(f"{TEST_DIR}/combine_yamls_out/compile")
+PP_OUT_DIR = Path(f"{TEST_DIR}/combine_yamls_out/pp")
+
+# If output directory exists, remove and create again
+for out in [COMP_OUT_DIR, PP_OUT_DIR]:
+ if out.exists():
+ shutil.rmtree(out)
+ Path(out).mkdir(parents=True,exist_ok=True)
+ else:
+ Path(out).mkdir(parents=True,exist_ok=True)
+
+## Set what would be click options
+# Compile
+COMP_EXPERIMENT = "am5"
+COMP_PLATFORM = "ncrc5.intel23"
+COMP_TARGET = "prod"
+
+# Post-processing
+PP_EXPERIMENT = "c96L65_am5f7b12r1_amip"
+PP_PLATFORM = "gfdl.ncrc5-intel22-classic"
+PP_TARGET = "prod"
+
+def test_modelyaml_exists():
+ """
+ Make sure main yaml file exists
+ """
+ assert Path(f"{IN_DIR}/am5.yaml").exists()
+
+def test_compileyaml_exists():
+ """
+ Make sure experiment yaml file exists
+ """
+ assert Path(f"{IN_DIR}/compile_yamls/compile.yaml").exists()
+
+def test_platformyaml_exists():
+ """
+ Make sure experiment yaml file exists
+ """
+ assert Path(f"{IN_DIR}/compile_yamls/platforms.yaml").exists()
+
+def test_merged_compile_yamls():
+ """
+ Check for the creation of the combined-[experiment] yaml
+ Check that the model yaml was merged into the combined yaml
+ """
+ # Model yaml path
+ modelyaml = str(Path(f"{IN_DIR}/am5.yaml"))
+ use = "compile"
+
+ # Merge the yamls
+ cy.consolidate_yamls(modelyaml, COMP_EXPERIMENT, COMP_PLATFORM, COMP_TARGET, use)
+
+ # Move combined yaml to output location
+ shutil.move(f"{IN_DIR}/combined-am5.yaml", COMP_OUT_DIR)
+
+ # Check that the combined yaml exists
+ assert Path(f"{COMP_OUT_DIR}/combined-{COMP_EXPERIMENT}.yaml").exists()
+
+def test_combined_compileyaml_validation():
+ """
+ Validate the combined compile yaml
+ """
+ combined_yamlfile =f"{COMP_OUT_DIR}/combined-{COMP_EXPERIMENT}.yaml"
+ schema_file = os.path.join(SCHEMA_DIR, "fre_make.json")
+
+ with open(combined_yamlfile,'r') as cf:
+ yml = yaml.safe_load(cf)
+
+ with open(schema_file,'r') as f:
+ s = f.read()
+ schema = json.loads(s)
+
+ # If the yaml is valid, no issues
+ # If the yaml is not valid, error
+ try:
+ validate(instance=yml,schema=schema)
+ except:
+ assert False
+
+def test_combined_compileyaml_combinefail():
+ """
+ Check to test if compile yaml is incorrect/does not exist,
+ the combine fails. (compile yaml path misspelled)
+ """
+ # Model yaml path
+ modelyaml = str(Path(f"{IN_DIR}/compile_yamls/compile_fail/am5-wrong_compilefile.yaml"))
+ use = "compile"
+
+ # Merge the yamls - should fail since there is no compile yaml specified in the model yaml
+ try:
+ cy.consolidate_yamls(modelyaml, COMP_EXPERIMENT, COMP_PLATFORM, COMP_TARGET, use)
+ # Move combined yaml to output location
+ shutil.move(f"{IN_DIR}/compile_yamls/compile_fail/combined-am5-wrong_compilefile.yaml", COMP_OUT_DIR)
+ except:
+ print("EXPECTED FAILURE")
+ # Move combined yaml to output location
+ shutil.move(f"{IN_DIR}/compile_yamls/compile_fail/combined-am5-wrong_compilefile.yaml", COMP_OUT_DIR)
+ assert True
+
+def test_combined_compileyaml_validatefail():
+ """
+ Check if the schema is validating correctly
+ Branch should be string
+ """
+ # Model yaml path
+ modelyaml = str(Path(f"{IN_DIR}/compile_yamls/compile_fail/am5-wrong_datatype.yaml"))
+ use = "compile"
+
+ # Merge the yamls
+ cy.consolidate_yamls(modelyaml, COMP_EXPERIMENT, COMP_PLATFORM, COMP_TARGET, use)
+
+ # Move combined yaml to output location
+ shutil.move(f"{IN_DIR}/compile_yamls/compile_fail/combined-am5-wrong_datatype.yaml", COMP_OUT_DIR)
+
+ # Validate against schema; should fail
+ wrong_combined = Path(f"{COMP_OUT_DIR}/combined-am5-wrong_datatype.yaml")
+ schema_file = os.path.join(SCHEMA_DIR, "fre_make.json")
+
+ # Open/load combined yaml file
+ with open(wrong_combined,'r') as cf:
+ yml = yaml.safe_load(cf)
+
+ # Open/load schema.jaon
+ with open(schema_file,'r') as f:
+ s = f.read()
+ schema = json.loads(s)
+
+ # Validation should fail
+ try:
+ validate(instance=yml,schema=schema)
+ except:
+ assert True
+
+############ PP ############
+def test_expyaml_exists():
+ """
+ Make sure experiment yaml file exists
+ """
+ assert Path(f"{IN_DIR}/pp_yamls/pp.c96_amip.yaml").exists()
+
+@pytest.mark.skip(reason='analysis scripts might not be defined yet')
+def test_analysisyaml_exists():
+ """
+ Make sure experiment yaml file exists
+ """
+ assert Path(f"{IN_DIR}/pp_yamls/analysis.yaml").exists()
+
+def test_merged_pp_yamls():
+ """
+ Check for the creation of the combined-[experiment] yaml
+ Check that the model yaml was merged into the combined yaml
+ """
+ # Model yaml path
+ modelyaml = Path(f"{IN_DIR}/am5.yaml")
+ use = "pp"
+
+ # Merge the yamls
+ cy.consolidate_yamls(modelyaml, PP_EXPERIMENT, PP_PLATFORM, PP_TARGET, use)
+
+ # Move combined yaml to output location
+ shutil.move(f"combined-{PP_EXPERIMENT}.yaml", PP_OUT_DIR)
+
+ # Check that the combined yaml exists
+ assert Path(f"{PP_OUT_DIR}/combined-{PP_EXPERIMENT}.yaml").exists()
+
+def test_combined_ppyaml_validation():
+ """
+ Validate the combined compile yaml
+ """
+ combined_yamlfile =f"{PP_OUT_DIR}/combined-{PP_EXPERIMENT}.yaml"
+ schema_dir = Path(f"{IN_DIR}/pp_yamls")
+ schema_file = os.path.join(schema_dir, 'schema.json')
+
+ with open(combined_yamlfile,'r') as cf:
+ yml = yaml.safe_load(cf)
+
+ with open(schema_file,'r') as f:
+ s = f.read()
+ schema = json.loads(s)
+
+ validate(instance=yml,schema=schema)
diff --git a/modules.html b/modules.html
new file mode 100644
index 00000000..6f31b357
--- /dev/null
+++ b/modules.html
@@ -0,0 +1,191 @@
+
+
+
+
+
+
+
+
+ fre — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/objects.inv b/objects.inv
new file mode 100644
index 00000000..fc03a279
Binary files /dev/null and b/objects.inv differ
diff --git a/py-modindex.html b/py-modindex.html
new file mode 100644
index 00000000..b583da43
--- /dev/null
+++ b/py-modindex.html
@@ -0,0 +1,507 @@
+
+
+
+
+
+
+
+ Python Module Index — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+ Python Module Index
+
+
+
+
+
+
+
+
+
+
Python Module Index
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/search.html b/search.html
new file mode 100644
index 00000000..d7741a37
--- /dev/null
+++ b/search.html
@@ -0,0 +1,127 @@
+
+
+
+
+
+
+
+ Search — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+
+
+ Please activate JavaScript to enable the search functionality.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/searchindex.js b/searchindex.js
new file mode 100644
index 00000000..0b4ceb49
--- /dev/null
+++ b/searchindex.js
@@ -0,0 +1 @@
+Search.setIndex({"alltitles": {"API": [[0, null]], "Adding Documentation": [[3, "adding-documentation"]], "Adding New Tools": [[3, "adding-new-tools"]], "Analysis scripts": [[91, "analysis-scripts"], [94, "analysis-scripts"]], "Badges": [[1, null]], "Build FMS model": [[91, "build-fms-model"]], "CMORize postprocessed output": [[91, "cmorize-postprocessed-output"]], "Checklist": [[3, "checklist"]], "Climatologies": [[91, "climatologies"], [94, "climatologies"]], "Compile Yaml": [[91, "compile-yaml"], [91, "id1"], [93, null], [95, "compile-yaml"]], "Contents:": [[81, null]], "Documentation-Documentation": [[2, null]], "Example fre/ Directory Structure": [[3, "example-fre-directory-structure"]], "FMS history files": [[91, "fms-history-files"], [94, null]], "For developers": [[3, null]], "From Other Repositories": [[3, "from-other-repositories"]], "Generate data catalogs": [[91, "generate-data-catalogs"]], "Generic": [[83, "generic"]], "Guide": [[91, "guide"], [93, "guide"]], "How to Contribute to fre-cli\u2019s documentation": [[2, "how-to-contribute-to-fre-cli-s-documentation"]], "Indices": [[81, "indices"]], "Legacy refineDiag scripts": [[91, "legacy-refinediag-scripts"], [94, "legacy-refinediag-scripts"]], "MANIFEST.in": [[3, "manifest-in"]], "Model Yaml": [[91, "model-yaml"], [95, "model-yaml"]], "Module contents": [[4, "module-fre"], [5, "module-fre.app"], [7, "module-fre.app.generate_time_averages"], [12, "module-fre.app.generate_time_averages.tests"], [16, "module-fre.app.regrid_xy"], [18, "module-fre.catalog"], [20, "module-fre.catalog.tests"], [22, "module-fre.check"], [25, "module-fre.cmor"], [28, "module-fre.cmor.tests"], [32, "module-fre.list"], [35, "module-fre.make"], [42, "module-fre.make.tests"], [44, "module-fre.pp"], [52, "module-fre.pp.tests"], [58, "module-fre.run"], [61, "module-fre.test"], [64, "module-fre.tests"], [76, "module-fre.yamltools"], [79, "module-fre.yamltools.tests"]], "Notes on command-line interface": [[84, "notes-on-command-line-interface"]], "On GFDL systems": [[83, "on-gfdl-systems"]], "Other Helpful Things": [[2, "other-helpful-things"]], "Platforms Yaml": [[91, "platforms-yaml"], [95, "platforms-yaml"]], "Post-Processing Yaml": [[91, "post-processing-yaml"], [95, "post-processing-yaml"]], "Postprocess FMS history output": [[91, "postprocess-fms-history-output"]], "Postprocess components": [[91, "postprocess-components"], [94, "postprocess-components"]], "Quickstart": [[91, "quickstart"], [93, "quickstart"]], "Run FMS model": [[91, "run-fms-model"]], "Setup": [[83, null]], "Statics": [[91, "statics"], [94, "statics"]], "Submodules": [[4, "submodules"], [5, "submodules"], [7, "submodules"], [12, "submodules"], [16, "submodules"], [18, "submodules"], [20, "submodules"], [22, "submodules"], [25, "submodules"], [28, "submodules"], [32, "submodules"], [35, "submodules"], [42, "submodules"], [44, "submodules"], [52, "submodules"], [58, "submodules"], [61, "submodules"], [64, "submodules"], [76, "submodules"], [79, "submodules"]], "Subpackages": [[4, "subpackages"], [5, "subpackages"], [7, "subpackages"], [18, "subpackages"], [25, "subpackages"], [35, "subpackages"], [44, "subpackages"], [76, "subpackages"]], "Surface masking for FMS pressure-level history": [[91, "surface-masking-for-fms-pressure-level-history"], [94, "surface-masking-for-fms-pressure-level-history"]], "Timeseries": [[91, "timeseries"], [94, "timeseries"]], "Tools": [[84, null]], "Usage": [[91, null]], "Welcome to fre-cli\u2019s documentation!": [[81, null]], "What is FRE?": [[96, null]], "XY-regridding": [[91, "xy-regridding"], [94, "xy-regridding"]], "YAML Framework": [[91, "yaml-framework"]], "Yaml Formatting": [[91, "yaml-formatting"], [95, null]], "arguments": [[84, "arguments"], [87, "arguments"]], "background": [[84, "background"], [87, "background"]], "builder": [[84, "builder"], [86, null]], "checkout": [[84, "checkout"], [89, "checkout"]], "combine-yamls": [[84, "combine-yamls"], [90, null]], "configure": [[84, "configure"], [89, null]], "create-checkout": [[84, "create-checkout"], [88, null]], "create-compile": [[84, "create-compile"], [88, "create-compile"]], "create-dockerfile": [[84, "create-dockerfile"], [88, "create-dockerfile"]], "create-makefile": [[84, "create-makefile"], [88, "create-makefile"]], "enable workflows for your fork": [[2, "enable-workflows-for-your-fork"]], "examples": [[84, "examples"], [87, "examples"]], "fork and poke at the settings": [[2, "fork-and-poke-at-the-settings"]], "fre": [[82, null]], "fre app": [[84, "fre-app"]], "fre catalog": [[84, "fre-catalog"]], "fre cmor": [[84, "fre-cmor"]], "fre make": [[84, "fre-make"]], "fre package": [[4, null]], "fre pp": [[84, "fre-pp"]], "fre yamltools": [[84, "fre-yamltools"]], "fre.app package": [[5, null]], "fre.app.freapp module": [[6, null]], "fre.app.generate_time_averages package": [[7, null]], "fre.app.generate_time_averages.cdoTimeAverager module": [[8, null]], "fre.app.generate_time_averages.frenctoolsTimeAverager module": [[9, null]], "fre.app.generate_time_averages.frepytoolsTimeAverager module": [[10, null]], "fre.app.generate_time_averages.generate_time_averages module": [[11, null]], "fre.app.generate_time_averages.tests package": [[12, null]], "fre.app.generate_time_averages.tests.test_generate_time_averages module": [[13, null]], "fre.app.generate_time_averages.timeAverager module": [[14, null]], "fre.app.mask_atmos_plevel module": [[15, null]], "fre.app.regrid_xy package": [[16, null]], "fre.app.regrid_xy.regrid_xy module": [[17, null]], "fre.catalog package": [[18, null]], "fre.catalog.frecatalog module": [[19, null]], "fre.catalog.tests package": [[20, null]], "fre.catalog.tests.test_fre_catalog module": [[21, null]], "fre.check package": [[22, null]], "fre.check.frecheck module": [[23, null]], "fre.check.frecheckexample module": [[24, null]], "fre.cmor package": [[25, null]], "fre.cmor.cmor_mixer module": [[26, null]], "fre.cmor.frecmor module": [[27, null]], "fre.cmor.tests package": [[28, null]], "fre.cmor.tests.test_cmor_run_subtool module": [[29, null]], "fre.fre module": [[30, null]], "fre.lazy_group module": [[31, null]], "fre.list package": [[32, null]], "fre.list.frelist module": [[33, null]], "fre.list.frelistexample module": [[34, null]], "fre.make package": [[35, null]], "fre.make.createCheckout module": [[36, null]], "fre.make.createCompile module": [[37, null]], "fre.make.createDocker module": [[38, null]], "fre.make.createMakefile module": [[39, null]], "fre.make.fremake module": [[40, null]], "fre.make.runFremake module": [[41, null]], "fre.make.tests package": [[42, null]], "fre.make.tests.test_create_makefile module": [[43, null]], "fre.pp package": [[44, null]], "fre.pp.checkout_script module": [[45, null]], "fre.pp.configure_script_xml module": [[46, null]], "fre.pp.configure_script_yaml module": [[47, null]], "fre.pp.frepp module": [[48, null]], "fre.pp.install module": [[49, null]], "fre.pp.run module": [[50, null]], "fre.pp.status module": [[51, null]], "fre.pp.tests package": [[52, null]], "fre.pp.tests.test_configure_script_yaml module": [[53, null]], "fre.pp.tests.test_rose_quoting module": [[54, null]], "fre.pp.trigger module": [[55, null]], "fre.pp.validate module": [[56, null]], "fre.pp.wrapper module": [[57, null]], "fre.run package": [[58, null]], "fre.run.frerun module": [[59, null]], "fre.run.frerunexample module": [[60, null]], "fre.test package": [[61, null]], "fre.test.fretest module": [[62, null]], "fre.test.fretestexample module": [[63, null]], "fre.tests package": [[64, null]], "fre.tests.test_fre_app_cli module": [[65, null]], "fre.tests.test_fre_catalog_cli module": [[66, null]], "fre.tests.test_fre_check_cli module": [[67, null]], "fre.tests.test_fre_cli module": [[68, null]], "fre.tests.test_fre_cmor_cli module": [[69, null]], "fre.tests.test_fre_list_cli module": [[70, null]], "fre.tests.test_fre_make_cli module": [[71, null]], "fre.tests.test_fre_pp_cli module": [[72, null]], "fre.tests.test_fre_run_cli module": [[73, null]], "fre.tests.test_fre_test_cli module": [[74, null]], "fre.tests.test_fre_yamltools_cli module": [[75, null]], "fre.yamltools package": [[76, null]], "fre.yamltools.combine_yamls module": [[77, null]], "fre.yamltools.freyamltools module": [[78, null]], "fre.yamltools.tests package": [[79, null]], "fre.yamltools.tests.test_combine_yamls module": [[80, null]], "run": [[84, "run"], [87, null]], "run your fork\u2019s first workflow": [[2, "run-your-fork-s-first-workflow"]], "run-fremake": [[84, "run-fremake"], [88, "run-fremake"]], "validate": [[84, "validate"], [86, "validate"]]}, "docnames": ["api", "badges", "contributing_to_doc", "for-developers", "fre", "fre.app", "fre.app.freapp", "fre.app.generate_time_averages", "fre.app.generate_time_averages.cdoTimeAverager", "fre.app.generate_time_averages.frenctoolsTimeAverager", "fre.app.generate_time_averages.frepytoolsTimeAverager", "fre.app.generate_time_averages.generate_time_averages", "fre.app.generate_time_averages.tests", "fre.app.generate_time_averages.tests.test_generate_time_averages", "fre.app.generate_time_averages.timeAverager", "fre.app.mask_atmos_plevel", "fre.app.regrid_xy", "fre.app.regrid_xy.regrid_xy", "fre.catalog", "fre.catalog.frecatalog", "fre.catalog.tests", "fre.catalog.tests.test_fre_catalog", "fre.check", "fre.check.frecheck", "fre.check.frecheckexample", "fre.cmor", "fre.cmor.cmor_mixer", "fre.cmor.frecmor", "fre.cmor.tests", "fre.cmor.tests.test_cmor_run_subtool", "fre.fre", "fre.lazy_group", "fre.list", "fre.list.frelist", "fre.list.frelistexample", "fre.make", "fre.make.createCheckout", "fre.make.createCompile", "fre.make.createDocker", "fre.make.createMakefile", "fre.make.fremake", "fre.make.runFremake", "fre.make.tests", "fre.make.tests.test_create_makefile", "fre.pp", "fre.pp.checkout_script", "fre.pp.configure_script_xml", "fre.pp.configure_script_yaml", "fre.pp.frepp", "fre.pp.install", "fre.pp.run", "fre.pp.status", "fre.pp.tests", "fre.pp.tests.test_configure_script_yaml", "fre.pp.tests.test_rose_quoting", "fre.pp.trigger", "fre.pp.validate", "fre.pp.wrapper", "fre.run", "fre.run.frerun", "fre.run.frerunexample", "fre.test", "fre.test.fretest", "fre.test.fretestexample", "fre.tests", "fre.tests.test_fre_app_cli", "fre.tests.test_fre_catalog_cli", "fre.tests.test_fre_check_cli", "fre.tests.test_fre_cli", "fre.tests.test_fre_cmor_cli", "fre.tests.test_fre_list_cli", "fre.tests.test_fre_make_cli", "fre.tests.test_fre_pp_cli", "fre.tests.test_fre_run_cli", "fre.tests.test_fre_test_cli", "fre.tests.test_fre_yamltools_cli", "fre.yamltools", "fre.yamltools.combine_yamls", "fre.yamltools.freyamltools", "fre.yamltools.tests", "fre.yamltools.tests.test_combine_yamls", "index", "modules", "setup", "tools", "tools/app", "tools/catalog", "tools/cmor", "tools/make", "tools/pp", "tools/yamltools", "usage", "usage/cmor", "usage/compile", "usage/postprocess", "usage/yaml_framework", "what-is-fre"], "envversion": {"sphinx": 62, "sphinx.domains.c": 3, "sphinx.domains.changeset": 1, "sphinx.domains.citation": 1, "sphinx.domains.cpp": 9, "sphinx.domains.index": 1, "sphinx.domains.javascript": 3, "sphinx.domains.math": 2, "sphinx.domains.python": 4, "sphinx.domains.rst": 2, "sphinx.domains.std": 2}, "filenames": ["api.rst", "badges.rst", "contributing_to_doc.rst", "for-developers.rst", "fre.rst", "fre.app.rst", "fre.app.freapp.rst", "fre.app.generate_time_averages.rst", "fre.app.generate_time_averages.cdoTimeAverager.rst", "fre.app.generate_time_averages.frenctoolsTimeAverager.rst", "fre.app.generate_time_averages.frepytoolsTimeAverager.rst", "fre.app.generate_time_averages.generate_time_averages.rst", "fre.app.generate_time_averages.tests.rst", "fre.app.generate_time_averages.tests.test_generate_time_averages.rst", "fre.app.generate_time_averages.timeAverager.rst", "fre.app.mask_atmos_plevel.rst", "fre.app.regrid_xy.rst", "fre.app.regrid_xy.regrid_xy.rst", "fre.catalog.rst", "fre.catalog.frecatalog.rst", "fre.catalog.tests.rst", "fre.catalog.tests.test_fre_catalog.rst", "fre.check.rst", "fre.check.frecheck.rst", "fre.check.frecheckexample.rst", "fre.cmor.rst", "fre.cmor.cmor_mixer.rst", "fre.cmor.frecmor.rst", "fre.cmor.tests.rst", "fre.cmor.tests.test_cmor_run_subtool.rst", "fre.fre.rst", "fre.lazy_group.rst", "fre.list.rst", "fre.list.frelist.rst", "fre.list.frelistexample.rst", "fre.make.rst", "fre.make.createCheckout.rst", "fre.make.createCompile.rst", "fre.make.createDocker.rst", "fre.make.createMakefile.rst", "fre.make.fremake.rst", "fre.make.runFremake.rst", "fre.make.tests.rst", "fre.make.tests.test_create_makefile.rst", "fre.pp.rst", "fre.pp.checkout_script.rst", "fre.pp.configure_script_xml.rst", "fre.pp.configure_script_yaml.rst", "fre.pp.frepp.rst", "fre.pp.install.rst", "fre.pp.run.rst", "fre.pp.status.rst", "fre.pp.tests.rst", "fre.pp.tests.test_configure_script_yaml.rst", "fre.pp.tests.test_rose_quoting.rst", "fre.pp.trigger.rst", "fre.pp.validate.rst", "fre.pp.wrapper.rst", "fre.run.rst", "fre.run.frerun.rst", "fre.run.frerunexample.rst", "fre.test.rst", "fre.test.fretest.rst", "fre.test.fretestexample.rst", "fre.tests.rst", "fre.tests.test_fre_app_cli.rst", "fre.tests.test_fre_catalog_cli.rst", "fre.tests.test_fre_check_cli.rst", "fre.tests.test_fre_cli.rst", "fre.tests.test_fre_cmor_cli.rst", "fre.tests.test_fre_list_cli.rst", "fre.tests.test_fre_make_cli.rst", "fre.tests.test_fre_pp_cli.rst", "fre.tests.test_fre_run_cli.rst", "fre.tests.test_fre_test_cli.rst", "fre.tests.test_fre_yamltools_cli.rst", "fre.yamltools.rst", "fre.yamltools.combine_yamls.rst", "fre.yamltools.freyamltools.rst", "fre.yamltools.tests.rst", "fre.yamltools.tests.test_combine_yamls.rst", "index.rst", "modules.rst", "setup.rst", "tools.rst", "tools/app.rst", "tools/catalog.rst", "tools/cmor.rst", "tools/make.rst", "tools/pp.rst", "tools/yamltools.rst", "usage.rst", "usage/cmor.rst", "usage/compile.rst", "usage/postprocess.rst", "usage/yaml_framework.rst", "what-is-fre.rst"], "indexentries": {"avg_type (fre.app.generate_time_averages.timeaverager.timeaverager attribute)": [[14, "fre.app.generate_time_averages.timeAverager.timeAverager.avg_type", false]], "cdotimeaverager (class in fre.app.generate_time_averages.cdotimeaverager)": [[8, "fre.app.generate_time_averages.cdoTimeAverager.cdoTimeAverager", false]], "check_dataset_for_ocean_grid() (in module fre.cmor.cmor_mixer)": [[26, "fre.cmor.cmor_mixer.check_dataset_for_ocean_grid", false]], "check_interp_method() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.check_interp_method", false]], "check_per_component_settings() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.check_per_component_settings", false]], "checkout_create() (in module fre.make.createcheckout)": [[36, "fre.make.createCheckout.checkout_create", false]], "checkout_template() (in module fre.pp.checkout_script)": [[45, "fre.pp.checkout_script.checkout_template", false]], "chunk_from_legacy() (in module fre.pp.configure_script_xml)": [[46, "fre.pp.configure_script_xml.chunk_from_legacy", false]], "clean_yaml() (fre.yamltools.combine_yamls.init_compile_yaml method)": [[77, "fre.yamltools.combine_yamls.init_compile_yaml.clean_yaml", false]], "clean_yaml() (fre.yamltools.combine_yamls.init_pp_yaml method)": [[77, "fre.yamltools.combine_yamls.init_pp_yaml.clean_yaml", false]], "cmor_run_subtool() (in module fre.cmor.cmor_mixer)": [[26, "fre.cmor.cmor_mixer.cmor_run_subtool", false]], "cmorize_target_var_files() (in module fre.cmor.cmor_mixer)": [[26, "fre.cmor.cmor_mixer.cmorize_target_var_files", false]], "combine_analysis() (fre.yamltools.combine_yamls.init_pp_yaml method)": [[77, "fre.yamltools.combine_yamls.init_pp_yaml.combine_analysis", false]], "combine_compile() (fre.yamltools.combine_yamls.init_compile_yaml method)": [[77, "fre.yamltools.combine_yamls.init_compile_yaml.combine_compile", false]], "combine_experiment() (fre.yamltools.combine_yamls.init_pp_yaml method)": [[77, "fre.yamltools.combine_yamls.init_pp_yaml.combine_experiment", false]], "combine_model() (fre.yamltools.combine_yamls.init_compile_yaml method)": [[77, "fre.yamltools.combine_yamls.init_compile_yaml.combine_model", false]], "combine_model() (fre.yamltools.combine_yamls.init_pp_yaml method)": [[77, "fre.yamltools.combine_yamls.init_pp_yaml.combine_model", false]], "combine_platforms() (fre.yamltools.combine_yamls.init_compile_yaml method)": [[77, "fre.yamltools.combine_yamls.init_compile_yaml.combine_platforms", false]], "combined_compile_existcheck() (in module fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.combined_compile_existcheck", false]], "compile_create() (in module fre.make.createcompile)": [[37, "fre.make.createCompile.compile_create", false]], "consolidate_yamls() (in module fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.consolidate_yamls", false]], "copy_nc() (in module fre.cmor.cmor_mixer)": [[26, "fre.cmor.cmor_mixer.copy_nc", false]], "create_tmp_dir() (in module fre.cmor.cmor_mixer)": [[26, "fre.cmor.cmor_mixer.create_tmp_dir", false]], "dockerfile_create() (in module fre.make.createdocker)": [[38, "fre.make.createDocker.dockerfile_create", false]], "duration_to_seconds() (in module fre.pp.configure_script_xml)": [[46, "fre.pp.configure_script_xml.duration_to_seconds", false]], "experiment_check() (in module fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.experiment_check", false]], "fre": [[4, "module-fre", false]], "fre.app": [[5, "module-fre.app", false]], "fre.app.freapp": [[6, "module-fre.app.freapp", false]], "fre.app.generate_time_averages": [[7, "module-fre.app.generate_time_averages", false]], "fre.app.generate_time_averages.cdotimeaverager": [[8, "module-fre.app.generate_time_averages.cdoTimeAverager", false]], "fre.app.generate_time_averages.frenctoolstimeaverager": [[9, "module-fre.app.generate_time_averages.frenctoolsTimeAverager", false]], "fre.app.generate_time_averages.frepytoolstimeaverager": [[10, "module-fre.app.generate_time_averages.frepytoolsTimeAverager", false]], "fre.app.generate_time_averages.generate_time_averages": [[11, "module-fre.app.generate_time_averages.generate_time_averages", false]], "fre.app.generate_time_averages.tests": [[12, "module-fre.app.generate_time_averages.tests", false]], "fre.app.generate_time_averages.tests.test_generate_time_averages": [[13, "module-fre.app.generate_time_averages.tests.test_generate_time_averages", false]], "fre.app.generate_time_averages.timeaverager": [[14, "module-fre.app.generate_time_averages.timeAverager", false]], "fre.app.mask_atmos_plevel": [[15, "module-fre.app.mask_atmos_plevel", false]], "fre.app.regrid_xy": [[16, "module-fre.app.regrid_xy", false]], "fre.app.regrid_xy.regrid_xy": [[17, "module-fre.app.regrid_xy.regrid_xy", false]], "fre.catalog": [[18, "module-fre.catalog", false]], "fre.catalog.frecatalog": [[19, "module-fre.catalog.frecatalog", false]], "fre.catalog.tests": [[20, "module-fre.catalog.tests", false]], "fre.catalog.tests.test_fre_catalog": [[21, "module-fre.catalog.tests.test_fre_catalog", false]], "fre.check": [[22, "module-fre.check", false]], "fre.check.frecheck": [[23, "module-fre.check.frecheck", false]], "fre.check.frecheckexample": [[24, "module-fre.check.frecheckexample", false]], "fre.cmor": [[25, "module-fre.cmor", false]], "fre.cmor.cmor_mixer": [[26, "module-fre.cmor.cmor_mixer", false]], "fre.cmor.frecmor": [[27, "module-fre.cmor.frecmor", false]], "fre.cmor.tests": [[28, "module-fre.cmor.tests", false]], "fre.cmor.tests.test_cmor_run_subtool": [[29, "module-fre.cmor.tests.test_cmor_run_subtool", false]], "fre.fre": [[30, "module-fre.fre", false]], "fre.lazy_group": [[31, "module-fre.lazy_group", false]], "fre.list": [[32, "module-fre.list", false]], "fre.list.frelist": [[33, "module-fre.list.frelist", false]], "fre.list.frelistexample": [[34, "module-fre.list.frelistexample", false]], "fre.make": [[35, "module-fre.make", false]], "fre.make.createcheckout": [[36, "module-fre.make.createCheckout", false]], "fre.make.createcompile": [[37, "module-fre.make.createCompile", false]], "fre.make.createdocker": [[38, "module-fre.make.createDocker", false]], "fre.make.createmakefile": [[39, "module-fre.make.createMakefile", false]], "fre.make.fremake": [[40, "module-fre.make.fremake", false]], "fre.make.runfremake": [[41, "module-fre.make.runFremake", false]], "fre.make.tests": [[42, "module-fre.make.tests", false]], "fre.make.tests.test_create_makefile": [[43, "module-fre.make.tests.test_create_makefile", false]], "fre.pp": [[44, "module-fre.pp", false]], "fre.pp.checkout_script": [[45, "module-fre.pp.checkout_script", false]], "fre.pp.configure_script_xml": [[46, "module-fre.pp.configure_script_xml", false]], "fre.pp.configure_script_yaml": [[47, "module-fre.pp.configure_script_yaml", false]], "fre.pp.frepp": [[48, "module-fre.pp.frepp", false]], "fre.pp.install": [[49, "module-fre.pp.install", false]], "fre.pp.run": [[50, "module-fre.pp.run", false]], "fre.pp.status": [[51, "module-fre.pp.status", false]], "fre.pp.tests": [[52, "module-fre.pp.tests", false]], "fre.pp.tests.test_configure_script_yaml": [[53, "module-fre.pp.tests.test_configure_script_yaml", false]], "fre.pp.tests.test_rose_quoting": [[54, "module-fre.pp.tests.test_rose_quoting", false]], "fre.pp.trigger": [[55, "module-fre.pp.trigger", false]], "fre.pp.validate": [[56, "module-fre.pp.validate", false]], "fre.pp.wrapper": [[57, "module-fre.pp.wrapper", false]], "fre.run": [[58, "module-fre.run", false]], "fre.run.frerun": [[59, "module-fre.run.frerun", false]], "fre.run.frerunexample": [[60, "module-fre.run.frerunexample", false]], "fre.test": [[61, "module-fre.test", false]], "fre.test.fretest": [[62, "module-fre.test.fretest", false]], "fre.test.fretestexample": [[63, "module-fre.test.fretestexample", false]], "fre.tests": [[64, "module-fre.tests", false]], "fre.tests.test_fre_app_cli": [[65, "module-fre.tests.test_fre_app_cli", false]], "fre.tests.test_fre_catalog_cli": [[66, "module-fre.tests.test_fre_catalog_cli", false]], "fre.tests.test_fre_check_cli": [[67, "module-fre.tests.test_fre_check_cli", false]], "fre.tests.test_fre_cli": [[68, "module-fre.tests.test_fre_cli", false]], "fre.tests.test_fre_cmor_cli": [[69, "module-fre.tests.test_fre_cmor_cli", false]], "fre.tests.test_fre_list_cli": [[70, "module-fre.tests.test_fre_list_cli", false]], "fre.tests.test_fre_make_cli": [[71, "module-fre.tests.test_fre_make_cli", false]], "fre.tests.test_fre_pp_cli": [[72, "module-fre.tests.test_fre_pp_cli", false]], "fre.tests.test_fre_run_cli": [[73, "module-fre.tests.test_fre_run_cli", false]], "fre.tests.test_fre_test_cli": [[74, "module-fre.tests.test_fre_test_cli", false]], "fre.tests.test_fre_yamltools_cli": [[75, "module-fre.tests.test_fre_yamltools_cli", false]], "fre.yamltools": [[76, "module-fre.yamltools", false]], "fre.yamltools.combine_yamls": [[77, "module-fre.yamltools.combine_yamls", false]], "fre.yamltools.freyamltools": [[78, "module-fre.yamltools.freyamltools", false]], "fre.yamltools.tests": [[79, "module-fre.yamltools.tests", false]], "fre.yamltools.tests.test_combine_yamls": [[80, "module-fre.yamltools.tests.test_combine_yamls", false]], "frelist_xpath() (in module fre.pp.configure_script_xml)": [[46, "fre.pp.configure_script_xml.frelist_xpath", false]], "fremake_run() (in module fre.make.runfremake)": [[41, "fre.make.runFremake.fremake_run", false]], "frenctoolstimeaverager (class in fre.app.generate_time_averages.frenctoolstimeaverager)": [[9, "fre.app.generate_time_averages.frenctoolsTimeAverager.frenctoolsTimeAverager", false]], "frepytoolstimeaverager (class in fre.app.generate_time_averages.frepytoolstimeaverager)": [[10, "fre.app.generate_time_averages.frepytoolsTimeAverager.frepytoolsTimeAverager", false]], "freq_from_legacy() (in module fre.pp.configure_script_xml)": [[46, "fre.pp.configure_script_xml.freq_from_legacy", false]], "freq_to_date_format() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.freq_to_date_format", false]], "generate_timavg() (fre.app.generate_time_averages.cdotimeaverager.cdotimeaverager method)": [[8, "fre.app.generate_time_averages.cdoTimeAverager.cdoTimeAverager.generate_timavg", false]], "generate_timavg() (fre.app.generate_time_averages.frenctoolstimeaverager.frenctoolstimeaverager method)": [[9, "fre.app.generate_time_averages.frenctoolsTimeAverager.frenctoolsTimeAverager.generate_timavg", false]], "generate_timavg() (fre.app.generate_time_averages.frepytoolstimeaverager.frepytoolstimeaverager method)": [[10, "fre.app.generate_time_averages.frepytoolsTimeAverager.frepytoolsTimeAverager.generate_timavg", false]], "generate_timavg() (fre.app.generate_time_averages.timeaverager.timeaverager method)": [[14, "fre.app.generate_time_averages.timeAverager.timeAverager.generate_timavg", false]], "generate_time_average() (in module fre.app.generate_time_averages.generate_time_averages)": [[11, "fre.app.generate_time_averages.generate_time_averages.generate_time_average", false]], "get_combined_compileyaml() (in module fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.get_combined_compileyaml", false]], "get_combined_ppyaml() (in module fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.get_combined_ppyaml", false]], "get_command() (fre.lazy_group.lazygroup method)": [[31, "fre.lazy_group.LazyGroup.get_command", false]], "get_compile_paths() (in module fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.get_compile_paths", false]], "get_iso_datetimes() (in module fre.cmor.cmor_mixer)": [[26, "fre.cmor.cmor_mixer.get_iso_datetimes", false]], "get_mosaic_file_name() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.get_mosaic_file_name", false]], "get_mosaic_grid_file_name() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.get_mosaic_grid_file_name", false]], "get_var_filenames() (in module fre.cmor.cmor_mixer)": [[26, "fre.cmor.cmor_mixer.get_var_filenames", false]], "get_vertical_dimension() (in module fre.cmor.cmor_mixer)": [[26, "fre.cmor.cmor_mixer.get_vertical_dimension", false]], "init_compile_yaml (class in fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.init_compile_yaml", false]], "init_pp_yaml (class in fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.init_pp_yaml", false]], "install_subtool() (in module fre.pp.install)": [[49, "fre.pp.install.install_subtool", false]], "join_constructor() (in module fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.join_constructor", false]], "lazygroup (class in fre.lazy_group)": [[31, "fre.lazy_group.LazyGroup", false]], "list_commands() (fre.lazy_group.lazygroup method)": [[31, "fre.lazy_group.LazyGroup.list_commands", false]], "main() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.main", false]], "main() (in module fre.pp.configure_script_xml)": [[46, "fre.pp.configure_script_xml.main", false]], "make_component_list() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.make_component_list", false]], "make_regrid_var_list() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.make_regrid_var_list", false]], "makefile_create() (in module fre.make.createmakefile)": [[39, "fre.make.createMakefile.makefile_create", false]], "mask_field_above_surface_pressure() (in module fre.app.mask_atmos_plevel)": [[15, "fre.app.mask_atmos_plevel.mask_field_above_surface_pressure", false]], "module": [[4, "module-fre", false], [5, "module-fre.app", false], [6, "module-fre.app.freapp", false], [7, "module-fre.app.generate_time_averages", false], [8, "module-fre.app.generate_time_averages.cdoTimeAverager", false], [9, "module-fre.app.generate_time_averages.frenctoolsTimeAverager", false], [10, "module-fre.app.generate_time_averages.frepytoolsTimeAverager", false], [11, "module-fre.app.generate_time_averages.generate_time_averages", false], [12, "module-fre.app.generate_time_averages.tests", false], [13, "module-fre.app.generate_time_averages.tests.test_generate_time_averages", false], [14, "module-fre.app.generate_time_averages.timeAverager", false], [15, "module-fre.app.mask_atmos_plevel", false], [16, "module-fre.app.regrid_xy", false], [17, "module-fre.app.regrid_xy.regrid_xy", false], [18, "module-fre.catalog", false], [19, "module-fre.catalog.frecatalog", false], [20, "module-fre.catalog.tests", false], [21, "module-fre.catalog.tests.test_fre_catalog", false], [22, "module-fre.check", false], [23, "module-fre.check.frecheck", false], [24, "module-fre.check.frecheckexample", false], [25, "module-fre.cmor", false], [26, "module-fre.cmor.cmor_mixer", false], [27, "module-fre.cmor.frecmor", false], [28, "module-fre.cmor.tests", false], [29, "module-fre.cmor.tests.test_cmor_run_subtool", false], [30, "module-fre.fre", false], [31, "module-fre.lazy_group", false], [32, "module-fre.list", false], [33, "module-fre.list.frelist", false], [34, "module-fre.list.frelistexample", false], [35, "module-fre.make", false], [36, "module-fre.make.createCheckout", false], [37, "module-fre.make.createCompile", false], [38, "module-fre.make.createDocker", false], [39, "module-fre.make.createMakefile", false], [40, "module-fre.make.fremake", false], [41, "module-fre.make.runFremake", false], [42, "module-fre.make.tests", false], [43, "module-fre.make.tests.test_create_makefile", false], [44, "module-fre.pp", false], [45, "module-fre.pp.checkout_script", false], [46, "module-fre.pp.configure_script_xml", false], [47, "module-fre.pp.configure_script_yaml", false], [48, "module-fre.pp.frepp", false], [49, "module-fre.pp.install", false], [50, "module-fre.pp.run", false], [51, "module-fre.pp.status", false], [52, "module-fre.pp.tests", false], [53, "module-fre.pp.tests.test_configure_script_yaml", false], [54, "module-fre.pp.tests.test_rose_quoting", false], [55, "module-fre.pp.trigger", false], [56, "module-fre.pp.validate", false], [57, "module-fre.pp.wrapper", false], [58, "module-fre.run", false], [59, "module-fre.run.frerun", false], [60, "module-fre.run.frerunexample", false], [61, "module-fre.test", false], [62, "module-fre.test.fretest", false], [63, "module-fre.test.fretestexample", false], [64, "module-fre.tests", false], [65, "module-fre.tests.test_fre_app_cli", false], [66, "module-fre.tests.test_fre_catalog_cli", false], [67, "module-fre.tests.test_fre_check_cli", false], [68, "module-fre.tests.test_fre_cli", false], [69, "module-fre.tests.test_fre_cmor_cli", false], [70, "module-fre.tests.test_fre_list_cli", false], [71, "module-fre.tests.test_fre_make_cli", false], [72, "module-fre.tests.test_fre_pp_cli", false], [73, "module-fre.tests.test_fre_run_cli", false], [74, "module-fre.tests.test_fre_test_cli", false], [75, "module-fre.tests.test_fre_yamltools_cli", false], [76, "module-fre.yamltools", false], [77, "module-fre.yamltools.combine_yamls", false], [78, "module-fre.yamltools.freyamltools", false], [79, "module-fre.yamltools.tests", false], [80, "module-fre.yamltools.tests.test_combine_yamls", false]], "pkg (fre.app.generate_time_averages.timeaverager.timeaverager attribute)": [[14, "fre.app.generate_time_averages.timeAverager.timeAverager.pkg", false]], "post_write() (in module fre.app.mask_atmos_plevel)": [[15, "fre.app.mask_atmos_plevel.post_write", false]], "pp_run_subtool() (in module fre.pp.run)": [[50, "fre.pp.run.pp_run_subtool", false]], "preprocess() (in module fre.app.mask_atmos_plevel)": [[15, "fre.app.mask_atmos_plevel.preprocess", false]], "pressure_coordinate() (in module fre.app.mask_atmos_plevel)": [[15, "fre.app.mask_atmos_plevel.pressure_coordinate", false]], "quote_rose_values() (in module fre.pp.configure_script_yaml)": [[47, "fre.pp.configure_script_yaml.quote_rose_values", false]], "regrid_xy() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.regrid_xy", false]], "rewrite_netcdf_file_var() (in module fre.cmor.cmor_mixer)": [[26, "fre.cmor.cmor_mixer.rewrite_netcdf_file_var", false]], "rose_init() (in module fre.pp.configure_script_yaml)": [[47, "fre.pp.configure_script_yaml.rose_init", false]], "run_avgtype_pkg_calculations() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.run_avgtype_pkg_calculations", false]], "safe_rose_config_get() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.safe_rose_config_get", false]], "set_netcdf_encoding() (in module fre.app.mask_atmos_plevel)": [[15, "fre.app.mask_atmos_plevel.set_netcdf_encoding", false]], "set_rose_apps() (in module fre.pp.configure_script_yaml)": [[47, "fre.pp.configure_script_yaml.set_rose_apps", false]], "set_rose_suite() (in module fre.pp.configure_script_yaml)": [[47, "fre.pp.configure_script_yaml.set_rose_suite", false]], "status_subtool() (in module fre.pp.status)": [[51, "fre.pp.status.status_subtool", false]], "stddev_type (fre.app.generate_time_averages.timeaverager.timeaverager attribute)": [[14, "fre.app.generate_time_averages.timeAverager.timeAverager.stddev_type", false]], "test_analysisyaml_exists() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_analysisyaml_exists", false]], "test_bm_makefile_creation() (in module fre.make.tests.test_create_makefile)": [[43, "fre.make.tests.test_create_makefile.test_bm_makefile_creation", false]], "test_boolean() (in module fre.pp.tests.test_rose_quoting)": [[54, "fre.pp.tests.test_rose_quoting.test_boolean", false]], "test_cdo_time_avgs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_cdo_time_avgs", false]], "test_cdo_time_unwgt_avgs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_cdo_time_unwgt_avgs", false]], "test_cdo_time_unwgt_stddevs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_cdo_time_unwgt_stddevs", false]], "test_cli_fre() (in module fre.tests.test_fre_cli)": [[68, "fre.tests.test_fre_cli.test_cli_fre", false]], "test_cli_fre_app() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app", false]], "test_cli_fre_app_gen_time_averages() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app_gen_time_averages", false]], "test_cli_fre_app_gen_time_averages_help() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app_gen_time_averages_help", false]], "test_cli_fre_app_gen_time_averages_opt_dne() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app_gen_time_averages_opt_dne", false]], "test_cli_fre_app_help() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app_help", false]], "test_cli_fre_app_opt_dne() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app_opt_dne", false]], "test_cli_fre_app_regrid() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app_regrid", false]], "test_cli_fre_app_regrid_help() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app_regrid_help", false]], "test_cli_fre_app_regrid_opt_dne() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app_regrid_opt_dne", false]], "test_cli_fre_app_regrid_test_case_1() (in module fre.tests.test_fre_app_cli)": [[65, "fre.tests.test_fre_app_cli.test_cli_fre_app_regrid_test_case_1", false]], "test_cli_fre_catalog() (in module fre.tests.test_fre_catalog_cli)": [[66, "fre.tests.test_fre_catalog_cli.test_cli_fre_catalog", false]], "test_cli_fre_catalog_builder() (in module fre.tests.test_fre_catalog_cli)": [[66, "fre.tests.test_fre_catalog_cli.test_cli_fre_catalog_builder", false]], "test_cli_fre_catalog_builder_help() (in module fre.tests.test_fre_catalog_cli)": [[66, "fre.tests.test_fre_catalog_cli.test_cli_fre_catalog_builder_help", false]], "test_cli_fre_catalog_help() (in module fre.tests.test_fre_catalog_cli)": [[66, "fre.tests.test_fre_catalog_cli.test_cli_fre_catalog_help", false]], "test_cli_fre_catalog_merge() (in module fre.tests.test_fre_catalog_cli)": [[66, "fre.tests.test_fre_catalog_cli.test_cli_fre_catalog_merge", false]], "test_cli_fre_catalog_merge_help() (in module fre.tests.test_fre_catalog_cli)": [[66, "fre.tests.test_fre_catalog_cli.test_cli_fre_catalog_merge_help", false]], "test_cli_fre_catalog_opt_dne() (in module fre.tests.test_fre_catalog_cli)": [[66, "fre.tests.test_fre_catalog_cli.test_cli_fre_catalog_opt_dne", false]], "test_cli_fre_check() (in module fre.tests.test_fre_check_cli)": [[67, "fre.tests.test_fre_check_cli.test_cli_fre_check", false]], "test_cli_fre_check_help() (in module fre.tests.test_fre_check_cli)": [[67, "fre.tests.test_fre_check_cli.test_cli_fre_check_help", false]], "test_cli_fre_check_opt_dne() (in module fre.tests.test_fre_check_cli)": [[67, "fre.tests.test_fre_check_cli.test_cli_fre_check_opt_dne", false]], "test_cli_fre_cmor() (in module fre.tests.test_fre_cmor_cli)": [[69, "fre.tests.test_fre_cmor_cli.test_cli_fre_cmor", false]], "test_cli_fre_cmor_help() (in module fre.tests.test_fre_cmor_cli)": [[69, "fre.tests.test_fre_cmor_cli.test_cli_fre_cmor_help", false]], "test_cli_fre_cmor_opt_dne() (in module fre.tests.test_fre_cmor_cli)": [[69, "fre.tests.test_fre_cmor_cli.test_cli_fre_cmor_opt_dne", false]], "test_cli_fre_cmor_run() (in module fre.tests.test_fre_cmor_cli)": [[69, "fre.tests.test_fre_cmor_cli.test_cli_fre_cmor_run", false]], "test_cli_fre_cmor_run_case1() (in module fre.tests.test_fre_cmor_cli)": [[69, "fre.tests.test_fre_cmor_cli.test_cli_fre_cmor_run_case1", false]], "test_cli_fre_cmor_run_case2() (in module fre.tests.test_fre_cmor_cli)": [[69, "fre.tests.test_fre_cmor_cli.test_cli_fre_cmor_run_case2", false]], "test_cli_fre_cmor_run_help() (in module fre.tests.test_fre_cmor_cli)": [[69, "fre.tests.test_fre_cmor_cli.test_cli_fre_cmor_run_help", false]], "test_cli_fre_cmor_run_opt_dne() (in module fre.tests.test_fre_cmor_cli)": [[69, "fre.tests.test_fre_cmor_cli.test_cli_fre_cmor_run_opt_dne", false]], "test_cli_fre_help() (in module fre.tests.test_fre_cli)": [[68, "fre.tests.test_fre_cli.test_cli_fre_help", false]], "test_cli_fre_list() (in module fre.tests.test_fre_list_cli)": [[70, "fre.tests.test_fre_list_cli.test_cli_fre_list", false]], "test_cli_fre_list_help() (in module fre.tests.test_fre_list_cli)": [[70, "fre.tests.test_fre_list_cli.test_cli_fre_list_help", false]], "test_cli_fre_list_opt_dne() (in module fre.tests.test_fre_list_cli)": [[70, "fre.tests.test_fre_list_cli.test_cli_fre_list_opt_dne", false]], "test_cli_fre_make() (in module fre.tests.test_fre_make_cli)": [[71, "fre.tests.test_fre_make_cli.test_cli_fre_make", false]], "test_cli_fre_make_create_checkout_baremetal() (in module fre.tests.test_fre_make_cli)": [[71, "fre.tests.test_fre_make_cli.test_cli_fre_make_create_checkout_baremetal", false]], "test_cli_fre_make_create_checkout_container() (in module fre.tests.test_fre_make_cli)": [[71, "fre.tests.test_fre_make_cli.test_cli_fre_make_create_checkout_container", false]], "test_cli_fre_make_help() (in module fre.tests.test_fre_make_cli)": [[71, "fre.tests.test_fre_make_cli.test_cli_fre_make_help", false]], "test_cli_fre_make_opt_dne() (in module fre.tests.test_fre_make_cli)": [[71, "fre.tests.test_fre_make_cli.test_cli_fre_make_opt_dne", false]], "test_cli_fre_option_dne() (in module fre.tests.test_fre_cli)": [[68, "fre.tests.test_fre_cli.test_cli_fre_option_dne", false]], "test_cli_fre_pp() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp", false]], "test_cli_fre_pp_checkout() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_checkout", false]], "test_cli_fre_pp_checkout_case() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_checkout_case", false]], "test_cli_fre_pp_checkout_help() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_checkout_help", false]], "test_cli_fre_pp_checkout_opt_dne() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_checkout_opt_dne", false]], "test_cli_fre_pp_configure_xml() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_configure_xml", false]], "test_cli_fre_pp_configure_xml_help() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_configure_xml_help", false]], "test_cli_fre_pp_configure_xml_opt_dne() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_configure_xml_opt_dne", false]], "test_cli_fre_pp_configure_yaml() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_configure_yaml", false]], "test_cli_fre_pp_configure_yaml_fail1() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_configure_yaml_fail1", false]], "test_cli_fre_pp_configure_yaml_help() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_configure_yaml_help", false]], "test_cli_fre_pp_configure_yaml_opt_dne() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_configure_yaml_opt_dne", false]], "test_cli_fre_pp_help() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_help", false]], "test_cli_fre_pp_install() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_install", false]], "test_cli_fre_pp_install_help() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_install_help", false]], "test_cli_fre_pp_install_opt_dne() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_install_opt_dne", false]], "test_cli_fre_pp_opt_dne() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_opt_dne", false]], "test_cli_fre_pp_run() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_run", false]], "test_cli_fre_pp_run_help() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_run_help", false]], "test_cli_fre_pp_run_opt_dne() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_run_opt_dne", false]], "test_cli_fre_pp_status() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_status", false]], "test_cli_fre_pp_status_help() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_status_help", false]], "test_cli_fre_pp_status_opt_dne() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_status_opt_dne", false]], "test_cli_fre_pp_validate() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_validate", false]], "test_cli_fre_pp_validate_help() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_validate_help", false]], "test_cli_fre_pp_validate_opt_dne() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_validate_opt_dne", false]], "test_cli_fre_pp_wrapper() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_wrapper", false]], "test_cli_fre_pp_wrapper_help() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_wrapper_help", false]], "test_cli_fre_pp_wrapper_opt_dne() (in module fre.tests.test_fre_pp_cli)": [[72, "fre.tests.test_fre_pp_cli.test_cli_fre_pp_wrapper_opt_dne", false]], "test_cli_fre_run() (in module fre.tests.test_fre_run_cli)": [[73, "fre.tests.test_fre_run_cli.test_cli_fre_run", false]], "test_cli_fre_run_help() (in module fre.tests.test_fre_run_cli)": [[73, "fre.tests.test_fre_run_cli.test_cli_fre_run_help", false]], "test_cli_fre_run_opt_dne() (in module fre.tests.test_fre_run_cli)": [[73, "fre.tests.test_fre_run_cli.test_cli_fre_run_opt_dne", false]], "test_cli_fre_test() (in module fre.tests.test_fre_test_cli)": [[74, "fre.tests.test_fre_test_cli.test_cli_fre_test", false]], "test_cli_fre_test_help() (in module fre.tests.test_fre_test_cli)": [[74, "fre.tests.test_fre_test_cli.test_cli_fre_test_help", false]], "test_cli_fre_test_opt_dne() (in module fre.tests.test_fre_test_cli)": [[74, "fre.tests.test_fre_test_cli.test_cli_fre_test_opt_dne", false]], "test_cli_fre_yamltools() (in module fre.tests.test_fre_yamltools_cli)": [[75, "fre.tests.test_fre_yamltools_cli.test_cli_fre_yamltools", false]], "test_cli_fre_yamltools_help() (in module fre.tests.test_fre_yamltools_cli)": [[75, "fre.tests.test_fre_yamltools_cli.test_cli_fre_yamltools_help", false]], "test_cli_fre_yamltools_opt_dne() (in module fre.tests.test_fre_yamltools_cli)": [[75, "fre.tests.test_fre_yamltools_cli.test_cli_fre_yamltools_opt_dne", false]], "test_combined_compileyaml_combinefail() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_combined_compileyaml_combinefail", false]], "test_combined_compileyaml_validatefail() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_combined_compileyaml_validatefail", false]], "test_combined_compileyaml_validation() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_combined_compileyaml_validation", false]], "test_combined_ppyaml_validation() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_combined_ppyaml_validation", false]], "test_combinedyaml_exists() (in module fre.pp.tests.test_configure_script_yaml)": [[53, "fre.pp.tests.test_configure_script_yaml.test_combinedyaml_exists", false]], "test_compare_cdo_to_fre_nctools() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_compare_cdo_to_fre_nctools", false]], "test_compare_fre_cli_to_cdo() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_compare_fre_cli_to_cdo", false]], "test_compare_fre_cli_to_fre_nctools() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_compare_fre_cli_to_fre_nctools", false]], "test_compare_unwgt_fre_cli_to_unwgt_cdo() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_compare_unwgt_fre_cli_to_unwgt_cdo", false]], "test_compileyaml_exists() (in module fre.make.tests.test_create_makefile)": [[43, "fre.make.tests.test_create_makefile.test_compileyaml_exists", false]], "test_compileyaml_exists() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_compileyaml_exists", false]], "test_configure_script() (in module fre.pp.tests.test_configure_script_yaml)": [[53, "fre.pp.tests.test_configure_script_yaml.test_configure_script", false]], "test_container_makefile_creation() (in module fre.make.tests.test_create_makefile)": [[43, "fre.make.tests.test_create_makefile.test_container_makefile_creation", false]], "test_expyaml_exists() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_expyaml_exists", false]], "test_fre_catalog_import() (in module fre.catalog.tests.test_fre_catalog)": [[21, "fre.catalog.tests.test_fre_catalog.test_fre_catalog_import", false]], "test_fre_cli_time_avgs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_fre_cli_time_avgs", false]], "test_fre_cli_time_avgs_stddevs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_fre_cli_time_avgs_stddevs", false]], "test_fre_cli_time_unwgt_avgs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_fre_cli_time_unwgt_avgs", false]], "test_fre_cli_time_unwgt_avgs_stddevs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_fre_cli_time_unwgt_avgs_stddevs", false]], "test_fre_cmor_run_subtool_case1() (in module fre.cmor.tests.test_cmor_run_subtool)": [[29, "fre.cmor.tests.test_cmor_run_subtool.test_fre_cmor_run_subtool_case1", false]], "test_fre_cmor_run_subtool_case1_output_compare_data() (in module fre.cmor.tests.test_cmor_run_subtool)": [[29, "fre.cmor.tests.test_cmor_run_subtool.test_fre_cmor_run_subtool_case1_output_compare_data", false]], "test_fre_cmor_run_subtool_case1_output_compare_metadata() (in module fre.cmor.tests.test_cmor_run_subtool)": [[29, "fre.cmor.tests.test_cmor_run_subtool.test_fre_cmor_run_subtool_case1_output_compare_metadata", false]], "test_fre_cmor_run_subtool_case2() (in module fre.cmor.tests.test_cmor_run_subtool)": [[29, "fre.cmor.tests.test_cmor_run_subtool.test_fre_cmor_run_subtool_case2", false]], "test_fre_cmor_run_subtool_case2_output_compare_data() (in module fre.cmor.tests.test_cmor_run_subtool)": [[29, "fre.cmor.tests.test_cmor_run_subtool.test_fre_cmor_run_subtool_case2_output_compare_data", false]], "test_fre_cmor_run_subtool_case2_output_compare_metadata() (in module fre.cmor.tests.test_cmor_run_subtool)": [[29, "fre.cmor.tests.test_cmor_run_subtool.test_fre_cmor_run_subtool_case2_output_compare_metadata", false]], "test_import() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.test_import", false]], "test_merged_compile_yamls() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_merged_compile_yamls", false]], "test_merged_pp_yamls() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_merged_pp_yamls", false]], "test_modelyaml_exists() (in module fre.make.tests.test_create_makefile)": [[43, "fre.make.tests.test_create_makefile.test_modelyaml_exists", false]], "test_modelyaml_exists() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_modelyaml_exists", false]], "test_monthly_cdo_time_unwgt_avgs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_monthly_cdo_time_unwgt_avgs", false]], "test_monthly_cdo_time_unwgt_stddevs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_monthly_cdo_time_unwgt_stddevs", false]], "test_platformyaml_exists() (in module fre.make.tests.test_create_makefile)": [[43, "fre.make.tests.test_create_makefile.test_platformyaml_exists", false]], "test_platformyaml_exists() (in module fre.yamltools.tests.test_combine_yamls)": [[80, "fre.yamltools.tests.test_combine_yamls.test_platformyaml_exists", false]], "test_seasonal_cdo_time_unwgt_avgs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_seasonal_cdo_time_unwgt_avgs", false]], "test_seasonal_cdo_time_unwgt_stddevs() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_seasonal_cdo_time_unwgt_stddevs", false]], "test_setup_cmor_cmip_table_repo() (in module fre.cmor.tests.test_cmor_run_subtool)": [[29, "fre.cmor.tests.test_cmor_run_subtool.test_setup_cmor_cmip_table_repo", false]], "test_setup_fre_cmor_run_subtool() (in module fre.cmor.tests.test_cmor_run_subtool)": [[29, "fre.cmor.tests.test_cmor_run_subtool.test_setup_fre_cmor_run_subtool", false]], "test_setup_fre_cmor_run_subtool_case2() (in module fre.cmor.tests.test_cmor_run_subtool)": [[29, "fre.cmor.tests.test_cmor_run_subtool.test_setup_fre_cmor_run_subtool_case2", false]], "test_string() (in module fre.pp.tests.test_rose_quoting)": [[54, "fre.pp.tests.test_rose_quoting.test_string", false]], "test_time_avg_file_dir_exists() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_time_avg_file_dir_exists", false]], "test_time_avg_input_file_exists() (in module fre.app.generate_time_averages.tests.test_generate_time_averages)": [[13, "fre.app.generate_time_averages.tests.test_generate_time_averages.test_time_avg_input_file_exists", false]], "timeaverager (class in fre.app.generate_time_averages.timeaverager)": [[14, "fre.app.generate_time_averages.timeAverager.timeAverager", false]], "trigger() (in module fre.pp.trigger)": [[55, "fre.pp.trigger.trigger", false]], "truncate_date() (in module fre.app.regrid_xy.regrid_xy)": [[17, "fre.app.regrid_xy.regrid_xy.truncate_date", false]], "unwgt (fre.app.generate_time_averages.timeaverager.timeaverager attribute)": [[14, "fre.app.generate_time_averages.timeAverager.timeAverager.unwgt", false]], "validate_subtool() (in module fre.pp.validate)": [[56, "fre.pp.validate.validate_subtool", false]], "validate_yaml() (in module fre.pp.configure_script_yaml)": [[47, "fre.pp.configure_script_yaml.validate_yaml", false]], "var (fre.app.generate_time_averages.timeaverager.timeaverager attribute)": [[14, "fre.app.generate_time_averages.timeAverager.timeAverager.var", false]], "var_has_time_units() (fre.app.generate_time_averages.timeaverager.timeaverager method)": [[14, "fre.app.generate_time_averages.timeAverager.timeAverager.var_has_time_units", false]], "write_dataset() (in module fre.app.mask_atmos_plevel)": [[15, "fre.app.mask_atmos_plevel.write_dataset", false]], "yaml_load() (in module fre.pp.configure_script_yaml)": [[47, "fre.pp.configure_script_yaml.yaml_load", false]], "yaml_load() (in module fre.yamltools.combine_yamls)": [[77, "fre.yamltools.combine_yamls.yaml_load", false]], "yamlinfo() (in module fre.pp.configure_script_yaml)": [[47, "fre.pp.configure_script_yaml.yamlInfo", false]]}, "objects": {"": [[4, 0, 0, "-", "fre"]], "fre": [[5, 0, 0, "-", "app"], [18, 0, 0, "-", "catalog"], [22, 0, 0, "-", "check"], [25, 0, 0, "-", "cmor"], [30, 0, 0, "-", "fre"], [31, 0, 0, "-", "lazy_group"], [32, 0, 0, "-", "list"], [35, 0, 0, "-", "make"], [44, 0, 0, "-", "pp"], [58, 0, 0, "-", "run"], [61, 0, 0, "-", "test"], [64, 0, 0, "-", "tests"], [76, 0, 0, "-", "yamltools"]], "fre.app": [[6, 0, 0, "-", "freapp"], [7, 0, 0, "-", "generate_time_averages"], [15, 0, 0, "-", "mask_atmos_plevel"], [16, 0, 0, "-", "regrid_xy"]], "fre.app.generate_time_averages": [[8, 0, 0, "-", "cdoTimeAverager"], [9, 0, 0, "-", "frenctoolsTimeAverager"], [10, 0, 0, "-", "frepytoolsTimeAverager"], [11, 0, 0, "-", "generate_time_averages"], [12, 0, 0, "-", "tests"], [14, 0, 0, "-", "timeAverager"]], "fre.app.generate_time_averages.cdoTimeAverager": [[8, 1, 1, "", "cdoTimeAverager"]], "fre.app.generate_time_averages.cdoTimeAverager.cdoTimeAverager": [[8, 2, 1, "", "generate_timavg"]], "fre.app.generate_time_averages.frenctoolsTimeAverager": [[9, 1, 1, "", "frenctoolsTimeAverager"]], "fre.app.generate_time_averages.frenctoolsTimeAverager.frenctoolsTimeAverager": [[9, 2, 1, "", "generate_timavg"]], "fre.app.generate_time_averages.frepytoolsTimeAverager": [[10, 1, 1, "", "frepytoolsTimeAverager"]], "fre.app.generate_time_averages.frepytoolsTimeAverager.frepytoolsTimeAverager": [[10, 2, 1, "", "generate_timavg"]], "fre.app.generate_time_averages.generate_time_averages": [[11, 3, 1, "", "generate_time_average"]], "fre.app.generate_time_averages.tests": [[13, 0, 0, "-", "test_generate_time_averages"]], "fre.app.generate_time_averages.tests.test_generate_time_averages": [[13, 3, 1, "", "run_avgtype_pkg_calculations"], [13, 3, 1, "", "test_cdo_time_avgs"], [13, 3, 1, "", "test_cdo_time_unwgt_avgs"], [13, 3, 1, "", "test_cdo_time_unwgt_stddevs"], [13, 3, 1, "", "test_compare_cdo_to_fre_nctools"], [13, 3, 1, "", "test_compare_fre_cli_to_cdo"], [13, 3, 1, "", "test_compare_fre_cli_to_fre_nctools"], [13, 3, 1, "", "test_compare_unwgt_fre_cli_to_unwgt_cdo"], [13, 3, 1, "", "test_fre_cli_time_avgs"], [13, 3, 1, "", "test_fre_cli_time_avgs_stddevs"], [13, 3, 1, "", "test_fre_cli_time_unwgt_avgs"], [13, 3, 1, "", "test_fre_cli_time_unwgt_avgs_stddevs"], [13, 3, 1, "", "test_monthly_cdo_time_unwgt_avgs"], [13, 3, 1, "", "test_monthly_cdo_time_unwgt_stddevs"], [13, 3, 1, "", "test_seasonal_cdo_time_unwgt_avgs"], [13, 3, 1, "", "test_seasonal_cdo_time_unwgt_stddevs"], [13, 3, 1, "", "test_time_avg_file_dir_exists"], [13, 3, 1, "", "test_time_avg_input_file_exists"]], "fre.app.generate_time_averages.timeAverager": [[14, 1, 1, "", "timeAverager"]], "fre.app.generate_time_averages.timeAverager.timeAverager": [[14, 4, 1, "", "avg_type"], [14, 2, 1, "", "generate_timavg"], [14, 4, 1, "", "pkg"], [14, 4, 1, "", "stddev_type"], [14, 4, 1, "", "unwgt"], [14, 4, 1, "", "var"], [14, 2, 1, "", "var_has_time_units"]], "fre.app.mask_atmos_plevel": [[15, 3, 1, "", "mask_field_above_surface_pressure"], [15, 3, 1, "", "post_write"], [15, 3, 1, "", "preprocess"], [15, 3, 1, "", "pressure_coordinate"], [15, 3, 1, "", "set_netcdf_encoding"], [15, 3, 1, "", "write_dataset"]], "fre.app.regrid_xy": [[17, 0, 0, "-", "regrid_xy"]], "fre.app.regrid_xy.regrid_xy": [[17, 3, 1, "", "check_interp_method"], [17, 3, 1, "", "check_per_component_settings"], [17, 3, 1, "", "freq_to_date_format"], [17, 3, 1, "", "get_mosaic_file_name"], [17, 3, 1, "", "get_mosaic_grid_file_name"], [17, 3, 1, "", "main"], [17, 3, 1, "", "make_component_list"], [17, 3, 1, "", "make_regrid_var_list"], [17, 3, 1, "", "regrid_xy"], [17, 3, 1, "", "safe_rose_config_get"], [17, 3, 1, "", "test_import"], [17, 3, 1, "", "truncate_date"]], "fre.catalog": [[19, 0, 0, "-", "frecatalog"], [20, 0, 0, "-", "tests"]], "fre.catalog.tests": [[21, 0, 0, "-", "test_fre_catalog"]], "fre.catalog.tests.test_fre_catalog": [[21, 3, 1, "", "test_fre_catalog_import"]], "fre.check": [[23, 0, 0, "-", "frecheck"], [24, 0, 0, "-", "frecheckexample"]], "fre.cmor": [[26, 0, 0, "-", "cmor_mixer"], [27, 0, 0, "-", "frecmor"], [28, 0, 0, "-", "tests"]], "fre.cmor.cmor_mixer": [[26, 3, 1, "", "check_dataset_for_ocean_grid"], [26, 3, 1, "", "cmor_run_subtool"], [26, 3, 1, "", "cmorize_target_var_files"], [26, 3, 1, "", "copy_nc"], [26, 3, 1, "", "create_tmp_dir"], [26, 3, 1, "", "get_iso_datetimes"], [26, 3, 1, "", "get_var_filenames"], [26, 3, 1, "", "get_vertical_dimension"], [26, 3, 1, "", "rewrite_netcdf_file_var"]], "fre.cmor.tests": [[29, 0, 0, "-", "test_cmor_run_subtool"]], "fre.cmor.tests.test_cmor_run_subtool": [[29, 3, 1, "", "test_fre_cmor_run_subtool_case1"], [29, 3, 1, "", "test_fre_cmor_run_subtool_case1_output_compare_data"], [29, 3, 1, "", "test_fre_cmor_run_subtool_case1_output_compare_metadata"], [29, 3, 1, "", "test_fre_cmor_run_subtool_case2"], [29, 3, 1, "", "test_fre_cmor_run_subtool_case2_output_compare_data"], [29, 3, 1, "", "test_fre_cmor_run_subtool_case2_output_compare_metadata"], [29, 3, 1, "", "test_setup_cmor_cmip_table_repo"], [29, 3, 1, "", "test_setup_fre_cmor_run_subtool"], [29, 3, 1, "", "test_setup_fre_cmor_run_subtool_case2"]], "fre.lazy_group": [[31, 1, 1, "", "LazyGroup"]], "fre.lazy_group.LazyGroup": [[31, 2, 1, "", "get_command"], [31, 2, 1, "", "list_commands"]], "fre.list": [[33, 0, 0, "-", "frelist"], [34, 0, 0, "-", "frelistexample"]], "fre.make": [[36, 0, 0, "-", "createCheckout"], [37, 0, 0, "-", "createCompile"], [38, 0, 0, "-", "createDocker"], [39, 0, 0, "-", "createMakefile"], [40, 0, 0, "-", "fremake"], [41, 0, 0, "-", "runFremake"], [42, 0, 0, "-", "tests"]], "fre.make.createCheckout": [[36, 3, 1, "", "checkout_create"]], "fre.make.createCompile": [[37, 3, 1, "", "compile_create"]], "fre.make.createDocker": [[38, 3, 1, "", "dockerfile_create"]], "fre.make.createMakefile": [[39, 3, 1, "", "makefile_create"]], "fre.make.runFremake": [[41, 3, 1, "", "fremake_run"]], "fre.make.tests": [[43, 0, 0, "-", "test_create_makefile"]], "fre.make.tests.test_create_makefile": [[43, 3, 1, "", "test_bm_makefile_creation"], [43, 3, 1, "", "test_compileyaml_exists"], [43, 3, 1, "", "test_container_makefile_creation"], [43, 3, 1, "", "test_modelyaml_exists"], [43, 3, 1, "", "test_platformyaml_exists"]], "fre.pp": [[45, 0, 0, "-", "checkout_script"], [46, 0, 0, "-", "configure_script_xml"], [47, 0, 0, "-", "configure_script_yaml"], [48, 0, 0, "-", "frepp"], [49, 0, 0, "-", "install"], [50, 0, 0, "-", "run"], [51, 0, 0, "-", "status"], [52, 0, 0, "-", "tests"], [55, 0, 0, "-", "trigger"], [56, 0, 0, "-", "validate"], [57, 0, 0, "-", "wrapper"]], "fre.pp.checkout_script": [[45, 3, 1, "", "checkout_template"]], "fre.pp.configure_script_xml": [[46, 3, 1, "", "chunk_from_legacy"], [46, 3, 1, "", "duration_to_seconds"], [46, 3, 1, "", "frelist_xpath"], [46, 3, 1, "", "freq_from_legacy"], [46, 3, 1, "", "main"]], "fre.pp.configure_script_yaml": [[47, 3, 1, "", "quote_rose_values"], [47, 3, 1, "", "rose_init"], [47, 3, 1, "", "set_rose_apps"], [47, 3, 1, "", "set_rose_suite"], [47, 3, 1, "", "validate_yaml"], [47, 3, 1, "", "yamlInfo"], [47, 3, 1, "", "yaml_load"]], "fre.pp.install": [[49, 3, 1, "", "install_subtool"]], "fre.pp.run": [[50, 3, 1, "", "pp_run_subtool"]], "fre.pp.status": [[51, 3, 1, "", "status_subtool"]], "fre.pp.tests": [[53, 0, 0, "-", "test_configure_script_yaml"], [54, 0, 0, "-", "test_rose_quoting"]], "fre.pp.tests.test_configure_script_yaml": [[53, 3, 1, "", "test_combinedyaml_exists"], [53, 3, 1, "", "test_configure_script"]], "fre.pp.tests.test_rose_quoting": [[54, 3, 1, "", "test_boolean"], [54, 3, 1, "", "test_string"]], "fre.pp.trigger": [[55, 3, 1, "", "trigger"]], "fre.pp.validate": [[56, 3, 1, "", "validate_subtool"]], "fre.run": [[59, 0, 0, "-", "frerun"], [60, 0, 0, "-", "frerunexample"]], "fre.test": [[62, 0, 0, "-", "fretest"], [63, 0, 0, "-", "fretestexample"]], "fre.tests": [[65, 0, 0, "-", "test_fre_app_cli"], [66, 0, 0, "-", "test_fre_catalog_cli"], [67, 0, 0, "-", "test_fre_check_cli"], [68, 0, 0, "-", "test_fre_cli"], [69, 0, 0, "-", "test_fre_cmor_cli"], [70, 0, 0, "-", "test_fre_list_cli"], [71, 0, 0, "-", "test_fre_make_cli"], [72, 0, 0, "-", "test_fre_pp_cli"], [73, 0, 0, "-", "test_fre_run_cli"], [74, 0, 0, "-", "test_fre_test_cli"], [75, 0, 0, "-", "test_fre_yamltools_cli"]], "fre.tests.test_fre_app_cli": [[65, 3, 1, "", "test_cli_fre_app"], [65, 3, 1, "", "test_cli_fre_app_gen_time_averages"], [65, 3, 1, "", "test_cli_fre_app_gen_time_averages_help"], [65, 3, 1, "", "test_cli_fre_app_gen_time_averages_opt_dne"], [65, 3, 1, "", "test_cli_fre_app_help"], [65, 3, 1, "", "test_cli_fre_app_opt_dne"], [65, 3, 1, "", "test_cli_fre_app_regrid"], [65, 3, 1, "", "test_cli_fre_app_regrid_help"], [65, 3, 1, "", "test_cli_fre_app_regrid_opt_dne"], [65, 3, 1, "", "test_cli_fre_app_regrid_test_case_1"]], "fre.tests.test_fre_catalog_cli": [[66, 3, 1, "", "test_cli_fre_catalog"], [66, 3, 1, "", "test_cli_fre_catalog_builder"], [66, 3, 1, "", "test_cli_fre_catalog_builder_help"], [66, 3, 1, "", "test_cli_fre_catalog_help"], [66, 3, 1, "", "test_cli_fre_catalog_merge"], [66, 3, 1, "", "test_cli_fre_catalog_merge_help"], [66, 3, 1, "", "test_cli_fre_catalog_opt_dne"]], "fre.tests.test_fre_check_cli": [[67, 3, 1, "", "test_cli_fre_check"], [67, 3, 1, "", "test_cli_fre_check_help"], [67, 3, 1, "", "test_cli_fre_check_opt_dne"]], "fre.tests.test_fre_cli": [[68, 3, 1, "", "test_cli_fre"], [68, 3, 1, "", "test_cli_fre_help"], [68, 3, 1, "", "test_cli_fre_option_dne"]], "fre.tests.test_fre_cmor_cli": [[69, 3, 1, "", "test_cli_fre_cmor"], [69, 3, 1, "", "test_cli_fre_cmor_help"], [69, 3, 1, "", "test_cli_fre_cmor_opt_dne"], [69, 3, 1, "", "test_cli_fre_cmor_run"], [69, 3, 1, "", "test_cli_fre_cmor_run_case1"], [69, 3, 1, "", "test_cli_fre_cmor_run_case2"], [69, 3, 1, "", "test_cli_fre_cmor_run_help"], [69, 3, 1, "", "test_cli_fre_cmor_run_opt_dne"]], "fre.tests.test_fre_list_cli": [[70, 3, 1, "", "test_cli_fre_list"], [70, 3, 1, "", "test_cli_fre_list_help"], [70, 3, 1, "", "test_cli_fre_list_opt_dne"]], "fre.tests.test_fre_make_cli": [[71, 3, 1, "", "test_cli_fre_make"], [71, 3, 1, "", "test_cli_fre_make_create_checkout_baremetal"], [71, 3, 1, "", "test_cli_fre_make_create_checkout_container"], [71, 3, 1, "", "test_cli_fre_make_help"], [71, 3, 1, "", "test_cli_fre_make_opt_dne"]], "fre.tests.test_fre_pp_cli": [[72, 3, 1, "", "test_cli_fre_pp"], [72, 3, 1, "", "test_cli_fre_pp_checkout"], [72, 3, 1, "", "test_cli_fre_pp_checkout_case"], [72, 3, 1, "", "test_cli_fre_pp_checkout_help"], [72, 3, 1, "", "test_cli_fre_pp_checkout_opt_dne"], [72, 3, 1, "", "test_cli_fre_pp_configure_xml"], [72, 3, 1, "", "test_cli_fre_pp_configure_xml_help"], [72, 3, 1, "", "test_cli_fre_pp_configure_xml_opt_dne"], [72, 3, 1, "", "test_cli_fre_pp_configure_yaml"], [72, 3, 1, "", "test_cli_fre_pp_configure_yaml_fail1"], [72, 3, 1, "", "test_cli_fre_pp_configure_yaml_help"], [72, 3, 1, "", "test_cli_fre_pp_configure_yaml_opt_dne"], [72, 3, 1, "", "test_cli_fre_pp_help"], [72, 3, 1, "", "test_cli_fre_pp_install"], [72, 3, 1, "", "test_cli_fre_pp_install_help"], [72, 3, 1, "", "test_cli_fre_pp_install_opt_dne"], [72, 3, 1, "", "test_cli_fre_pp_opt_dne"], [72, 3, 1, "", "test_cli_fre_pp_run"], [72, 3, 1, "", "test_cli_fre_pp_run_help"], [72, 3, 1, "", "test_cli_fre_pp_run_opt_dne"], [72, 3, 1, "", "test_cli_fre_pp_status"], [72, 3, 1, "", "test_cli_fre_pp_status_help"], [72, 3, 1, "", "test_cli_fre_pp_status_opt_dne"], [72, 3, 1, "", "test_cli_fre_pp_validate"], [72, 3, 1, "", "test_cli_fre_pp_validate_help"], [72, 3, 1, "", "test_cli_fre_pp_validate_opt_dne"], [72, 3, 1, "", "test_cli_fre_pp_wrapper"], [72, 3, 1, "", "test_cli_fre_pp_wrapper_help"], [72, 3, 1, "", "test_cli_fre_pp_wrapper_opt_dne"]], "fre.tests.test_fre_run_cli": [[73, 3, 1, "", "test_cli_fre_run"], [73, 3, 1, "", "test_cli_fre_run_help"], [73, 3, 1, "", "test_cli_fre_run_opt_dne"]], "fre.tests.test_fre_test_cli": [[74, 3, 1, "", "test_cli_fre_test"], [74, 3, 1, "", "test_cli_fre_test_help"], [74, 3, 1, "", "test_cli_fre_test_opt_dne"]], "fre.tests.test_fre_yamltools_cli": [[75, 3, 1, "", "test_cli_fre_yamltools"], [75, 3, 1, "", "test_cli_fre_yamltools_help"], [75, 3, 1, "", "test_cli_fre_yamltools_opt_dne"]], "fre.yamltools": [[77, 0, 0, "-", "combine_yamls"], [78, 0, 0, "-", "freyamltools"], [79, 0, 0, "-", "tests"]], "fre.yamltools.combine_yamls": [[77, 3, 1, "", "combined_compile_existcheck"], [77, 3, 1, "", "consolidate_yamls"], [77, 3, 1, "", "experiment_check"], [77, 3, 1, "", "get_combined_compileyaml"], [77, 3, 1, "", "get_combined_ppyaml"], [77, 3, 1, "", "get_compile_paths"], [77, 1, 1, "", "init_compile_yaml"], [77, 1, 1, "", "init_pp_yaml"], [77, 3, 1, "", "join_constructor"], [77, 3, 1, "", "yaml_load"]], "fre.yamltools.combine_yamls.init_compile_yaml": [[77, 2, 1, "", "clean_yaml"], [77, 2, 1, "", "combine_compile"], [77, 2, 1, "", "combine_model"], [77, 2, 1, "", "combine_platforms"]], "fre.yamltools.combine_yamls.init_pp_yaml": [[77, 2, 1, "", "clean_yaml"], [77, 2, 1, "", "combine_analysis"], [77, 2, 1, "", "combine_experiment"], [77, 2, 1, "", "combine_model"]], "fre.yamltools.tests": [[80, 0, 0, "-", "test_combine_yamls"]], "fre.yamltools.tests.test_combine_yamls": [[80, 3, 1, "", "test_analysisyaml_exists"], [80, 3, 1, "", "test_combined_compileyaml_combinefail"], [80, 3, 1, "", "test_combined_compileyaml_validatefail"], [80, 3, 1, "", "test_combined_compileyaml_validation"], [80, 3, 1, "", "test_combined_ppyaml_validation"], [80, 3, 1, "", "test_compileyaml_exists"], [80, 3, 1, "", "test_expyaml_exists"], [80, 3, 1, "", "test_merged_compile_yamls"], [80, 3, 1, "", "test_merged_pp_yamls"], [80, 3, 1, "", "test_modelyaml_exists"], [80, 3, 1, "", "test_platformyaml_exists"]]}, "objnames": {"0": ["py", "module", "Python module"], "1": ["py", "class", "Python class"], "2": ["py", "method", "Python method"], "3": ["py", "function", "Python function"], "4": ["py", "attribute", "Python attribute"]}, "objtypes": {"0": "py:module", "1": "py:class", "2": "py:method", "3": "py:function", "4": "py:attribute"}, "terms": {"": [3, 9, 13, 14, 24, 26, 29, 34, 41, 46, 60, 63, 84, 86, 87, 89, 91, 92, 94, 95, 96], "01": [83, 91, 94, 95, 96], "01t0000z": [91, 94], "02": 96, "1": [31, 45], "10": [91, 94], "180": [91, 94], "1979": [91, 94], "19790101": [91, 94], "19790701": [91, 94], "19800101": [91, 94], "19800701": [91, 94], "19810101": [91, 94], "19810701": [91, 94], "19820101": [91, 94], "19820701": [91, 94], "19830101": [91, 94], "19830701": [91, 94], "1x1": [91, 94], "2": 45, "2004": 96, "2020": [91, 94], "2023": [30, 41, 71, 91, 93], "2024": [30, 83, 91, 95, 96], "202401": 83, "2025": [91, 96], "23": 96, "3": [2, 45, 46], "31": 3, "360": [91, 94], "4": 45, "5": [91, 94], "6": [84, 87, 91, 94], "8": 31, "8601": 17, "A": [84, 87], "At": 3, "Be": [3, 91, 95], "By": [17, 91, 94], "For": [81, 91, 93, 94, 95], "If": [2, 77, 83, 91, 92, 95], "In": [2, 3, 91, 93, 94, 95, 96], "It": [15, 17, 84, 87, 96], "On": [2, 3, 81], "TO": 53, "The": [2, 3, 26, 46, 47, 84, 87, 91, 92, 93, 94, 95, 96], "These": [91, 93, 95], "To": [2, 83, 84, 91, 93, 94], "With": [84, 88], "__": [49, 50, 51, 56], "__init__": 3, "_compil": [91, 93], "_unmsk": 15, "abl": [2, 91, 93], "about": [2, 84, 91, 92, 93], "abov": [3, 11, 91, 94], "abstract": [8, 9, 10, 14], "accept": [15, 26, 84, 87], "access": [3, 91, 93], "account": 45, "acronym": [84, 91, 92], "across": [91, 95], "action": 2, "activ": 83, "ad": 81, "add": [3, 15], "addit": [3, 26, 84, 87, 91, 93], "additionalinstruct": [91, 93], "adopt": 96, "adventur": [91, 94], "after": [3, 26, 84, 87, 91, 93, 96], "again": [91, 94], "aka": 9, "albedo": 15, "alert": [84, 91, 92], "all": [3, 26, 84, 87, 91, 92, 93], "allow": [30, 77, 91, 92, 94, 95, 96], "almost": 26, "along": 3, "alreadi": 77, "also": [26, 29, 47, 84, 91, 92, 94, 95, 96], "alwai": 26, "am5": [71, 84, 86, 91, 93, 94], "am5f3b1r0": [84, 86], "am5f7c1r0": [91, 94], "amongst": 26, "an": [3, 13, 26, 77, 83, 84, 87, 91, 92, 93, 94, 95], "an_nc_var": 14, "anaconda": [3, 83], "analysi": [77, 95], "analyz": 96, "ani": [2, 3, 15, 84, 88, 91, 92, 94, 95], "annual": [91, 94], "anoth": [14, 84, 87, 96], "answer": 13, "anyth": 2, "anywai": 26, "api": [26, 81], "app": [3, 4, 46, 47, 53, 65, 81, 82, 85, 91, 95], "appear": [17, 31], "append": [26, 84, 87], "appli": [91, 92, 95], "approach": [3, 8, 10, 96], "apptain": [91, 95], "ar": [3, 14, 15, 17, 26, 47, 83, 84, 85, 87, 91, 92, 93, 94, 95, 96], "arbitrari": [91, 94], "arch5": [91, 94], "archiv": [84, 86, 91, 94], "arg": [17, 26, 31, 46], "argpars": 46, "argument": [3, 26, 46, 77, 91, 92], "arkansa": 96, "around": [2, 96], "arrai": [17, 91, 95], "assess": [91, 94], "assoc": [2, 26], "associ": [26, 84, 87, 91, 95], "atmo": [91, 94], "atmos_annu": [91, 94], "atmos_month": [91, 94], "atmospher": [91, 94], "attach": [26, 84, 87], "attribut": [15, 17], "author": [24, 30, 34, 41, 57, 60, 63], "auto": 0, "automag": 2, "automat": [3, 91, 93], "avail": [83, 91, 92, 94, 95, 96], "averag": [8, 9, 10, 11, 13, 14], "avg_typ": [7, 8, 9, 10, 11, 13, 14], "avoid": 10, "back": [2, 91], "badboi": 26, "bar": 72, "bare": [43, 91, 93, 95], "baremetal_linkerflag": [91, 93], "base": [8, 9, 10, 14, 26, 31, 77, 91, 96], "bash": 57, "batch": 96, "baz": 72, "bcc2761": [24, 30, 34, 60, 63], "been": 96, "befor": 26, "begin": [3, 17, 91, 94, 95], "being": [3, 84, 87, 91, 94], "below": [91, 92, 93], "bennett": [24, 30, 34, 41, 60, 63], "better": [3, 53], "bin": 57, "bind": 8, "bit": 2, "blob": 57, "block": [91, 93, 95], "board": [84, 91, 92], "bool": [14, 17], "boolean": [47, 84, 87, 91, 93, 95], "both": [26, 91, 95], "bounds_vari": 15, "box": [84, 87], "branch": [2, 3, 45, 80, 91, 93, 95], "break": [3, 53], "brief": [84, 87, 91, 92], "bring": 83, "bronx": [17, 46, 91, 94, 96], "build": [2, 81, 84, 86, 88, 93, 95], "build_conda": 2, "builder": 66, "bulk": [84, 87], "c": [91, 95], "c96_grid": [91, 94], "c96_om4_025_grid_no_mg_drag_v20160808": [91, 94], "c96l65_am5f3b1r0_pdclim1850f": [84, 86], "c96l65_am5f4b4r0_amip": [84, 89], "c96l65_am5f7c1r0_amip": [91, 94], "call": [6, 9, 13, 17, 26, 30, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 84, 87, 91, 92], "can": [2, 3, 15, 26, 84, 87, 88, 91, 92, 93, 94, 95, 96], "canopi": [46, 91, 93], "capabl": [17, 84, 91, 95, 96], "capfd": [29, 65], "carolyn": 57, "case": [3, 29, 47, 69, 91, 94], "case1": 29, "case2": 29, "catalog": [4, 66, 81, 82, 86, 94], "cc": [91, 95], "cdf": 17, "cdo": [8, 13, 91, 94], "cdotimeaverag": [5, 7], "ceas": [84, 87], "certain": [91, 93, 95], "cf": [91, 94], "chang": [2, 3, 24, 30, 34, 41, 60, 63, 91, 92], "channel": [3, 83, 96], "charact": [91, 95], "cheat": 2, "check": [4, 14, 15, 17, 26, 29, 43, 67, 77, 80, 82, 84, 87, 88, 91], "check_dataset_for_ocean_grid": [4, 25, 26], "check_interp_method": [5, 16, 17], "check_per_component_set": [5, 16, 17], "checkout": [41, 45, 71, 72, 91, 93], "checkout_cr": [4, 35, 36], "checkout_script": [4, 44], "checkout_templ": [4, 44, 45], "chunk": [46, 91, 94], "chunk_from_legaci": [4, 44, 46], "clash": 17, "class": [8, 9, 10, 14, 31, 77], "clean": 77, "clean_yaml": [76, 77], "cleverli": [84, 91, 92], "cli": [3, 9, 30, 83, 84, 87, 91, 93, 96], "click": [2, 3, 26, 30, 31, 41], "climat": [84, 87, 91, 92, 96], "clone": [3, 29, 84, 87, 89, 91, 92, 93], "cmd_name": 31, "cmip": [26, 84, 87], "cmip6": [29, 84, 87], "cmip6_omon": [84, 87], "cmor": [4, 65, 69, 81, 82, 87, 91, 92, 94], "cmor_input_exampl": [84, 87], "cmor_mix": [4, 25, 84, 87], "cmor_run_subtool": [4, 25, 26, 29], "cmorcommand": [84, 87], "cmoriz": [26, 81, 84, 87, 92], "cmorize_target_var_fil": [4, 25, 26], "code": [3, 41, 84, 87, 88, 91, 93], "collect": [84, 85], "com": [3, 31, 83, 91, 93, 94, 96], "comb": 77, "combin": [53, 77, 80, 91, 93, 94, 95], "combine_analysi": [76, 77], "combine_compil": [76, 77], "combine_experi": [76, 77], "combine_model": [76, 77], "combine_platform": [76, 77], "combine_yaml": [4, 76, 80], "combined_compile_existcheck": [4, 76, 77], "come": [0, 91, 94], "command": [3, 31, 46, 81, 91, 92, 95, 96], "comment": 2, "commit": 2, "common": [91, 94], "commonli": [91, 94], "commun": [91, 94], "companion": 96, "compar": 13, "comparison": 29, "compat": [91, 94, 96], "compil": [41, 43, 77, 80, 96], "compile_cr": [4, 35, 37], "compileyaml": [91, 95], "complex": [31, 84, 96], "compliant": [26, 84, 87], "compon": [3, 17, 26, 46, 93, 95], "componenet": [17, 91, 93], "component_list": 17, "comput": 15, "concaten": 77, "concurr": [84, 88], "conda": [3, 83, 96], "conf": [46, 47], "confgur": 53, "config": 17, "configur": [17, 26, 47, 72, 87, 91, 93, 94], "configure_script_xml": [4, 44], "configure_script_yaml": [4, 44, 53], "conjunct": [91, 93], "conserv": 17, "conserve_order1": [91, 94], "conserve_order2": [91, 94], "consid": [3, 96], "consolidate_yaml": [4, 76, 77], "constant": [84, 87], "constructor": [91, 95], "contain": [3, 15, 26, 43, 84, 87, 88, 91, 93, 94, 95, 96], "container": [91, 93], "container_addlib": [91, 93], "containerbuild": [91, 95], "containerrun": [91, 95], "content": [3, 46, 82], "context": [3, 31], "continu": 91, "control": [84, 87], "conveni": [84, 87], "convent": [26, 84, 87], "convers": 46, "convert": 46, "coordin": [15, 84, 87], "copi": [3, 26, 29, 47, 84, 88], "copy_nc": [4, 25, 26], "core": 14, "correct": 3, "correctli": 80, "correspond": [84, 87, 96], "coupl": 96, "cover": [91, 94], "cpdef": [91, 93], "cppdef": [91, 93], "creat": [2, 3, 17, 26, 41, 43, 47, 53, 71, 77, 83, 87, 91, 93, 95], "create_test_conda_env": 2, "create_tmp_dir": [4, 25, 26], "createcheckout": [4, 35], "createcompil": [4, 35], "createdock": [4, 35], "createmakefil": [4, 35], "creation": [80, 84, 88, 91, 93], "criteria": 3, "crunch": 10, "csh": 9, "csv": [84, 86], "ctx": 31, "cube": [91, 94], "cubedspher": [91, 94], "current": [3, 91, 93, 94], "cv": [84, 87], "cylc": [47, 49, 50, 51, 56, 84, 96], "d": [15, 17, 26, 84, 87], "daili": [91, 94], "dana": 41, "dang": 26, "dash": [91, 95], "data": [15, 26, 29, 81, 84, 87, 94, 95], "dataarrai": 15, "dataset": [15, 26, 91, 94], "date": [17, 41, 84, 87, 91, 94], "datetim": [26, 84, 87], "deactiv": [91, 92], "debug": [17, 71, 84, 87, 91, 93], "def_xy_interp": 17, "default": [17, 91, 94], "defici": [91, 94], "defin": [3, 14, 31, 77, 91, 92, 93, 94, 95], "definit": [49, 56, 91, 94, 96], "degre": [91, 94], "deliber": 10, "depend": [2, 3, 26, 84, 87, 88, 91, 93], "deploi": [2, 83, 84, 86, 89, 91, 94, 96], "deploy": 2, "deprec": 3, "deriv": [84, 96], "desc": 26, "descript": [45, 84, 87], "design": [10, 14], "desir": [29, 84, 87, 91, 94], "determin": 26, "develop": [81, 83, 91, 94, 96], "diag": [91, 94], "diagnost": [91, 94], "dictionari": [26, 84, 87, 91, 95], "did": [2, 15], "differ": [3, 17, 26, 29, 45], "digit": 96, "dimens": 26, "dimension": 26, "dir": 26, "directli": [2, 91, 92], "directori": [13, 26, 47, 84, 86, 87, 91, 92, 94, 95], "disabl": 2, "discov": [26, 84], "discoveri": [84, 96], "discuss": [91, 95], "distinct": [91, 94], "distribut": [91, 94], "distrubut": [91, 95], "divis": 96, "do": [2, 3, 15, 47, 53], "do_analysi": 46, "do_refinediag": 46, "doc": [2, 83], "dockerfil": [91, 93, 95], "dockerfile_cr": [4, 35, 38], "doe": [17, 45, 80], "doesnt": 29, "dof90cpp": [91, 93], "domain": [91, 94], "don": 17, "done": [3, 84, 87], "downstream": [84, 87], "ds_p": 15, "dual": 46, "due": 3, "dump": 15, "durat": [17, 46, 91, 94], "duration_to_second": [4, 44, 46], "dure": [84, 87, 91, 94], "e": [2, 3, 17, 26, 30, 46, 72, 84, 87, 88, 89, 91, 94, 96], "each": [26, 84, 87, 91, 93, 94, 95], "earth": 96, "eas": [91, 94], "ecosystem": [84, 91, 92, 94, 96], "edit": [2, 84, 89], "effect": 26, "effici": 3, "either": [91, 95], "element": [46, 91, 95, 96], "els": [91, 95], "empti": [3, 26], "en": 31, "enabl": [84, 96], "encod": 15, "encount": 3, "end": [15, 26, 84, 87, 91, 95], "engin": 96, "enrich": [91, 94], "ensur": [3, 96], "enter": [91, 92], "entir": [26, 91, 94], "entri": [3, 19, 26, 30, 59, 62, 84, 87, 91, 92], "environ": [3, 83, 91, 92, 94, 96], "equal": 26, "error": [17, 26], "esgf": [91, 94], "esm": [91, 94], "essenti": 26, "etc": [3, 91, 92], "even": [3, 91, 94], "everi": 3, "everyth": [3, 91, 95], "ex": 9, "exampl": [86, 89, 91, 93, 94, 95], "exchang": [91, 94], "exec": [91, 95], "execut": [36, 37, 38, 41, 84, 88, 91, 92, 93], "exist": [3, 29, 31, 43, 45, 53, 77, 80], "exp": 46, "exp_config": [84, 87], "expect": 84, "experi": [45, 46, 47, 49, 50, 51, 55, 56, 77, 80, 84, 87, 88, 89, 91, 93, 94, 95, 96], "experiment": [24, 34, 60, 63], "experiment_check": [4, 76, 77], "explicitli": [91, 93], "expos": 26, "extens": 26, "extern": [84, 91, 92], "extract": [77, 84, 87, 91, 95], "face": 96, "facillit": 3, "fail": 80, "fals": [11, 91, 95], "familiar": 3, "faster": 2, "fc": [91, 95], "few": 3, "field": [15, 17, 91, 93, 95], "file": [3, 13, 15, 17, 24, 26, 29, 30, 34, 45, 46, 47, 53, 60, 63, 77, 80, 84, 87, 89, 92, 93, 95], "fileextens": 3, "filenam": [3, 15, 26, 84, 87], "filepath": 46, "final": [26, 77], "find": [2, 26], "fine": [91, 94], "first": 96, "fit": 15, "fix": 15, "flag": [84, 87, 91, 92, 93, 95], "flexibl": 96, "fm": [81, 95, 96], "folder": [3, 45], "follow": [3, 84, 87, 91, 92, 95, 96], "foo": 72, "forg": 83, "fork": 3, "format": [17, 47, 84, 86, 93, 94], "formula": [84, 87], "formula_term": [84, 87], "fortan": [91, 95], "fortran": [9, 91, 95], "forward": 3, "found": [2, 26, 84, 87], "framework": [81, 94, 95], "fre": [83, 85, 86, 87, 88, 89, 91, 92, 93, 94, 95], "fre2": [57, 84, 89], "fre_cli": 13, "fre_nctool": 13, "fre_properti": [91, 93, 95], "freapp": [4, 5], "frecatalog": [4, 18], "frecheck": [4, 22], "frecheckexampl": [4, 22], "frecmor": [4, 25], "free": 3, "fregrid": 17, "frelist": [4, 32, 46], "frelist_xpath": [4, 44, 46], "frelistexampl": [4, 32], "fremak": [4, 35, 41, 91, 93], "fremake_run": [4, 35, 41], "frenc": 9, "frenctoolstimeaverag": [5, 7], "frepp": [4, 44, 57], "frepytool": 10, "frepytoolstimeaverag": [5, 7], "freq": 17, "freq_from_legaci": [4, 44, 46], "freq_to_date_format": [5, 16, 17], "frequenc": [15, 17, 46, 91, 94], "frerun": [4, 58, 91, 94], "frerunexampl": [4, 58], "fretest": [4, 61], "fretestexampl": [4, 61], "freyamltool": [4, 76], "from": [2, 8, 9, 10, 11, 14, 17, 26, 29, 45, 47, 77, 83, 84, 86, 87, 88, 91, 92, 93, 94, 95, 96], "full": [3, 26], "fulli": 14, "function": [3, 7, 8, 10, 11, 13, 14, 24, 26, 30, 34, 60, 63, 84, 87], "further": [84, 87, 91, 93, 95], "futur": [3, 91, 94, 96], "g": [17, 26, 84, 87, 91, 94, 96], "gaea": [83, 91, 93], "gain": 3, "gener": [8, 9, 10, 11, 13, 14, 26, 46, 81, 84, 86, 94], "generate_timavg": [7, 8, 9, 10, 14], "generate_time_averag": [4, 5], "get": [2, 17, 26, 91, 93], "get_combined_compileyaml": [4, 76, 77], "get_combined_ppyaml": [4, 76, 77], "get_command": [4, 31], "get_compile_path": [4, 76, 77], "get_iso_datetim": [4, 25, 26], "get_mosaic_file_nam": [5, 16, 17], "get_mosaic_grid_file_nam": [5, 16, 17], "get_var_filenam": [4, 25, 26], "get_vertical_dimens": [4, 25, 26], "gfdl": [3, 24, 26, 30, 34, 57, 60, 63, 81, 84, 86, 87, 89, 91, 93, 94, 96], "gh": 2, "git": [3, 84, 89, 96], "github": [2, 3, 91, 93, 94, 96], "gitlab": [57, 84, 89, 96], "given": [17, 26, 31, 45, 46, 47, 77], "global": [91, 94], "go": [91, 95], "goal": [91, 94], "good": 0, "gov": [24, 30, 34, 57, 60, 63, 84, 89, 96], "grain": [91, 94], "grid": [17, 26, 84, 87, 91, 94], "grid_label": [84, 87], "grid_spec": 17, "grid_spec_fil": 17, "group": [3, 30, 31, 91, 92, 95], "group_vers": [91, 93], "guid": 3, "guidanc": [91, 92], "ha": [3, 15, 26, 96], "half": 91, "handler": 96, "har": 13, "harvest": 0, "have": [3, 26, 83, 84, 87, 91, 93, 94], "head": 2, "header": [26, 84, 87], "heavili": 3, "help": [26, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 83, 84, 87, 91, 92, 95, 96], "helpdesk": [91, 93], "here": [3, 14, 26], "hint": 14, "histori": [81, 95], "history_seg": [91, 94], "historydir": 46, "hold": [26, 29, 84, 87], "home": [84, 89], "hopelessli": 26, "horizont": [91, 94], "host": 30, "hous": [26, 84, 87], "howev": 3, "hpcme": [71, 91, 93], "http": [2, 3, 31, 57, 83, 91, 93, 94, 96], "i": [2, 3, 14, 17, 26, 29, 30, 41, 43, 47, 77, 80, 81, 83, 84, 86, 87, 88, 91, 92, 93, 94, 95], "ic": [91, 94], "id": [84, 87], "ideal": 3, "ident": 26, "identifi": [26, 50], "implement": 3, "implemet": 3, "import": [3, 7, 17, 25, 84, 87], "in_nc": 26, "includ": [3, 26, 91, 93, 94, 95], "incorrect": 80, "increment": 96, "independ": [91, 94], "index": 81, "indic": [55, 91, 95], "indir": [26, 84, 87], "individu": [91, 95], "infil": [8, 9, 10, 11, 13, 14], "inform": [26, 77, 84, 87, 91, 93, 95], "ing": [84, 87], "inherit": [8, 9, 10, 14, 91, 95], "init_compile_yaml": [4, 76, 77], "init_pp_yaml": [4, 76, 77], "initi": 47, "input": [13, 15, 17, 26, 29, 46, 84, 86, 87, 91, 94], "input_dir": 17, "input_mosa": 17, "inputrealm": [91, 94], "insid": [3, 26, 91, 95], "inspect": [91, 94], "inspir": [84, 96], "instal": [3, 4, 44, 72, 83, 96], "install_subtool": [4, 44, 49], "instead": [91, 92], "instruct": [83, 91, 93, 94], "intak": [91, 94], "integr": [24, 34, 60, 63], "intel23": [71, 91, 93], "intend": [84, 85], "interact": 2, "intercomparison": [84, 91, 92], "interfac": [81, 91, 94], "interp_method": 17, "interpmethod": [91, 94], "interpol": [91, 94], "intput": 26, "inwhich": [84, 87], "io": 2, "iso": [17, 26, 84, 87], "iso8601": [46, 91, 94], "iso_datetime_arr": 26, "iso_freq": 17, "issu": 3, "item": 47, "iter": [84, 87], "its": 3, "j": [84, 88], "job": [36, 37, 41, 84, 88, 96], "join": [91, 93, 95], "join_constructor": [4, 76, 77], "json": [26, 47, 84, 86, 87], "json_exp_config": 26, "json_table_config": 26, "json_var_list": 26, "jupyt": 96, "just": [91, 92, 94], "kei": [3, 26, 46, 84, 87, 91, 95], "kept": [84, 87], "kick": [91, 93], "known": [91, 94], "kwarg": 31, "l": [84, 87], "label": [26, 84, 87, 91, 94], "land": [91, 94], "larg": [84, 87], "larger": 15, "last": [2, 3, 96], "lat": [91, 94], "later": [84, 87], "latest": [83, 96], "latter": 91, "layer": [91, 94], "lazi": 31, "lazy_group": [4, 82], "lazy_subcommand": 31, "lazygroup": [4, 31, 82], "left": 2, "legaci": 17, "legacy_chunk": 46, "legacy_freq": 46, "length": 17, "less": [91, 94], "level": 15, "leverag": [84, 91, 92], "lib": 15, "librari": [91, 93], "like": [3, 17, 91, 92, 95], "line": [3, 46, 81, 91, 92, 93, 96], "link": [3, 91, 93], "linker": [91, 93], "linux": [84, 96], "list": [4, 17, 26, 31, 70, 77, 82, 84, 87, 91, 92, 93, 95], "list_command": [4, 31], "load": [31, 47, 77, 83, 91, 95], "loader": 77, "local": [3, 17, 26, 84, 87], "local_var": 26, "locat": [3, 26, 49, 56, 57, 84, 88, 91, 93, 95], "lon": [91, 94], "long": [91, 92], "longtim": [91, 94], "look": 13, "loop": 17, "lord": 26, "low": 15, "macro": 26, "made": [3, 91, 95], "mai": [2, 3, 83, 91, 92, 95], "main": [2, 3, 4, 5, 16, 17, 24, 26, 30, 34, 44, 46, 57, 60, 63, 77, 80, 91, 93, 95], "mainfunctionoftoolcommand": 3, "maintain": 3, "mainyaml_dir": 77, "major": 96, "make": [2, 3, 4, 17, 26, 29, 53, 71, 80, 81, 82, 88, 91, 92, 93, 95, 96], "make_component_list": [5, 16, 17], "make_regrid_var_list": [5, 16, 17], "makefil": [43, 91, 93], "makefile_cr": [4, 35, 39], "makeoverrid": [91, 93], "malyshev": [84, 87], "manag": [91, 94], "manipul": 26, "manner": [10, 26, 84, 87], "mask": [15, 17], "mask_atmos_plevel": [4, 5], "mask_field_above_surface_pressur": [4, 5, 15], "match": [3, 26], "md": 26, "meat": 46, "merg": [77, 80], "messag": [91, 92], "meta": 3, "metadata": [26, 29, 84, 87, 91, 94], "metal": [43, 91, 93, 95], "metdata": [84, 87], "method": [91, 94], "might": [91, 94], "min": 2, "miniconda": 83, "miniforg": 83, "minim": [84, 86, 89], "minor": 96, "mip": [26, 84, 87, 91, 92], "miss": [84, 91, 92], "misspel": 80, "mk": [91, 95], "mkmf": [91, 95], "mktemplat": [91, 95], "model": [41, 43, 77, 80, 81, 84, 87, 88, 92, 93, 94, 96], "model_gen5": [91, 94], "modelroot": [91, 95], "modern": [84, 96], "modil": [84, 88], "modul": [3, 81, 82, 83, 84, 86, 89, 91, 92, 95, 96], "modulefil": 96, "modulesinit": [91, 95], "mom6": [91, 93], "month": [91, 94], "monthli": [13, 91, 94], "more": [3, 84, 87, 91, 94, 95, 96], "mosaic": 17, "mosaic_typ": 17, "most": [17, 26, 91, 94], "mostli": 8, "move": 26, "movement": 26, "msd": 96, "multi": [91, 93], "multipl": [3, 17, 84, 91, 93, 96], "must": [3, 14, 47, 91, 93, 95], "n": [84, 86, 88, 89], "name": [2, 3, 17, 26, 29, 31, 77, 83, 84, 87, 89, 91, 93, 94, 95, 96], "name_of_set": 26, "namespac": [46, 91, 94], "nativ": [10, 91, 94], "navig": 2, "nc": [17, 26, 91, 94], "nc_fl": 26, "nc_variabl": 17, "ncdump": [91, 94], "nco": [91, 94], "ncrc5": [71, 84, 86, 89, 91, 93, 94], "nctool": [9, 17, 96], "necessari": [84, 88, 91, 93, 95], "need": [3, 26, 84, 86, 89, 91, 92, 93, 95], "needs_atmos_mask": 15, "nest": 17, "net": 17, "netcdf": [15, 26, 84, 87, 91, 94], "netcdf4": [10, 26], "netcdf_fil": 26, "never": 14, "new": [2, 81, 91, 94, 96], "newest": 3, "newlin": [91, 95], "next": 96, "nine": 26, "no_parallel_checkout": [36, 41], "noaa": [3, 24, 30, 34, 57, 60, 63, 83, 84, 89, 91, 93, 94, 96], "node": 77, "non": [3, 26], "none": [8, 9, 10, 11, 13, 14, 17, 26, 31, 45], "nor": 2, "normal": 3, "notat": 46, "note": [2, 81, 91, 93], "notebook": 96, "now": [3, 91, 94], "npc": [84, 88, 91, 93], "number": [84, 88, 96], "numpi": [10, 17], "o": [29, 84, 86, 87], "oar": [91, 94], "object": [14, 26, 31, 77], "objet": 26, "ocean": [26, 91, 94], "ocean_annu": [91, 94], "ocean_month": [91, 94], "ocean_sos_var_fil": [84, 87], "off": [91, 93], "often": [26, 91, 94], "omon": 26, "onc": [3, 83, 91, 92], "one": [2, 3, 14, 17, 24, 26, 34, 60, 63, 84, 87, 91, 93, 94, 96], "ones": [91, 95], "onli": [2, 15, 26, 29, 84, 87], "onto": 17, "opaqu": 26, "open": 26, "openmp": [84, 86, 89, 91, 93, 94], "oper": 26, "opt_var_nam": [26, 84, 87], "option": [3, 15, 17, 26, 84, 87, 88, 91, 92, 93, 94, 96], "optiondn": [65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75], "order": [31, 84, 91, 92, 93, 95], "org": [3, 83], "organ": [91, 94], "orient": [84, 87], "origin": [26, 46, 84, 87, 96], "ot": [91, 93], "other": [10, 13, 84, 87, 88, 91, 93, 95, 96], "otherflag": [91, 93], "out": [2, 3, 26, 84, 87, 88], "out_nc": 26, "outdir": [26, 84, 87], "outfil": [8, 9, 10, 11, 13, 14, 15], "output": [3, 17, 26, 81, 84, 86, 87, 92, 94, 95, 96], "output_dir": 17, "outputfil": 29, "outsid": 83, "over": [84, 87, 96], "overrid": [91, 93, 95], "overwrit": [46, 84, 86, 91, 95], "p": [46, 71, 72, 84, 87, 88, 89, 91, 93], "p1y": [91, 94], "packag": [3, 8, 10, 11, 82, 84, 91, 92, 93], "page": [2, 81], "pair": [26, 46, 84, 87, 91, 95], "palletsproject": 31, "parallel": [37, 41, 84, 88, 91, 93], "paramet": [3, 84, 91, 92], "pars": [91, 93, 95], "part": [3, 96], "parti": 10, "particip": [84, 87], "pass": 26, "pass_context": 3, "past": 3, "patch": 96, "path": [26, 46, 77, 80, 83, 84, 86, 87, 91, 93, 94, 95], "per": [17, 26, 91, 94], "pip": 3, "pkg": [7, 8, 9, 10, 11, 13, 14], "placehold": [84, 90], "platform": [36, 37, 38, 39, 41, 43, 45, 46, 47, 49, 50, 51, 55, 56, 77, 84, 88, 89, 93], "platformyaml": [91, 95], "pleas": [3, 91], "plug": [91, 94], "podman": [91, 93, 95], "point": [3, 19, 26, 30, 59, 62, 84, 87, 91, 92], "popul": [91, 93], "posit": [84, 91, 92], "possibl": 96, "post": 26, "post_writ": [4, 5, 15], "posteriori": 15, "postprocess": [81, 84, 89, 95, 96], "potenti": 3, "pp": [4, 72, 81, 82, 83, 86, 89, 91, 92, 94, 95, 96], "pp_grid_spec": [91, 94], "pp_run_subtool": [4, 44, 50], "pp_start": [46, 91, 94], "pp_stop": [46, 91, 94], "ppdir": 46, "practic": 96, "pre": 10, "predict": [91, 94], "prefer": [3, 15], "prefix": [84, 87], "prepar": 15, "preprocess": [4, 5, 15], "preprocessor": [91, 93], "present": [3, 15, 17, 29, 91, 94], "pressur": 15, "pressure_coordin": [4, 5, 15], "pressure_var": 15, "presum": 26, "prev": 29, "previous": 96, "primari": [26, 46, 84, 87], "primarili": 96, "princip": 30, "print": 17, "process": [26, 84, 87, 92], "prod": [84, 86, 89, 91, 93, 94], "produc": 15, "program": [30, 91, 95], "programmat": [91, 94], "proj_table_var": 26, "project": [84, 91, 92], "properli": 15, "properti": 77, "prototyp": [24, 34, 60, 63], "provid": [3, 84, 87, 91, 92], "public": [91, 94], "publish": [3, 84, 87], "publish_conda": 3, "purpos": [84, 85, 88], "push": [2, 3], "py": [3, 26, 57, 84, 87, 91, 92], "pylint": 2, "pytest": [2, 17, 29], "python": [3, 8, 10, 26, 84, 91, 92, 94, 96], "quantiti": 14, "quickli": 17, "quiet": 46, "quit": [84, 91, 92], "quot": 47, "quote_rose_valu": [4, 44, 47], "r": [84, 87], "ran": [91, 92], "rang": [91, 94], "raw": [91, 94], "re": [2, 83], "reactiv": [91, 92], "read": [17, 26], "readi": [91, 94], "readm": [26, 84, 91, 92], "recommend": [3, 91, 95], "recurs": 3, "reduct": 15, "reengin": 96, "ref": [2, 17], "refer": [3, 26, 84, 91, 92, 93, 95], "refinediag": 15, "refinedir": 46, "reflect": 3, "regardless": 3, "regrid": [17, 46, 47, 53], "regrid_xi": [4, 5], "reimagin": [91, 94], "relat": [14, 91, 95], "releas": 96, "relev": 17, "remain": 3, "remap": [17, 46, 47, 53], "remap_dir": 17, "remot": 2, "remov": [29, 77], "renam": [91, 94], "repeat": [91, 95], "replac": 57, "replic": 3, "repo": [2, 29, 45, 91, 93, 95], "report": 51, "repositori": [84, 87, 89, 91, 92, 93, 96], "repres": [26, 91, 94], "represent": 26, "represnt": 26, "request": [26, 84, 87], "requir": [3, 7, 15, 26, 46, 84, 87, 88, 91, 92, 93, 94, 95], "resid": 3, "respect": 3, "rest": [84, 91, 92], "restart": 50, "restructur": 2, "retain": 96, "retriev": [91, 93, 96], "return": [15, 26, 31, 46], "reuasbl": [91, 95], "reusabl": [91, 95], "reusablevari": [91, 95], "rewrit": [26, 84, 87, 91, 92], "rewrite_netcdf_file_var": [4, 25, 26], "rewritten": [84, 87, 91, 94], "richer": [91, 94], "right": 26, "river": 96, "robinson": 41, "root": [3, 26, 84, 87, 91, 95], "rose": [17, 46, 47, 53], "rose_app_cfg": 17, "rose_init": [4, 44, 47], "rose_regrid": 47, "rose_remap": 47, "rose_suit": 47, "routin": [8, 10, 26, 29, 80, 84, 87], "rsa": 3, "rule": 26, "run": [3, 4, 26, 29, 41, 44, 49, 65, 69, 72, 73, 81, 82, 92, 93, 94, 95, 96], "run_avgtype_pkg_calcul": [7, 12, 13], "rundown": [91, 92], "runenv": [91, 95], "runfremak": [4, 35], "runtim": 96, "safe_rose_config_get": [5, 16, 17], "sai": 3, "sake": 3, "same": [3, 15, 17, 26, 91, 94, 95, 96], "satisfi": 3, "save": [3, 91, 94], "scalar": [17, 91, 94], "scenario": 45, "schema": [47, 80], "scheme": [17, 96], "script": [3, 9, 15, 17, 30, 45, 47, 53, 57, 77, 84, 87, 88, 93, 95, 96], "search": 81, "searchabl": [91, 94], "season": 13, "second": [46, 96], "section": [3, 17, 77, 91, 93, 95], "see": [3, 8, 26, 31, 84, 91, 92], "seen": [91, 93], "segment": [91, 94], "select": 2, "semant": 96, "separ": 3, "sergei": [84, 87], "serv": 3, "set": [15, 17, 47, 57, 84, 91, 92, 93, 94, 95, 96], "set_netcdf_encod": [4, 5, 15], "set_rose_app": [4, 44, 47], "set_rose_suit": [4, 44, 47], "setup": [3, 29, 81, 91, 92], "share": [91, 94, 95], "shared_directori": [91, 95], "shared_set": [91, 95], "shared_switch": [91, 95], "sheet": 2, "shell": [84, 96], "should": [2, 3, 31, 80, 84, 87, 91, 93], "shown": [91, 92, 94], "shutil": 26, "similarli": [91, 94], "simul": [91, 94], "simultn": [84, 88], "singh": 41, "singl": [84, 85, 91, 94], "six": 26, "size": [91, 94], "skip": 83, "slightli": [26, 29], "small": 2, "so": [2, 17, 26, 91, 93], "solut": 3, "some": [26, 84, 96], "someth": [3, 91, 92], "somewher": 2, "soon": 0, "sort": [91, 94], "sourc": [2, 17, 84, 88, 91, 94], "sourcegrid": [91, 94], "specif": [2, 26, 77, 83, 84, 87, 91, 92, 95, 96], "specifi": [3, 26, 84, 87, 91, 92, 94], "sphere": [91, 94], "spheric": 17, "src": [47, 49, 56, 91, 93, 95], "ssh": 3, "stai": 96, "stamp": [91, 94], "standalon": [91, 94], "standard": [83, 91, 94], "start": 50, "starter": 55, "state": 51, "statist": [10, 14], "statu": [4, 44, 72], "status_subtool": [4, 44, 51], "stddev_typ": [7, 8, 9, 10, 11, 13, 14], "steer": [11, 17, 26], "stem": [91, 95], "step": [2, 3, 91, 95], "str": [14, 46], "string": [17, 26, 80, 84, 87, 91, 93, 95], "structur": [14, 26, 84, 87, 91, 94, 95, 96], "style": [31, 46], "sub": 84, "subcommand": [3, 19, 31, 59, 62, 84, 91, 92, 96], "subdirectori": 3, "subgroup": 30, "subject": 3, "submit": [91, 93], "submodul": 82, "subpackag": 82, "subprocess": [9, 46], "subsect": [91, 95], "subtool": [3, 96], "succeed": [84, 87], "success": [2, 53], "successfulli": [91, 93], "successor": 96, "suffici": 26, "suit": [46, 47, 53], "support": [91, 93, 96], "sure": [2, 3, 26, 29, 53, 80, 91, 95], "surfac": 15, "switch": [91, 94, 95], "syntax": [3, 84, 86, 89], "system": [57, 81, 96], "t": [3, 17, 46, 71, 72, 84, 88, 89, 91, 93], "tabl": [26, 29, 84, 87], "table_config": [84, 87], "tag": 96, "take": [91, 94], "tar": [91, 94], "tarfil": [91, 94], "target": [17, 26, 36, 37, 38, 39, 41, 45, 46, 47, 49, 50, 51, 55, 56, 77, 84, 87, 88, 89, 91, 93], "target_fil": 17, "target_var": 26, "task": [3, 46, 55], "tavg": 9, "team": 96, "templat": [3, 15, 17, 45, 47, 84, 87, 89, 91, 95], "tempor": [91, 94], "temporari": [84, 88], "term": [84, 87], "test": [3, 4, 5, 7, 17, 18, 25, 35, 44, 76, 82, 84, 87], "test_analysisyaml_exist": [76, 79, 80], "test_bm_makefile_cr": [35, 42, 43], "test_boolean": [44, 52, 54], "test_cdo_time_avg": [7, 12, 13], "test_cdo_time_unwgt_avg": [7, 12, 13], "test_cdo_time_unwgt_stddev": [7, 12, 13], "test_cli_fr": [4, 64, 68], "test_cli_fre_app": [4, 64, 65], "test_cli_fre_app_gen_time_averag": [4, 64, 65], "test_cli_fre_app_gen_time_averages_help": [4, 64, 65], "test_cli_fre_app_gen_time_averages_opt_dn": [4, 64, 65], "test_cli_fre_app_help": [4, 64, 65], "test_cli_fre_app_opt_dn": [4, 64, 65], "test_cli_fre_app_regrid": [4, 64, 65], "test_cli_fre_app_regrid_help": [4, 64, 65], "test_cli_fre_app_regrid_opt_dn": [4, 64, 65], "test_cli_fre_app_regrid_test_case_1": [4, 64, 65], "test_cli_fre_catalog": [4, 64, 66], "test_cli_fre_catalog_build": [4, 64, 66], "test_cli_fre_catalog_builder_help": [4, 64, 66], "test_cli_fre_catalog_help": [4, 64, 66], "test_cli_fre_catalog_merg": [4, 64, 66], "test_cli_fre_catalog_merge_help": [4, 64, 66], "test_cli_fre_catalog_opt_dn": [4, 64, 66], "test_cli_fre_check": [4, 64, 67], "test_cli_fre_check_help": [4, 64, 67], "test_cli_fre_check_opt_dn": [4, 64, 67], "test_cli_fre_cmor": [4, 64, 69], "test_cli_fre_cmor_help": [4, 64, 69], "test_cli_fre_cmor_opt_dn": [4, 64, 69], "test_cli_fre_cmor_run": [4, 64, 69], "test_cli_fre_cmor_run_case1": [4, 64, 69], "test_cli_fre_cmor_run_case2": [4, 64, 69], "test_cli_fre_cmor_run_help": [4, 64, 69], "test_cli_fre_cmor_run_opt_dn": [4, 64, 69], "test_cli_fre_help": [4, 64, 68], "test_cli_fre_list": [4, 64, 70], "test_cli_fre_list_help": [4, 64, 70], "test_cli_fre_list_opt_dn": [4, 64, 70], "test_cli_fre_mak": [4, 64, 71], "test_cli_fre_make_create_checkout_baremet": [4, 64, 71], "test_cli_fre_make_create_checkout_contain": [4, 64, 71], "test_cli_fre_make_help": [4, 64, 71], "test_cli_fre_make_opt_dn": [4, 64, 71], "test_cli_fre_option_dn": [4, 64, 68], "test_cli_fre_pp": [4, 64, 72], "test_cli_fre_pp_checkout": [4, 64, 72], "test_cli_fre_pp_checkout_cas": [4, 64, 72], "test_cli_fre_pp_checkout_help": [4, 64, 72], "test_cli_fre_pp_checkout_opt_dn": [4, 64, 72], "test_cli_fre_pp_configure_xml": [4, 64, 72], "test_cli_fre_pp_configure_xml_help": [4, 64, 72], "test_cli_fre_pp_configure_xml_opt_dn": [4, 64, 72], "test_cli_fre_pp_configure_yaml": [4, 64, 72], "test_cli_fre_pp_configure_yaml_fail1": [4, 64, 72], "test_cli_fre_pp_configure_yaml_help": [4, 64, 72], "test_cli_fre_pp_configure_yaml_opt_dn": [4, 64, 72], "test_cli_fre_pp_help": [4, 64, 72], "test_cli_fre_pp_instal": [4, 64, 72], "test_cli_fre_pp_install_help": [4, 64, 72], "test_cli_fre_pp_install_opt_dn": [4, 64, 72], "test_cli_fre_pp_opt_dn": [4, 64, 72], "test_cli_fre_pp_run": [4, 64, 72], "test_cli_fre_pp_run_help": [4, 64, 72], "test_cli_fre_pp_run_opt_dn": [4, 64, 72], "test_cli_fre_pp_statu": [4, 64, 72], "test_cli_fre_pp_status_help": [4, 64, 72], "test_cli_fre_pp_status_opt_dn": [4, 64, 72], "test_cli_fre_pp_valid": [4, 64, 72], "test_cli_fre_pp_validate_help": [4, 64, 72], "test_cli_fre_pp_validate_opt_dn": [4, 64, 72], "test_cli_fre_pp_wrapp": [4, 64, 72], "test_cli_fre_pp_wrapper_help": [4, 64, 72], "test_cli_fre_pp_wrapper_opt_dn": [4, 64, 72], "test_cli_fre_run": [4, 64, 73], "test_cli_fre_run_help": [4, 64, 73], "test_cli_fre_run_opt_dn": [4, 64, 73], "test_cli_fre_test": [4, 64, 74], "test_cli_fre_test_help": [4, 64, 74], "test_cli_fre_test_opt_dn": [4, 64, 74], "test_cli_fre_yamltool": [4, 64, 75], "test_cli_fre_yamltools_help": [4, 64, 75], "test_cli_fre_yamltools_opt_dn": [4, 64, 75], "test_cmor_run_subtool": [25, 28], "test_combine_yaml": [76, 79], "test_combined_compileyaml_combinefail": [76, 79, 80], "test_combined_compileyaml_valid": [76, 79, 80], "test_combined_compileyaml_validatefail": [76, 79, 80], "test_combined_ppyaml_valid": [76, 79, 80], "test_combinedyaml_exist": [44, 52, 53], "test_compare_cdo_to_fre_nctool": [7, 12, 13], "test_compare_fre_cli_to_cdo": [7, 12, 13], "test_compare_fre_cli_to_fre_nctool": [7, 12, 13], "test_compare_unwgt_fre_cli_to_unwgt_cdo": [7, 12, 13], "test_compileyaml_exist": [35, 42, 43, 76, 79, 80], "test_configure_script": [44, 52, 53], "test_configure_script_yaml": [44, 52], "test_container_makefile_cr": [35, 42, 43], "test_create_makefil": [35, 42], "test_expyaml_exist": [76, 79, 80], "test_fil": [84, 87], "test_fre_app_cli": [4, 64], "test_fre_catalog": [18, 20], "test_fre_catalog_cli": [4, 64], "test_fre_catalog_import": [18, 20, 21], "test_fre_check_cli": [4, 64], "test_fre_cli": [4, 64], "test_fre_cli_time_avg": [7, 12, 13], "test_fre_cli_time_avgs_stddev": [7, 12, 13], "test_fre_cli_time_unwgt_avg": [7, 12, 13], "test_fre_cli_time_unwgt_avgs_stddev": [7, 12, 13], "test_fre_cmor_cli": [4, 64], "test_fre_cmor_run_subtool_case1": [25, 28, 29], "test_fre_cmor_run_subtool_case1_output_compare_data": [25, 28, 29], "test_fre_cmor_run_subtool_case1_output_compare_metadata": [25, 28, 29], "test_fre_cmor_run_subtool_case2": [25, 28, 29], "test_fre_cmor_run_subtool_case2_output_compare_data": [25, 28, 29], "test_fre_cmor_run_subtool_case2_output_compare_metadata": [25, 28, 29], "test_fre_list_cli": [4, 64], "test_fre_make_cli": [4, 64], "test_fre_pp_cli": [4, 64], "test_fre_run_cli": [4, 64], "test_fre_test_cli": [4, 64], "test_fre_yamltools_cli": [4, 64], "test_generate_time_averag": [7, 12], "test_import": [5, 16, 17], "test_merged_compile_yaml": [76, 79, 80], "test_merged_pp_yaml": [76, 79, 80], "test_modelyaml_exist": [35, 42, 43, 76, 79, 80], "test_monthly_cdo_time_unwgt_avg": [7, 12, 13], "test_monthly_cdo_time_unwgt_stddev": [7, 12, 13], "test_platformyaml_exist": [35, 42, 43, 76, 79, 80], "test_rose_quot": [44, 52], "test_seasonal_cdo_time_unwgt_avg": [7, 12, 13], "test_seasonal_cdo_time_unwgt_stddev": [7, 12, 13], "test_setup_cmor_cmip_table_repo": [25, 28, 29], "test_setup_fre_cmor_run_subtool": [25, 28, 29], "test_setup_fre_cmor_run_subtool_case2": [25, 28, 29], "test_str": [44, 52, 54], "test_time_avg_file_dir_exist": [7, 12, 13], "test_time_avg_input_file_exist": [7, 12, 13], "text": [2, 84, 87], "tf": [91, 94], "than": [15, 91, 94], "thei": [31, 47, 91, 92, 93, 95], "them": [47, 91, 95], "theres": 2, "thi": [2, 3, 14, 15, 17, 26, 29, 30, 31, 53, 84, 87, 91, 92, 93, 94, 95, 96], "think": 26, "third": 10, "those": [26, 91, 95], "though": 96, "three": 26, "through": [91, 93, 96], "throughout": [91, 95], "throw": 26, "ticket": [91, 93], "tile": [91, 94], "tile1": [91, 94], "tile2": [91, 94], "tile3": [91, 94], "tile4": [91, 94], "tile5": [91, 94], "tile6": [91, 94], "timavg": 9, "time": [2, 3, 8, 9, 10, 11, 13, 14, 17, 55, 91, 94, 96], "timeaverag": [5, 7, 8, 9, 10, 91, 94], "tmp": 26, "tmp_dir": [17, 26], "to_netcdf": 15, "togeth": 96, "tom": 41, "tool": [9, 11, 26, 81, 83, 85, 91, 92, 93, 94, 95, 96], "toolcommandscript": 3, "top": [2, 84, 87], "tradit": [91, 93, 94], "tree": [84, 87, 91, 93], "trigger": [4, 44], "tripolar": [17, 91, 94], "trivial": [2, 26], "true": [84, 87, 91, 93, 95], "truncat": 17, "truncate_d": [5, 16, 17], "try": 3, "tune": 96, "turn": 2, "two": [26, 84, 87, 96], "type": [3, 91, 94, 95], "typic": [3, 84, 87], "u": 26, "ue2": [84, 89], "under": [2, 3, 91, 93, 95], "underbak": [91, 94], "underneath": 96, "uniqu": [17, 26], "unit": 14, "unless": [47, 91, 93], "unnecessari": 77, "unweight": 13, "unwgt": [7, 8, 9, 10, 11, 13, 14], "up": 53, "updat": [3, 91, 94, 96], "upon": 26, "url": [91, 93], "us": [3, 8, 9, 10, 13, 15, 17, 26, 29, 31, 41, 43, 46, 47, 69, 77, 84, 87, 91, 92, 93, 94, 95, 96], "usag": [26, 46, 81], "user": [2, 3, 46, 84, 86, 89, 91, 92, 93, 94, 95, 96], "usu": 10, "usual": [91, 94], "util": [9, 26, 84, 87, 91, 95], "v": [84, 87], "valid": [4, 17, 44, 46, 47, 72, 80], "validate_subtool": [4, 44, 56], "validate_yaml": [4, 44, 47], "valu": [17, 26, 46, 47, 84, 87, 91, 95], "value1": [91, 95], "value2": [91, 95], "var": [7, 8, 9, 10, 11, 14, 15], "var_filenam": 26, "var_has_time_unit": [7, 14], "var_with_bound": 15, "variabl": [14, 17, 26, 47, 84, 87, 91, 94, 95], "variable1": [91, 95], "variable2": [91, 95], "variou": 11, "varlist": [84, 87], "varnam": 15, "vector": 17, "verbos": [36, 37, 41, 46], "version": [13, 83, 84, 87, 91, 93, 95, 96], "vertic": [26, 91, 94], "via": [8, 30, 41, 46], "virtual": [91, 94], "vocabulari": [84, 87], "wa": [80, 84, 87, 96], "wai": 3, "want": [91, 95], "warn": 17, "we": [2, 26], "webpag": 2, "weight": [8, 13], "welcom": 2, "well": [91, 95], "were": [26, 91, 94, 96], "what": [3, 81], "when": [2, 3, 43], "where": [2, 3, 17, 91, 95], "which": [3, 26, 45, 47, 84, 87, 91, 94, 96], "while": [31, 91, 94], "whitlock": 57, "whole": 26, "why": 17, "wish": 26, "within": [3, 17, 26, 84, 87, 91, 92, 94], "without": [91, 92, 93], "won": 3, "work": [84, 87], "workflow": [3, 45, 47, 49, 50, 51, 56, 84, 89, 96], "workstat": 83, "worth": [91, 94], "would": [3, 26, 91, 92, 96], "wrapper": [4, 44, 72], "write_dataset": [4, 5, 15], "x": [31, 46], "xarrai": 15, "xml": [46, 72, 91, 93, 95], "xpath": 46, "xy": 46, "xyinterp": [91, 94], "y": [71, 84, 88, 89, 91, 93], "yaml": [3, 43, 47, 53, 71, 72, 77, 80, 81, 88, 89, 94], "yaml_load": [4, 44, 47, 76, 77], "yamlfil": [36, 37, 38, 39, 41, 47, 77, 84, 88], "yamlinfo": [4, 44, 47], "yamltool": [4, 75, 81, 82], "year": [91, 94, 96], "yet": [3, 91, 96], "yml": [2, 3, 77], "you": [2, 3, 83, 91, 95], "your": [3, 83, 91, 94], "your_usernam": 2, "yyyymmdd": [91, 94]}, "titles": ["API", "Badges", "Documentation-Documentation", "For developers", "fre package", "fre.app package", "fre.app.freapp module", "fre.app.generate_time_averages package", "fre.app.generate_time_averages.cdoTimeAverager module", "fre.app.generate_time_averages.frenctoolsTimeAverager module", "fre.app.generate_time_averages.frepytoolsTimeAverager module", "fre.app.generate_time_averages.generate_time_averages module", "fre.app.generate_time_averages.tests package", "fre.app.generate_time_averages.tests.test_generate_time_averages module", "fre.app.generate_time_averages.timeAverager module", "fre.app.mask_atmos_plevel module", "fre.app.regrid_xy package", "fre.app.regrid_xy.regrid_xy module", "fre.catalog package", "fre.catalog.frecatalog module", "fre.catalog.tests package", "fre.catalog.tests.test_fre_catalog module", "fre.check package", "fre.check.frecheck module", "fre.check.frecheckexample module", "fre.cmor package", "fre.cmor.cmor_mixer module", "fre.cmor.frecmor module", "fre.cmor.tests package", "fre.cmor.tests.test_cmor_run_subtool module", "fre.fre module", "fre.lazy_group module", "fre.list package", "fre.list.frelist module", "fre.list.frelistexample module", "fre.make package", "fre.make.createCheckout module", "fre.make.createCompile module", "fre.make.createDocker module", "fre.make.createMakefile module", "fre.make.fremake module", "fre.make.runFremake module", "fre.make.tests package", "fre.make.tests.test_create_makefile module", "fre.pp package", "fre.pp.checkout_script module", "fre.pp.configure_script_xml module", "fre.pp.configure_script_yaml module", "fre.pp.frepp module", "fre.pp.install module", "fre.pp.run module", "fre.pp.status module", "fre.pp.tests package", "fre.pp.tests.test_configure_script_yaml module", "fre.pp.tests.test_rose_quoting module", "fre.pp.trigger module", "fre.pp.validate module", "fre.pp.wrapper module", "fre.run package", "fre.run.frerun module", "fre.run.frerunexample module", "fre.test package", "fre.test.fretest module", "fre.test.fretestexample module", "fre.tests package", "fre.tests.test_fre_app_cli module", "fre.tests.test_fre_catalog_cli module", "fre.tests.test_fre_check_cli module", "fre.tests.test_fre_cli module", "fre.tests.test_fre_cmor_cli module", "fre.tests.test_fre_list_cli module", "fre.tests.test_fre_make_cli module", "fre.tests.test_fre_pp_cli module", "fre.tests.test_fre_run_cli module", "fre.tests.test_fre_test_cli module", "fre.tests.test_fre_yamltools_cli module", "fre.yamltools package", "fre.yamltools.combine_yamls module", "fre.yamltools.freyamltools module", "fre.yamltools.tests package", "fre.yamltools.tests.test_combine_yamls module", "Welcome to fre-cli
\u2019s documentation!", "fre", "Setup", "Tools", "<no title>", "builder
", "run
", "create-checkout
", "configure
", "combine-yamls
", "Usage", "<no title>", "Compile Yaml", "FMS history files", "Yaml Formatting", "What is FRE?"], "titleterms": {"": [2, 81], "For": 3, "On": 83, "ad": 3, "analysi": [91, 94], "api": 0, "app": [5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 84], "argument": [84, 87], "background": [84, 87], "badg": 1, "build": 91, "builder": [84, 86], "catalog": [18, 19, 20, 21, 84, 91], "cdotimeaverag": 8, "check": [22, 23, 24], "checklist": 3, "checkout": [84, 88, 89], "checkout_script": 45, "cli": [2, 81], "climatologi": [91, 94], "cmor": [25, 26, 27, 28, 29, 84], "cmor_mix": 26, "cmoriz": 91, "combin": [84, 90], "combine_yaml": 77, "command": 84, "compil": [84, 88, 91, 93, 95], "compon": [91, 94], "configur": [84, 89], "configure_script_xml": 46, "configure_script_yaml": 47, "content": [4, 5, 7, 12, 16, 18, 20, 22, 25, 28, 32, 35, 42, 44, 52, 58, 61, 64, 76, 79, 81], "contribut": 2, "creat": [84, 88], "createcheckout": 36, "createcompil": 37, "createdock": 38, "createmakefil": 39, "data": 91, "develop": 3, "directori": 3, "dockerfil": [84, 88], "document": [2, 3, 81], "enabl": 2, "exampl": [3, 84, 87], "file": [91, 94], "first": 2, "fm": [91, 94], "fork": 2, "format": [91, 95], "framework": 91, "fre": [2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 84, 96], "freapp": 6, "frecatalog": 19, "frecheck": 23, "frecheckexampl": 24, "frecmor": 27, "frelist": 33, "frelistexampl": 34, "fremak": [40, 84, 88], "frenctoolstimeaverag": 9, "frepp": 48, "frepytoolstimeaverag": 10, "frerun": 59, "frerunexampl": 60, "fretest": 62, "fretestexampl": 63, "freyamltool": 78, "from": 3, "gener": [83, 91], "generate_time_averag": [7, 8, 9, 10, 11, 12, 13, 14], "gfdl": 83, "guid": [91, 93], "help": 2, "histori": [91, 94], "how": 2, "i": 96, "indic": 81, "instal": 49, "interfac": 84, "lazy_group": 31, "legaci": [91, 94], "level": [91, 94], "line": 84, "list": [32, 33, 34], "make": [35, 36, 37, 38, 39, 40, 41, 42, 43, 84], "makefil": [84, 88], "manifest": 3, "mask": [91, 94], "mask_atmos_plevel": 15, "model": [91, 95], "modul": [4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80], "new": 3, "note": 84, "other": [2, 3], "output": 91, "packag": [4, 5, 7, 12, 16, 18, 20, 22, 25, 28, 32, 35, 42, 44, 52, 58, 61, 64, 76, 79], "platform": [91, 95], "poke": 2, "post": [91, 95], "postprocess": [91, 94], "pp": [44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 84], "pressur": [91, 94], "process": [91, 95], "quickstart": [91, 93], "refinediag": [91, 94], "regrid": [91, 94], "regrid_xi": [16, 17], "repositori": 3, "run": [2, 50, 58, 59, 60, 84, 87, 88, 91], "runfremak": 41, "script": [91, 94], "set": 2, "setup": 83, "static": [91, 94], "statu": 51, "structur": 3, "submodul": [4, 5, 7, 12, 16, 18, 20, 22, 25, 28, 32, 35, 42, 44, 52, 58, 61, 64, 76, 79], "subpackag": [4, 5, 7, 18, 25, 35, 44, 76], "surfac": [91, 94], "system": 83, "test": [12, 13, 20, 21, 28, 29, 42, 43, 52, 53, 54, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 79, 80], "test_cmor_run_subtool": 29, "test_combine_yaml": 80, "test_configure_script_yaml": 53, "test_create_makefil": 43, "test_fre_app_cli": 65, "test_fre_catalog": 21, "test_fre_catalog_cli": 66, "test_fre_check_cli": 67, "test_fre_cli": 68, "test_fre_cmor_cli": 69, "test_fre_list_cli": 70, "test_fre_make_cli": 71, "test_fre_pp_cli": 72, "test_fre_run_cli": 73, "test_fre_test_cli": 74, "test_fre_yamltools_cli": 75, "test_generate_time_averag": 13, "test_rose_quot": 54, "thing": 2, "timeaverag": 14, "timeseri": [91, 94], "tool": [3, 84], "trigger": 55, "usag": 91, "valid": [56, 84, 86], "welcom": 81, "what": 96, "workflow": 2, "wrapper": 57, "xy": [91, 94], "yaml": [84, 90, 91, 93, 95], "yamltool": [76, 77, 78, 79, 80, 84], "your": 2}})
\ No newline at end of file
diff --git a/setup.html b/setup.html
new file mode 100644
index 00000000..39e8b217
--- /dev/null
+++ b/setup.html
@@ -0,0 +1,153 @@
+
+
+
+
+
+
+
+
+ Setup — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+Setup
+fre-cli is conda-installable from the “noaa-gfdl” anaconda channel (https://anaconda.org/NOAA-GFDL/fre-cli )
+and is deployed on GFDL systems as Environment Modules.
+
+On GFDL systems
+If you are at GFDL (gaea, PP/AN, workstations), you may skip installation:
+module load fre / 2024.01
+
+fre -- help
+
+
+
+
+Generic
+If you are outside GFDL or are a FRE developer, install with conda. If you’re at GFDL, bring conda into your PATH:
+
+If you are outside GFDL, install the miniconda tool with the standard instructions (https://docs.anaconda.com/miniconda/miniconda-install/ ).
+Once you have conda available, install the latest fre-cli from the NOAA-GFDL anaconda channel:
+conda create -- name fre -- channel noaa - gfdl -- channel conda - forge fre - cli
+
+
+To install a specific version:
+conda create -- name fre - 202401 -- channel noaa - gfdl -- channel conda - forge fre - cli :: 2024.01
+
+
+and activate it:
+conda activate fre
+
+fre -- help
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tools.html b/tools.html
new file mode 100644
index 00000000..ed1b26b3
--- /dev/null
+++ b/tools.html
@@ -0,0 +1,415 @@
+
+
+
+
+
+
+
+
+ Tools — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tools/app.html b/tools/app.html
new file mode 100644
index 00000000..b25d71a1
--- /dev/null
+++ b/tools/app.html
@@ -0,0 +1,109 @@
+
+
+
+
+
+
+
+
+ <no title> — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
fre app tools are intended to be a collection of single-purpose tools.
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tools/catalog.html b/tools/catalog.html
new file mode 100644
index 00000000..1b85a54a
--- /dev/null
+++ b/tools/catalog.html
@@ -0,0 +1,124 @@
+
+
+
+
+
+
+
+
+ builder — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+builder
+
+
+
+Builds json and csv format catalogs from user input directory path
+Minimal Syntax: fre catalog builder -i [input path] -o [output path]
+Module(s) needed: n/a
+Example: fre catalog builder -i /archive/am5/am5/am5f3b1r0/c96L65_am5f3b1r0_pdclim1850F/gfdl.ncrc5-deploy-prod-openmp/pp -o ~/output --overwrite
+
+
+
+validate
+Validate the catalog
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tools/cmor.html b/tools/cmor.html
new file mode 100644
index 00000000..856e1699
--- /dev/null
+++ b/tools/cmor.html
@@ -0,0 +1,188 @@
+
+
+
+
+
+
+
+
+ run — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+run
+fre cmor run
rewrites climate model output files in a target directory in a CMIP-compliant manner
+for downstream publishing. It accepts 6 arguments, only one being optional. A brief description of each:
+
+arguments
+
+(required) -d, --indir TEXT
, input directory containing netCDF files to CMORize.
+
+all netCDF files within indir
will have their filename checked for local variables
+specified in varlist
as keys, and ISO datetime strings extracted and kept in a list
+for later iteration over target files
+a debugging-oriented boolean flag constant at the top of cmor_mixer.py
, if True
+will process one file of all files found within indir
and cease processing for that
+variable after succeeding on one file
+
+
+(required) -l, --varlist TEXT
, path to variable list dictionary.
+
+each entry in the variable list dictionary corresponds to a key/value pair
+the key (local variable) is used for ID’ing files within indir
to be processed
+associated with the key (local variable), is the value (target variable), which should
+be the label attached to the data within the targeted file(s)
+
+
+(required) -r, --table_config TEXT
, path to MIP json configuration holding variable
+metadata.
+
+
+(required) -p, --exp_config TEXT
, path to json configuration holding experiment/model
+metadata
+
+contains e.g. grid_label
, and points to other important configuration files
+associated with the MIP
+the other configuration files are typically housing metadata associated with coordinates
,
+formula_terms
, and controlled-vocabulary (CV
).
+
+
+(required) -o, --outdir TEXT
, path-prefix inwhich the output directory structure is created.
+
+further output-directories and structure/template information is specified specified in exp_config
+in addition to the output-structure/template used, an additional directory corresponding to the
+date the CMORizing was done is created near the end of the directory tree structure
+
+
+(optional) -v, --opt_var_name TEXT
, a specific variable to target for processing
+
+
+
+
+
+examples
+with a local clone of fre-cli
, the following call should work out-of-the-box from
+the root directory of the repository.
+fre cmor run \
+ - d fre / tests / test_files / ocean_sos_var_file \
+ - l fre / tests / test_files / varlist \
+ - r fre / tests / test_files / cmip6 - cmor - tables / Tables / CMIP6_Omon . json \
+ - p fre / tests / test_files / CMOR_input_example . json \
+ - o fre / tests / test_files / outdir
+
+
+
+
+background
+The bulk of this routine is housed in fre/cmor/cmor_mixer.py
, which is a rewritten version of
+Sergey Malyshev’s original CMORcommander.py
script, utilized during GFDL’s CMIP6 publishing run.
+This code is dependent on two primary json configuration files- a MIP
+variable table and another containing experiment (i.e. model) specific metdata (e.g. grid) to append
+to the output netCDF file headers, in addition to other configuration options such as output directory
+name specification, output path templates, and specification of other json configuration files containing
+controlled-vocabulary (CV), coordinate, and formula term conventions for rewriting the output metadata.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tools/make.html b/tools/make.html
new file mode 100644
index 00000000..3dfca509
--- /dev/null
+++ b/tools/make.html
@@ -0,0 +1,208 @@
+
+
+
+
+
+
+
+
+ create-checkout — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+create-checkout
+
+fre make create-checkout [options]
+
+
+
+
+create-makefile
+
+fre make create-makefile [options]
+
+
+
+
+create-compile
+
+fre make create-compile [options]
+
+
+
+
+create-dockerfile
+
+fre make create-dockerfile [options]
+Purpose: Creates the dockerfile and creates the container (with execute option)
+With the creation of the dockerfile, the Makefile, checkout script, and any other necessary script is copied into the container from a temporary location
+
+Options:
+-y, –yamlfile [experiment yaml] (required)
+-p, –platform [platform] (required)
+-t, –target [target] (required)
+
+
+
+
+
+
+
+
+
+run-fremake
+
+fre make run-fremake [options]
+Purpose: Create the checkout script, Makefile, compile script, and dockerfile (platform dependent) for the compilation of the model
+
+Options:
+-y, –yamlfile [experiment yaml] (required)
+-p, –platform [platform] (required)
+-t, –target [target] (required)
+-npc, –no-parallel-checkout (for container build)
+-j, –jobs [number of jobs to run simultneously]
+-n, –parallel [number of concurrent modile compiles]
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tools/pp.html b/tools/pp.html
new file mode 100644
index 00000000..97ae45ee
--- /dev/null
+++ b/tools/pp.html
@@ -0,0 +1,126 @@
+
+
+
+
+
+
+
+
+ configure — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+
+checkout
+
+Checkout template file and clone gitlab.gfdl.noaa.gov/fre2/workflows/postprocessing.git repository
+Minimal Syntax: fre pp checkout -e [experiment name] -p [platform name] -t [target name]
+Module(s) needed: n/a
+Example: fre pp checkout -e c96L65_am5f4b4r0_amip -p gfdl.ncrc5-deploy -t prod-openmp
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tools/yamltools.html b/tools/yamltools.html
new file mode 100644
index 00000000..8ca09781
--- /dev/null
+++ b/tools/yamltools.html
@@ -0,0 +1,114 @@
+
+
+
+
+
+
+
+
+ combine-yamls — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/usage.html b/usage.html
new file mode 100644
index 00000000..24f2be35
--- /dev/null
+++ b/usage.html
@@ -0,0 +1,690 @@
+
+
+
+
+
+
+
+
+ Usage — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+Usage
+Using a set of YAML configuration files, fre make
compiles a FMS-based model, and fre pp
postprocesses the history output and runs diagnostic analysis scripts. Please note that model running is not yet supported in FRE 2024; continue to use FRE Bronx frerun.
+
+YAML Framework
+In order to utilize FRE 2024.01 tools, a distrubuted YAML structure is required. This framework includes a main model yaml, a compile yaml, a platforms yaml, and post-processing yamls. Throughout the compilation and post-processing steps, combined yamls that will be parsed for information are created. Yamls follow a dictionary-like structure with [key]: [value]
fields.
+
+
+Model Yaml
+The model yaml defines reusable variables, shared directories, switches, post-processing settings, and paths to compile and post-processing yamls. Required fields in the model yaml include: fre_properties
, build
, shared
, and experiments
.
+
+fre_properties : Reusable variables
+
+- & variable1 "value1" ( string )
+- & variable2 "value2" ( string )
+
+
+
+build : paths to information needed for compilation
+
+build :
+ compileYaml : "path the compile yaml in relation to model yaml" ( string )
+ platformYaml : "path to platforms.yaml in relation to model yaml" ( string )
+
+
+
+shared : shared settings across experiments
+
+shared :
+ directories : & shared_directories
+ key : "value" ( string )
+
+ postprocess :
+ settings : & shared_settings
+ key : "value" ( string )
+ switches : & shared_switches
+ key : True / False ( boolean )
+
+
+
+
+experiments : list of post-processing experiments
+
+experiments :
+ - name : name of post - processing experiment ( string )
+ pp :
+ - path / to / post - processing / yaml for that experiment in relation to model yaml ( string )
+ analysis :
+ - path / to / analysis / yaml for that experiment in relation to model yaml ( string )
+
+
+
+
+
+
+Compile Yaml
+The compile yaml defines compilation information including component names, repos, branches, necessary flags, and necessary overrides. This is discussed more in the “Build FMS Model” section.
+
+
+
+Post-Processing Yaml
+The post-processing yamls include information specific to experiments, such as directories to data and other scripts used, switches, and component information. The post-processing yaml can further define more fre_properties
that may be experiment specific. If there are any repeated reusable variables, the ones set in this yaml will overwrite those set in the model yaml. This is discussed further in the “Postprocess FMS History Output” section.
+
+
+
+Build FMS model
+fre make
can compile a traditional “bare metal” executable or a containerized executable using a set of YAML configuration files.
+Through the fre-cli, fre make
can be used to create and run a checkout script, makefile, and compile a model.
+
+Fremake Canopy Supports:
+multiple target use; -t
flag to define each target (for multiple platform-target combinations)
+bare-metal build
+container creation
+parallel checkouts for bare-metal build
+parallel model builds
+one yaml format
+additional library support if needed
+
+
+
+Note: Users will not be able to create containers without access to podman. To get access, submit a helpdesk ticket.
+Required configuration files:
+
+
+Model Yaml
+Compile Yaml
+Platforms yaml
+
+
+These yamls are combined and further parsed through the fre make
tools.
+
+Compile Yaml
+To create the compile yaml, reference the compile section on an XML. Certain fields should be included under “compile”. These include experiment
, container_addlibs
, baremetal_linkerflags
, and src
.
+
+
+The experiment can be explicitly defined or can be used in conjunction with defined fre_properties
from the model yaml, as seen in the code block below
+container_addlibs
: list of strings of packages needed for the model to compile (used to create the link line in the Makefile)
+baremetal_linkerflags
: list of strings of linker flags (used to populate the link line in the Makefile
+src
: contains information about components needed for model compilation
+
+
+ compile:
+ experiment: !join [*group_version, "_compile"]
+ container_addlibs: "libraries and packages needed for linking in container" (string)
+ baremetal_linkerflags: "linker flags of libraries and packages needed" (string)
+ src:
+
+
+The src
section is used to include component information. This will include: component
, repo
, cpdefs
, branch
, paths
, otherFlags
, and makeOverrides
.
+src :
+ - component : "component name" ( string )
+ requires : [ "list of components that this component depends on" ] ( list of string )
+ repo : "url of code repository" ( string )
+ branch : "version of code to clone" ( string / list of strings )
+ paths : [ "paths in the component to compile" ] ( list of strings )
+ cppdefs : "CPPDEFS ot include in compiling componenet (string)
+ makeOverrides : "overrides openmp target for MOM6" ( 'OPENMP=""' ) ( string )
+ otherFlags : "Include flags needed to retrieve other necessary code" ( string )
+ doF90Cpp : True if the preprocessor needs to be run ( boolean )
+ additionalInstructions : additional instructions to run after checkout ( string )
+
+
+
+
+Guide
+
+Bare-metal Build:
+
+# Create checkout script
+fre make create - checkout - y [ model yaml file ] - p [ platform ] - t [ target ]
+
+# Create and run checkout script
+fre make create - checkout - y [ model yaml file ] - p [ platform ] - t [ target ] -- execute
+
+# Create Makefile
+fre make create - makefile - y [ model yaml file ] - p [ platform ] - t [ target ]
+
+# Creat the compile script
+fre make create - compile - y [ model yaml file ] - p [ platform ] - t [ target ]
+
+# Create and run the compile script
+fre make create - compile - y [ model yaml file ] - p [ platform ] - t [ target ] -- execute
+
+# Run all of fremake
+fre make run - fremake - y [ model yaml file ] - p [ platform ] - t [ target ] [ other options ... ]
+
+
+
+Container Build:
+
+For the container build, parallel checkouts are not supported, so the -npc options must be used for the checkout script. In addition the platform must be a container platform.
+Users will not be able to create containers unless they have podman access on gaea.
+# Create checkout script
+fre make create - checkout - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ] - npc
+
+# Create and run checkout script
+fre make create - checkout - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ] -- execute
+
+# Create Makefile
+fre make create - makefile - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ]
+
+# Create a Dockerfile
+fre make create - dockerfile - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ]
+
+# Create and run the Dockerfile
+fre make create - dockerfile - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ] -- execute
+
+
+
+
+Quickstart
+The quickstart instructions can be used with the am5-compile examples located in the fre-examples repository: https://github.com/NOAA-GFDL/fre-examples/tree/main/AM5/am5-compile
+
+Bare-metal Build:
+
+# Create checkout script
+fre make create - checkout - y am5 . yaml - p ncrc5 . intel23 - t prod
+
+# Create and run checkout script
+fre make create - checkout - y am5 . yaml - p ncrc5 . intel23 - t prod -- execute
+
+# Create Makefile
+fre make create - makefile - y am5 . yaml - p ncrc5 . intel23 - t prod
+
+# Create the compile script
+fre make create - compile - y am5 . yaml - p ncrc5 . intel23 - t prod
+
+# Create and run the compile script
+fre make create - compile - y am5 . yaml - p ncrc5 . intel23 - t prod -- execute
+
+
+
+Bare-metal Build Multi-target:
+
+# Create checkout script
+fre make create - checkout - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug
+
+# Create and run checkout script
+fre make create - checkout - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug -- execute
+
+# Create Makefile
+fre make create - makefile - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug
+
+# Create the compile script
+fre make create - compile - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug
+
+# Create and run the compile script
+fre make create - compile - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug -- execute
+
+
+
+Container Build:
+
+In order for the container to build successfully, a -npc , or –no-parallel-checkout is needed.
+# Create checkout script
+fre make create - checkout - y am5 . yaml - p hpcme .2023 - t prod - npc
+
+# Create and run checkout script
+fre make create - checkout - y am5 . yaml - p hpcme .2023 - t prod - npc -- execute
+
+# Create Makefile
+fre make create - makefile - y am5 . yaml - p hpcme .2023 - t prod
+
+# Create Dockerfile
+fre make create - dockerfile - y am5 . yaml - p hpcme .2023 - t prod
+
+# Create and run the Dockerfile
+fre make create - dockerfile - y am5 . yaml - p hpcme .2023 - t prod -- execute
+
+
+
+Run all of fremake:
+
+Currently, run-fremake kicks off the compilation automatically; no --execute
option needed.
+# Bare-metal
+fre make run - fremake - y am5 . yaml - p ncrc5 . intel23 - t prod
+
+# Container
+fre make run - fremake - y am5 . yaml - p hpcme .2023 - t prod - npc
+
+
+
+
+
+Run FMS model
+Check back in the latter half of 2025 or so.
+
+
+Postprocess FMS history output
+fre pp
regrids FMS history files and generates timeseries, climatologies, and static postprocessed files, with instructions specified in YAML.
+Bronx plug-in refineDiag and analysis scripts can also be used, and a reimagined analysis script ecosystem is being developed and is available now (for adventurous users). The new analysis script framework is independent of and compatible with FRE (https://github.com/NOAA-GFDL/analysis-scripts ). The goal is to combine the ease-of-use of legacy FRE analysis scripts with the standardization of model output data catalogs and python virtual environments.
+In the future, output NetCDF files will be rewritten by CMOR by default, ready for publication to community archives (e.g. ESGF). Presently, standalone CMOR tooling is available as fre cmor
.
+By default, an intake-esm-compatible data catalog is generated and updated, containing a programmatic metadata-enriched searchable interface to the postprocessed output. The catalog tooling can be independently assessed as fre catalog
.
+
+FMS history files
+FRE experiments are run in segments of simulated time. The FMS diagnostic manager, as configured in
+experiment configuration files (diag yamls) saves a set of diagnostic output files, or “history files.”
+The history files are organized by label and can contain one or more temporal or static diagnostics.
+FRE (Bronx frerun) renames and combines the raw model output (that is usually on a distributed grid),
+and saves the history files in one tarfile per segment, date-stamped with the date of the beginning of the segment.
+The FMS diagnostic manager requires
+that variables within one history file be the same temporal frequency (e.g. daily, monthly, annual),
+but statics are allowed in any history file. Usually, variables in a history file
+share a horizontal and vertical grid.
+Each history tarfile, again, is date-stamped with the date of the beginning of the segment, in YYYYMMDD format.
+For example, for a 5-year experiment with 6-month segments, there will be 10 history files containing the
+raw model output. Each history tarfile contains a segment’s worth of time (in this case 6 months).:
+19790101. nc . tar 19800101. nc . tar 19810101. nc . tar 19820101. nc . tar 19830101. nc . tar
+19790701. nc . tar 19800701. nc . tar 19810701. nc . tar 19820701. nc . tar 19830701. nc . tar
+
+
+Each history file within the history tarfiles are also similarly date-stamped. Atmosphere and land history files
+are on the native cubed-sphere grid, which have 6 tiles that represent the global domain. Ocean, ice, and
+global scalar output have just one file that covers the global domain.
+For example, if the diagnostic manager were configured to save atmospheric and ocean annual and monthly history files,
+the 19790101.nc.tar tarfile might contain:
+tar - tf 19790101. nc . tar | sort
+
+./ 19790101. atmos_month . tile1 . nc
+./ 19790101. atmos_month . tile2 . nc
+./ 19790101. atmos_month . tile3 . nc
+./ 19790101. atmos_month . tile4 . nc
+./ 19790101. atmos_month . tile5 . nc
+./ 19790101. atmos_month . tile6 . nc
+./ 19790101. atmos_annual . tile1 . nc
+./ 19790101. atmos_annual . tile2 . nc
+./ 19790101. atmos_annual . tile3 . nc
+./ 19790101. atmos_annual . tile4 . nc
+./ 19790101. atmos_annual . tile5 . nc
+./ 19790101. atmos_annual . tile6 . nc
+./ 19790101. ocean_month . nc
+./ 19790101. ocean_annual . nc
+
+
+The name of the history file, while often predictably named, are arbitrary labels within the Diagnostic Manager configuration
+(diag yamls). Each history file is a CF-standard NetCDF file that can be inspected with common NetCDF tools such as the NCO or CDO tools, or even ncdump
.
+Required configuration
+
+Set the history directory in your postprocessing yaml:
+
+
+
+directories: history: /arch5/am5/am5/am5f7c1r0/c96L65_am5f7c1r0_amip/gfdl.ncrc5-deploy-prod-openmp/history
+
+
+
+
+Set the segment size as an ISO8601 duration (e.g. P1Y is “one year”):
+
+
+
+postprocess:
+settings: history_segment: P1Y
+
+
+
+
+
+
+Set the date range to postprocess as ISO8601 dates:
+
+
+
+postprocess:
+settings: pp_start: 1979-01-01T0000Z
+pp_stop: 2020-01-01T0000Z
+
+
+
+
+
+
+
+Postprocess components
+The history-file namespace is a single layer as shown above. By longtime tradition, FRE postprocessing namespaces are richer, with
+a distinction for timeseries, timeaveraged, and static output datasets, and includes frequency and chunk-size in the directory structure.
+Postprocessed files within a “component” share a horizontal grid; which can be the native grid or regridded to lat/lon.
+Required configuration
+
+Define the atmos and ocean postprocess components:
+
+
+
+postprocess:
+components:
+type: atmos
+sources: [atmos_month, atmos_annual]
+
+type: ocean
+sources: [ocean_month, ocean_annual]
+
+
+
+
+
+
+
+
+
+XY-regridding
+Commonly, native grid history files are regridded during postprocessing. To regrid to a lat/lon grid, configure your
+desired output grid, interpolation method, input grid type, and path to your FMS exchange grid definition.
+Optional configuration (i.e. if xy-regridding is desired)
+
+Regrid the atmos and ocean components to a 1x1 degree grid:
+
+
+
+directories: pp_grid_spec: /archive/oar.gfdl.am5/model_gen5/inputs/c96_grid/c96_OM4_025_grid_No_mg_drag_v20160808.tar
+
+postprocess:
+components:
+type: atmos
+sources: [atmos_month, atmos_annual]
+sourceGrid: cubedsphere
+inputRealm: atmos
+xyInterp: [180, 360]
+interpMethod: conserve_order2
+
+type: ocean
+sources: [ocean_month, ocean_annual]
+sourceGrid: tripolar
+inputRealm: ocean
+xyInterp: [180, 360]
+interpMethod: conserve_order1
+
+
+
+
+
+
+
+
+
+Timeseries
+Timeseries output is the most common type of postprocessed output.
+
+
+Climatologies
+annual and monthly climatologies
+less fine-grained than bronx
+per-component switch coming
+now it’s one switch for entire pp
+
+
+Statics
+underbaked, known deficiency
+currently, takes statics from “source” history files
+
+
+
+Surface masking for FMS pressure-level history
+
+
+Legacy refineDiag scripts
+
+
+
+CMORize postprocessed output
+Brief rundown of commands also provided below:
+
+Enter commands and follow --help
messages for guidance
+If the user just runs fre
, it will list all the command groups following fre
, such as
+run
, make
, pp
, etc. and once the user specifies a command group, the list of available
+subcommands for that group will be shown
+Commands that require arguments to run will alert user about missing arguments, and will also list
+the rest of the optional parameters if --help
is executed
+Argument flags are not positional, can be specified in any order as long as they are specified
+Can run directly from any directory, no need to clone repository
+May need to deactivate environment and reactivate it in order for changes to apply
+fre/setup.py
allows fre/fre.py
to be ran as fre
on the command line by defining it as an
+entry point . Without it, the call would be instead, something like python fre/fre.py
+See also, fre cmor
’s README
+See also, fre cmor
’s project board
+
+This set of tools leverages the external cmor
python package within the fre
ecosystem. cmor
is an
+acronym for “climate model output rewriter”. The process of rewriting model-specific output files for model
+intercomparisons (MIPs) using the cmor
module is, quite cleverly, referred to as “CMORizing”.
+
+
+Generate data catalogs
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/usage/cmor.html b/usage/cmor.html
new file mode 100644
index 00000000..b122ef06
--- /dev/null
+++ b/usage/cmor.html
@@ -0,0 +1,127 @@
+
+
+
+
+
+
+
+
+ <no title> — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
Brief rundown of commands also provided below:
+
+Enter commands and follow --help
messages for guidance
+If the user just runs fre
, it will list all the command groups following fre
, such as
+run
, make
, pp
, etc. and once the user specifies a command group, the list of available
+subcommands for that group will be shown
+Commands that require arguments to run will alert user about missing arguments, and will also list
+the rest of the optional parameters if --help
is executed
+Argument flags are not positional, can be specified in any order as long as they are specified
+Can run directly from any directory, no need to clone repository
+May need to deactivate environment and reactivate it in order for changes to apply
+fre/setup.py
allows fre/fre.py
to be ran as fre
on the command line by defining it as an
+entry point . Without it, the call would be instead, something like python fre/fre.py
+See also, fre cmor
’s README
+See also, fre cmor
’s project board
+
+
This set of tools leverages the external cmor
python package within the fre
ecosystem. cmor
is an
+acronym for “climate model output rewriter”. The process of rewriting model-specific output files for model
+intercomparisons (MIPs) using the cmor
module is, quite cleverly, referred to as “CMORizing”.
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/usage/compile.html b/usage/compile.html
new file mode 100644
index 00000000..c37bb724
--- /dev/null
+++ b/usage/compile.html
@@ -0,0 +1,284 @@
+
+
+
+
+
+
+
+
+ Compile Yaml — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
fre make
can compile a traditional “bare metal” executable or a containerized executable using a set of YAML configuration files.
+
Through the fre-cli, fre make
can be used to create and run a checkout script, makefile, and compile a model.
+
+Fremake Canopy Supports:
+multiple target use; -t
flag to define each target (for multiple platform-target combinations)
+bare-metal build
+container creation
+parallel checkouts for bare-metal build
+parallel model builds
+one yaml format
+additional library support if needed
+
+
+
+
Note: Users will not be able to create containers without access to podman. To get access, submit a helpdesk ticket.
+
Required configuration files:
+
+
+Model Yaml
+Compile Yaml
+Platforms yaml
+
+
+
These yamls are combined and further parsed through the fre make
tools.
+
+Compile Yaml
+To create the compile yaml, reference the compile section on an XML. Certain fields should be included under “compile”. These include experiment
, container_addlibs
, baremetal_linkerflags
, and src
.
+
+
+The experiment can be explicitly defined or can be used in conjunction with defined fre_properties
from the model yaml, as seen in the code block below
+container_addlibs
: list of strings of packages needed for the model to compile (used to create the link line in the Makefile)
+baremetal_linkerflags
: list of strings of linker flags (used to populate the link line in the Makefile
+src
: contains information about components needed for model compilation
+
+
+ compile:
+ experiment: !join [*group_version, "_compile"]
+ container_addlibs: "libraries and packages needed for linking in container" (string)
+ baremetal_linkerflags: "linker flags of libraries and packages needed" (string)
+ src:
+
+
+The src
section is used to include component information. This will include: component
, repo
, cpdefs
, branch
, paths
, otherFlags
, and makeOverrides
.
+src :
+ - component : "component name" ( string )
+ requires : [ "list of components that this component depends on" ] ( list of string )
+ repo : "url of code repository" ( string )
+ branch : "version of code to clone" ( string / list of strings )
+ paths : [ "paths in the component to compile" ] ( list of strings )
+ cppdefs : "CPPDEFS ot include in compiling componenet (string)
+ makeOverrides : "overrides openmp target for MOM6" ( 'OPENMP=""' ) ( string )
+ otherFlags : "Include flags needed to retrieve other necessary code" ( string )
+ doF90Cpp : True if the preprocessor needs to be run ( boolean )
+ additionalInstructions : additional instructions to run after checkout ( string )
+
+
+
+
+Guide
+
+Bare-metal Build:
+
+# Create checkout script
+fre make create - checkout - y [ model yaml file ] - p [ platform ] - t [ target ]
+
+# Create and run checkout script
+fre make create - checkout - y [ model yaml file ] - p [ platform ] - t [ target ] -- execute
+
+# Create Makefile
+fre make create - makefile - y [ model yaml file ] - p [ platform ] - t [ target ]
+
+# Creat the compile script
+fre make create - compile - y [ model yaml file ] - p [ platform ] - t [ target ]
+
+# Create and run the compile script
+fre make create - compile - y [ model yaml file ] - p [ platform ] - t [ target ] -- execute
+
+# Run all of fremake
+fre make run - fremake - y [ model yaml file ] - p [ platform ] - t [ target ] [ other options ... ]
+
+
+
+Container Build:
+
+For the container build, parallel checkouts are not supported, so the -npc options must be used for the checkout script. In addition the platform must be a container platform.
+Users will not be able to create containers unless they have podman access on gaea.
+# Create checkout script
+fre make create - checkout - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ] - npc
+
+# Create and run checkout script
+fre make create - checkout - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ] -- execute
+
+# Create Makefile
+fre make create - makefile - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ]
+
+# Create a Dockerfile
+fre make create - dockerfile - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ]
+
+# Create and run the Dockerfile
+fre make create - dockerfile - y [ model yaml file ] - p [ CONTAINER PLATFORM ] - t [ target ] -- execute
+
+
+
+
+Quickstart
+The quickstart instructions can be used with the am5-compile examples located in the fre-examples repository: https://github.com/NOAA-GFDL/fre-examples/tree/main/AM5/am5-compile
+
+Bare-metal Build:
+
+# Create checkout script
+fre make create - checkout - y am5 . yaml - p ncrc5 . intel23 - t prod
+
+# Create and run checkout script
+fre make create - checkout - y am5 . yaml - p ncrc5 . intel23 - t prod -- execute
+
+# Create Makefile
+fre make create - makefile - y am5 . yaml - p ncrc5 . intel23 - t prod
+
+# Create the compile script
+fre make create - compile - y am5 . yaml - p ncrc5 . intel23 - t prod
+
+# Create and run the compile script
+fre make create - compile - y am5 . yaml - p ncrc5 . intel23 - t prod -- execute
+
+
+
+Bare-metal Build Multi-target:
+
+# Create checkout script
+fre make create - checkout - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug
+
+# Create and run checkout script
+fre make create - checkout - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug -- execute
+
+# Create Makefile
+fre make create - makefile - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug
+
+# Create the compile script
+fre make create - compile - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug
+
+# Create and run the compile script
+fre make create - compile - y am5 . yaml - p ncrc5 . intel23 - t prod - t debug -- execute
+
+
+
+Container Build:
+
+In order for the container to build successfully, a -npc , or –no-parallel-checkout is needed.
+# Create checkout script
+fre make create - checkout - y am5 . yaml - p hpcme .2023 - t prod - npc
+
+# Create and run checkout script
+fre make create - checkout - y am5 . yaml - p hpcme .2023 - t prod - npc -- execute
+
+# Create Makefile
+fre make create - makefile - y am5 . yaml - p hpcme .2023 - t prod
+
+# Create Dockerfile
+fre make create - dockerfile - y am5 . yaml - p hpcme .2023 - t prod
+
+# Create and run the Dockerfile
+fre make create - dockerfile - y am5 . yaml - p hpcme .2023 - t prod -- execute
+
+
+
+Run all of fremake:
+
+Currently, run-fremake kicks off the compilation automatically; no --execute
option needed.
+# Bare-metal
+fre make run - fremake - y am5 . yaml - p ncrc5 . intel23 - t prod
+
+# Container
+fre make run - fremake - y am5 . yaml - p hpcme .2023 - t prod - npc
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/usage/postprocess.html b/usage/postprocess.html
new file mode 100644
index 00000000..a9567c67
--- /dev/null
+++ b/usage/postprocess.html
@@ -0,0 +1,277 @@
+
+
+
+
+
+
+
+
+ FMS history files — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
fre pp
regrids FMS history files and generates timeseries, climatologies, and static postprocessed files, with instructions specified in YAML.
+
Bronx plug-in refineDiag and analysis scripts can also be used, and a reimagined analysis script ecosystem is being developed and is available now (for adventurous users). The new analysis script framework is independent of and compatible with FRE (https://github.com/NOAA-GFDL/analysis-scripts ). The goal is to combine the ease-of-use of legacy FRE analysis scripts with the standardization of model output data catalogs and python virtual environments.
+
In the future, output NetCDF files will be rewritten by CMOR by default, ready for publication to community archives (e.g. ESGF). Presently, standalone CMOR tooling is available as fre cmor
.
+
By default, an intake-esm-compatible data catalog is generated and updated, containing a programmatic metadata-enriched searchable interface to the postprocessed output. The catalog tooling can be independently assessed as fre catalog
.
+
+FMS history files
+FRE experiments are run in segments of simulated time. The FMS diagnostic manager, as configured in
+experiment configuration files (diag yamls) saves a set of diagnostic output files, or “history files.”
+The history files are organized by label and can contain one or more temporal or static diagnostics.
+FRE (Bronx frerun) renames and combines the raw model output (that is usually on a distributed grid),
+and saves the history files in one tarfile per segment, date-stamped with the date of the beginning of the segment.
+The FMS diagnostic manager requires
+that variables within one history file be the same temporal frequency (e.g. daily, monthly, annual),
+but statics are allowed in any history file. Usually, variables in a history file
+share a horizontal and vertical grid.
+Each history tarfile, again, is date-stamped with the date of the beginning of the segment, in YYYYMMDD format.
+For example, for a 5-year experiment with 6-month segments, there will be 10 history files containing the
+raw model output. Each history tarfile contains a segment’s worth of time (in this case 6 months).:
+19790101. nc . tar 19800101. nc . tar 19810101. nc . tar 19820101. nc . tar 19830101. nc . tar
+19790701. nc . tar 19800701. nc . tar 19810701. nc . tar 19820701. nc . tar 19830701. nc . tar
+
+
+Each history file within the history tarfiles are also similarly date-stamped. Atmosphere and land history files
+are on the native cubed-sphere grid, which have 6 tiles that represent the global domain. Ocean, ice, and
+global scalar output have just one file that covers the global domain.
+For example, if the diagnostic manager were configured to save atmospheric and ocean annual and monthly history files,
+the 19790101.nc.tar tarfile might contain:
+tar - tf 19790101. nc . tar | sort
+
+./ 19790101. atmos_month . tile1 . nc
+./ 19790101. atmos_month . tile2 . nc
+./ 19790101. atmos_month . tile3 . nc
+./ 19790101. atmos_month . tile4 . nc
+./ 19790101. atmos_month . tile5 . nc
+./ 19790101. atmos_month . tile6 . nc
+./ 19790101. atmos_annual . tile1 . nc
+./ 19790101. atmos_annual . tile2 . nc
+./ 19790101. atmos_annual . tile3 . nc
+./ 19790101. atmos_annual . tile4 . nc
+./ 19790101. atmos_annual . tile5 . nc
+./ 19790101. atmos_annual . tile6 . nc
+./ 19790101. ocean_month . nc
+./ 19790101. ocean_annual . nc
+
+
+The name of the history file, while often predictably named, are arbitrary labels within the Diagnostic Manager configuration
+(diag yamls). Each history file is a CF-standard NetCDF file that can be inspected with common NetCDF tools such as the NCO or CDO tools, or even ncdump
.
+Required configuration
+
+Set the history directory in your postprocessing yaml:
+
+
+
+directories: history: /arch5/am5/am5/am5f7c1r0/c96L65_am5f7c1r0_amip/gfdl.ncrc5-deploy-prod-openmp/history
+
+
+
+
+Set the segment size as an ISO8601 duration (e.g. P1Y is “one year”):
+
+
+
+postprocess:
+settings: history_segment: P1Y
+
+
+
+
+
+
+Set the date range to postprocess as ISO8601 dates:
+
+
+
+postprocess:
+settings: pp_start: 1979-01-01T0000Z
+pp_stop: 2020-01-01T0000Z
+
+
+
+
+
+
+
+Postprocess components
+The history-file namespace is a single layer as shown above. By longtime tradition, FRE postprocessing namespaces are richer, with
+a distinction for timeseries, timeaveraged, and static output datasets, and includes frequency and chunk-size in the directory structure.
+Postprocessed files within a “component” share a horizontal grid; which can be the native grid or regridded to lat/lon.
+Required configuration
+
+Define the atmos and ocean postprocess components:
+
+
+
+postprocess:
+components:
+type: atmos
+sources: [atmos_month, atmos_annual]
+
+type: ocean
+sources: [ocean_month, ocean_annual]
+
+
+
+
+
+
+
+
+
+XY-regridding
+Commonly, native grid history files are regridded during postprocessing. To regrid to a lat/lon grid, configure your
+desired output grid, interpolation method, input grid type, and path to your FMS exchange grid definition.
+Optional configuration (i.e. if xy-regridding is desired)
+
+Regrid the atmos and ocean components to a 1x1 degree grid:
+
+
+
+directories: pp_grid_spec: /archive/oar.gfdl.am5/model_gen5/inputs/c96_grid/c96_OM4_025_grid_No_mg_drag_v20160808.tar
+
+postprocess:
+components:
+type: atmos
+sources: [atmos_month, atmos_annual]
+sourceGrid: cubedsphere
+inputRealm: atmos
+xyInterp: [180, 360]
+interpMethod: conserve_order2
+
+type: ocean
+sources: [ocean_month, ocean_annual]
+sourceGrid: tripolar
+inputRealm: ocean
+xyInterp: [180, 360]
+interpMethod: conserve_order1
+
+
+
+
+
+
+
+
+
+Timeseries
+Timeseries output is the most common type of postprocessed output.
+
+
+Climatologies
+annual and monthly climatologies
+less fine-grained than bronx
+per-component switch coming
+now it’s one switch for entire pp
+
+
+Statics
+underbaked, known deficiency
+currently, takes statics from “source” history files
+
+
+
+Surface masking for FMS pressure-level history
+
+
+Legacy refineDiag scripts
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/usage/yaml_framework.html b/usage/yaml_framework.html
new file mode 100644
index 00000000..0072e8d2
--- /dev/null
+++ b/usage/yaml_framework.html
@@ -0,0 +1,267 @@
+
+
+
+
+
+
+
+
+ Yaml Formatting — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
In order to utilize FRE 2024.01 tools, a distrubuted YAML structure is required. This framework includes a main model yaml, a compile yaml, a platforms yaml, and post-processing yamls. Throughout the compilation and post-processing steps, combined yamls that will be parsed for information are created. Yamls follow a dictionary-like structure with [key]: [value]
fields.
+
+
+Model Yaml
+The model yaml defines reusable variables, shared directories, switches, post-processing settings, and paths to compile and post-processing yamls. Required fields in the model yaml include: fre_properties
, build
, shared
, and experiments
.
+
+fre_properties : Reusable variables
+
+- & variable1 "value1" ( string )
+- & variable2 "value2" ( string )
+
+
+
+build : paths to information needed for compilation
+
+build :
+ compileYaml : "path the compile yaml in relation to model yaml" ( string )
+ platformYaml : "path to platforms.yaml in relation to model yaml" ( string )
+
+
+
+shared : shared settings across experiments
+
+shared :
+ directories : & shared_directories
+ key : "value" ( string )
+
+ postprocess :
+ settings : & shared_settings
+ key : "value" ( string )
+ switches : & shared_switches
+ key : True / False ( boolean )
+
+
+
+
+experiments : list of post-processing experiments
+
+experiments :
+ - name : name of post - processing experiment ( string )
+ pp :
+ - path / to / post - processing / yaml for that experiment in relation to model yaml ( string )
+ analysis :
+ - path / to / analysis / yaml for that experiment in relation to model yaml ( string )
+
+
+
+
+
+
+Compile Yaml
+The compile yaml defines compilation information including component names, repos, branches, necessary flags, and necessary overrides. This is discussed more in the “Build FMS Model” section.
+
+
+
+Post-Processing Yaml
+The post-processing yamls include information specific to experiments, such as directories to data and other scripts used, switches, and component information. The post-processing yaml can further define more fre_properties
that may be experiment specific. If there are any repeated reusable variables, the ones set in this yaml will overwrite those set in the model yaml. This is discussed further in the “Postprocess FMS History Output” section.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/what-is-fre.html b/what-is-fre.html
new file mode 100644
index 00000000..8972e108
--- /dev/null
+++ b/what-is-fre.html
@@ -0,0 +1,121 @@
+
+
+
+
+
+
+
+
+ What is FRE? — Fre-Cli 1.0 documentation
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Fre-Cli
+
+
+
+
+
+
+
+
+
+What is FRE?
+FRE, the FMS Runtime Environment, is the companion runtime workflow for FMS-based climate and earth system models, and contains scripts and batch job handlers to compile models, run experiments, and postprocess and analyze the output. Developed around 2004 by GFDL’s Modeling System Division, FRE was developed primarily in one repository (“fre-commands”, https://github.com/NOAA-GFDL/FRE ), used subtools in another repository (FRE-NCtools, https://github.com/NOAA-GFDL/fre-nctools ), and was deployed using a set of Environment Modules (https://gitlab.gfdl.noaa.gov/fre/modulefiles ). Originally, the major releases of FRE were rivers (Arkansas, Bronx) and the minor releases were numbers. In practice, though, the “Bronx” release name was retained and the number has been incremented over the years. e.g. Bronx-23 is the latest release.
+Over the last couple years, MSD’s workflow team has reengineered the compiling and postprocessing parts of FRE, in a modern python and Cylc-based ecosystem (running experiments is not yet possible with this new FRE; stay tuned). Following a semantic versioning adopted in other FMS repositories, the reengineered FRE is versioned with a year and incrementing two-digit number. e.g. the first release of 2024 is 2024.01, the second 2024.02, and the first release next year will be 2025.01. (Optional minor releases are also available in the scheme; e.g. 2024.01.01 would be the first minor/patch release after 2024.01.) This version is used as tags in FRE repositories and in the corresponding conda (and in the future, container) release, and can be retrieved from fre --version
.
+fre-cli (this repository) can be considered a successor to the FRE Bronx “fre-commands” repository, which primarily contains user-facing tools and subtools. fre-workflows (https://github.com/NOAA-GFDL/fre-workflows ) is a companion repository containing workflow definitions that can be run by the Cylc workflow engine. It contains workflow-specific elements previously in FRE Bronx, and allows flexibility to support multiple and more complex workflows. The two new FRE repositories are versioned with the same approach, and updates will be released together for some time to ensure compatibility.
+The “cli” in fre-cli derives from the shell “fre SUBCOMMAND COMMAND” structure inspired by git, cylc, and other modern Linux command-line tools. This enables discovery of the tooling capability, useful for complex tools with multiple options. e.g. fre --help
, fre make --help
, fre pp --help
.
+Underneath, fre-cli is python, and the workflows and tooling can be run through a Jupyter notebook, or through other python scripts. fre-cli is conda-installable from the “noaa-gfdl” channel (conda install --channel noaa-gfdl fre-cli
).
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file