From ba20862ee89ad2403abcec462e1da3bd5a7edf16 Mon Sep 17 00:00:00 2001 From: Jonas Juselius Date: Fri, 25 Feb 2022 15:55:51 +0100 Subject: [PATCH 1/6] Enable read-only NetCDF DAP uri:s --- .../Providers/NetCDF/NetCDFDataSet.cs | 56 +++++++++++-------- 1 file changed, 32 insertions(+), 24 deletions(-) diff --git a/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs b/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs index 9ce5eb2..71e1c7a 100644 --- a/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs +++ b/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs @@ -24,7 +24,7 @@ namespace Microsoft.Research.Science.Data.NetCDF4 /// /// /// The provider supports variables of any non-negative rank. - /// + /// /// /// The provider is associated with the provider name "nc" and extensions ".nc". /// @@ -44,7 +44,7 @@ namespace Microsoft.Research.Science.Data.NetCDF4 /// /// openMode=createNew|create|open|openOrCreate|readOnly /// - /// The flag "openMode" specifies how the data set should open a file, + /// The flag "openMode" specifies how the data set should open a file, /// data base or whatever resource it uses to store the data. /// Possible values for the flag are: /// @@ -106,13 +106,13 @@ namespace Microsoft.Research.Science.Data.NetCDF4 /// include /// Allows including variables as references from another , defined as a URI, /// into this . - /// Example: msds:memory?include=msds%3Acsv%3Ffile%example.csv%23lat%2Clon + /// Example: msds:memory?include=msds%3Acsv%3Ffile%example.csv%23lat%2Clon /// (escaped version of "msds:memory?include=escape[msds:csv?file=example.csv#lat,lon]") /// includes variables lat and lon from msds:csv?file=example.csv. If variables names are not specified, /// all variables are included. /// /// - /// + /// /// /// The provider supports URIs containing a path /// and appended through '?' parameters: c:\data\air0.nc?openMode=open&enableRollback=false. @@ -132,17 +132,17 @@ namespace Microsoft.Research.Science.Data.NetCDF4 /// To enable it, implements the interface. /// /// - /// Names. If a variable name is not complied with NetCDF naming rules, + /// Names. If a variable name is not complied with NetCDF naming rules, /// its simplified version is used at the NetCDF layer, but through the it is still visible /// unmodified. /// /// - /// support. - /// Since the unmanaged NetCDF4 doesn't support a date/time type, it is internally stored - /// as a double variable with special attribute + /// support. + /// Since the unmanaged NetCDF4 doesn't support a date/time type, it is internally stored + /// as a double variable with special attribute /// Units = "100-nanosecond intervals that have elapsed since 12:00:00 midnight, January 1, 0001". /// NetCDF variables with this attribute are loaded as variables. - /// In the current release, any other units descriptions are not considered as a date/time and + /// In the current release, any other units descriptions are not considered as a date/time and /// hence the variable's final type of data is . /// /// @@ -201,8 +201,8 @@ public class NetCDFDataSet : DataSet, IChunkSizesAdjustable /// DataSet URI (see remarks for ). /// /// - /// If the file specified by exists, the NetCDF will be - /// initialized with that file. Otherwise, the new file will be created and + /// If the file specified by exists, the NetCDF will be + /// initialized with that file. Otherwise, the new file will be created and /// the resulting NetCDFDataSet will be empty. /// /// @@ -235,7 +235,7 @@ public NetCDFDataSet(string uri) /// Initializes new instance of the NetCDFDataSet class. /// /// DataSet URI (see remarks for ). - /// The open mode (see + /// The open mode (see /// ). /// /// @@ -256,7 +256,6 @@ public NetCDFDataSet(string uri, ResourceOpenMode openMode) rollbackEnabled = ((NetCDFUri)this.uri).EnableRollback; deflate = ((NetCDFUri)this.uri).Deflate; trimTrailingZero = ((NetCDFUri)this.uri).TrimTrailingZero; - InitializeFromFile(((NetCDFUri)this.uri).FileName, openMode); } @@ -269,7 +268,16 @@ private void InitializeFromFile(string fileName, ResourceOpenMode openMode) initializing = true; int res; - bool exists = File.Exists(fileName); + bool exists = true; + bool isUrl = false; + + if (((NetCDFUri)this.uri).FileName.StartsWith("http")) { + openMode = ResourceOpenMode.ReadOnly; + isUrl = true; + } else { + exists = File.Exists(fileName); + } + if (openMode == ResourceOpenMode.Create && exists) { File.Delete(fileName); @@ -293,7 +301,7 @@ private void InitializeFromFile(string fileName, ResourceOpenMode openMode) if (openMode == ResourceOpenMode.CreateNew) throw new IOException("The open mode is createNew but the file already exists"); - if (((File.GetAttributes(fileName) & FileAttributes.ReadOnly) != 0) && openMode != ResourceOpenMode.ReadOnly) + if (!isUrl && ((File.GetAttributes(fileName) & FileAttributes.ReadOnly) != 0) && openMode != ResourceOpenMode.ReadOnly) { openMode = ResourceOpenMode.ReadOnly; Trace.WriteLineIf(TraceNetCDFDataSet.TraceWarning, "NetCDFDataSet: Opening file in read only mode"); @@ -505,7 +513,7 @@ private static bool ContainsRecent(ReadOnlyVariableCollection vars, string var) } /// - /// + /// /// /// /// @@ -522,7 +530,7 @@ internal static NetCDFDataSet Open(string fileName, bool enableRollback) } /// - /// + /// /// /// /// @@ -567,7 +575,7 @@ internal bool RollbackEnabled #region Overriden Methods /// - /// + /// /// /// /// @@ -581,7 +589,7 @@ protected override Variable CreateVariable(string varName, s } /// - /// + /// /// protected override void OnTransactionOpened() { @@ -614,7 +622,7 @@ private void UpdateDimIds() } /// - /// + /// /// /// /// @@ -626,7 +634,7 @@ protected override bool OnPrecommitting(DataSet.Changes changes) return true; } /// - /// + /// /// /// protected override void OnPrecommit(DataSet.Changes changes) @@ -657,7 +665,7 @@ protected override void OnPrecommit(DataSet.Changes changes) } } /// - /// + /// /// protected override void OnCommit() { @@ -673,7 +681,7 @@ protected override void OnCommit() tempFileName = null; } /// - /// + /// /// protected override void OnRollback() { @@ -719,7 +727,7 @@ protected override void OnRollback() #region Utils /// - /// + /// /// protected override void Dispose(bool disposing) { From 4887db84d599599ed83f18025ca045ecc607df32 Mon Sep 17 00:00:00 2001 From: Jonas Juselius Date: Mon, 28 Feb 2022 14:55:51 +0100 Subject: [PATCH 2/6] Add Url property for DAP access --- .../Providers/NetCDF/NetCDFUri.cs | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/ScientificDataSet/Providers/NetCDF/NetCDFUri.cs b/ScientificDataSet/Providers/NetCDF/NetCDFUri.cs index 44f065b..8aa834f 100644 --- a/ScientificDataSet/Providers/NetCDF/NetCDFUri.cs +++ b/ScientificDataSet/Providers/NetCDF/NetCDFUri.cs @@ -22,6 +22,8 @@ public NetCDFUri(string uri) { if (GetParameterOccurences("file") > 1) throw new ArgumentException("The given uri must define single file path"); + if (GetParameterOccurences("url") > 1) + throw new ArgumentException("The given uri must define single DAP url"); } /// /// Instantiates an instance of the class with default parameters. @@ -98,6 +100,22 @@ public string FileName set { this["file"] = value; } } + /// + /// Specifies the file to open or create. + /// + [FileNameProperty] + [Description("Specifies the file to open or create.")] + public string Url + { + get + { + if (GetParameterOccurences("url") == 0) + return ""; + return this["url"]; + } + set { this["url"] = value; } + } + /// /// Indicates whether the rollback is enabled or not. /// From a7ded0752d953bb45eca49756c0748db9caa6d4a Mon Sep 17 00:00:00 2001 From: Jonas Juselius Date: Mon, 28 Feb 2022 14:56:41 +0100 Subject: [PATCH 3/6] Enable DAP for Url type Uris, and refactor --- .../Providers/NetCDF/NetCDFDataSet.cs | 413 ++++++++++-------- 1 file changed, 229 insertions(+), 184 deletions(-) diff --git a/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs b/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs index 9ce5eb2..a6ae6d1 100644 --- a/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs +++ b/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs @@ -24,7 +24,7 @@ namespace Microsoft.Research.Science.Data.NetCDF4 /// /// /// The provider supports variables of any non-negative rank. - /// + /// /// /// The provider is associated with the provider name "nc" and extensions ".nc". /// @@ -44,7 +44,7 @@ namespace Microsoft.Research.Science.Data.NetCDF4 /// /// openMode=createNew|create|open|openOrCreate|readOnly /// - /// The flag "openMode" specifies how the data set should open a file, + /// The flag "openMode" specifies how the data set should open a file, /// data base or whatever resource it uses to store the data. /// Possible values for the flag are: /// @@ -106,13 +106,13 @@ namespace Microsoft.Research.Science.Data.NetCDF4 /// include /// Allows including variables as references from another , defined as a URI, /// into this . - /// Example: msds:memory?include=msds%3Acsv%3Ffile%example.csv%23lat%2Clon + /// Example: msds:memory?include=msds%3Acsv%3Ffile%example.csv%23lat%2Clon /// (escaped version of "msds:memory?include=escape[msds:csv?file=example.csv#lat,lon]") /// includes variables lat and lon from msds:csv?file=example.csv. If variables names are not specified, /// all variables are included. /// /// - /// + /// /// /// The provider supports URIs containing a path /// and appended through '?' parameters: c:\data\air0.nc?openMode=open&enableRollback=false. @@ -132,17 +132,17 @@ namespace Microsoft.Research.Science.Data.NetCDF4 /// To enable it, implements the interface. /// /// - /// Names. If a variable name is not complied with NetCDF naming rules, + /// Names. If a variable name is not complied with NetCDF naming rules, /// its simplified version is used at the NetCDF layer, but through the it is still visible /// unmodified. /// /// - /// support. - /// Since the unmanaged NetCDF4 doesn't support a date/time type, it is internally stored - /// as a double variable with special attribute + /// support. + /// Since the unmanaged NetCDF4 doesn't support a date/time type, it is internally stored + /// as a double variable with special attribute /// Units = "100-nanosecond intervals that have elapsed since 12:00:00 midnight, January 1, 0001". /// NetCDF variables with this attribute are loaded as variables. - /// In the current release, any other units descriptions are not considered as a date/time and + /// In the current release, any other units descriptions are not considered as a date/time and /// hence the variable's final type of data is . /// /// @@ -201,8 +201,8 @@ public class NetCDFDataSet : DataSet, IChunkSizesAdjustable /// DataSet URI (see remarks for ). /// /// - /// If the file specified by exists, the NetCDF will be - /// initialized with that file. Otherwise, the new file will be created and + /// If the file specified by exists, the NetCDF will be + /// initialized with that file. Otherwise, the new file will be created and /// the resulting NetCDFDataSet will be empty. /// /// @@ -227,15 +227,19 @@ public NetCDFDataSet(string uri) Debug.WriteLineIf(TraceNetCDFDataSet.TraceInfo, "Deflate mode: " + deflate); - ResourceOpenMode openMode = this.uri.GetOpenModeOrDefault(ResourceOpenMode.OpenOrCreate); - InitializeFromFile(((NetCDFUri)this.uri).FileName, openMode); + if (((NetCDFUri)this.uri).Url.Length > 0) { + InitializeFromUrl(((NetCDFUri)this.uri).Url); + } else { + ResourceOpenMode openMode = this.uri.GetOpenModeOrDefault(ResourceOpenMode.OpenOrCreate); + InitializeFromFile(((NetCDFUri)this.uri).FileName, openMode); + } } /// /// Initializes new instance of the NetCDFDataSet class. /// /// DataSet URI (see remarks for ). - /// The open mode (see + /// The open mode (see /// ). /// /// @@ -257,7 +261,205 @@ public NetCDFDataSet(string uri, ResourceOpenMode openMode) deflate = ((NetCDFUri)this.uri).Deflate; trimTrailingZero = ((NetCDFUri)this.uri).TrimTrailingZero; - InitializeFromFile(((NetCDFUri)this.uri).FileName, openMode); + if (((NetCDFUri)this.uri).Url.Length > 0) { + InitializeFromUrl(((NetCDFUri)this.uri).Url); + } else { + InitializeFromFile(((NetCDFUri)this.uri).FileName, openMode); + } + } + + private void LoadMetadata(string fileName) + { + int res; + //******************************************************** + // Loading schema + int ndims, nvars, ngatts, unlimdimid; + res = NetCDF.nc_inq(ncid, out ndims, out nvars, out ngatts, out unlimdimid); + HandleResult(res); + + Debug.WriteLineIf(NetCDFDataSet.TraceNetCDFDataSet.TraceInfo, String.Format("NetCDF dataset {0} opened: {1} dimensions, {2} variables, {3} global attributes.", + fileName, ndims, nvars, ngatts)); + + /******************************************************** + // Reading Global Attributes + int nattrs; + res = NetCDF.nc_inq_varnatts(ncid, NetCDF.NC_GLOBAL, out nattrs); + AttributeTypeMap atm = new AttributeTypeMap(ncid, NetCDF.NC_GLOBAL); + NetCDFDataSet.HandleResult(res); + MetadataDictionary globalMetadata = Metadata; + for (int i = 0; i < nattrs; i++) + { + // Name + StringBuilder attName = new StringBuilder(512); + res = NetCDF.nc_inq_attname(ncid, NetCDF.NC_GLOBAL, i, attName); + NetCDFDataSet.HandleResult(res); + string aname = attName.ToString(); + + // Skip out internal attribute + if (aname == AttributeTypeMap.AttributeName) + continue; + + // Type + object value = ReadNetCdfAttribute(NetCDF.NC_GLOBAL, aname, atm); + globalMetadata[aname] = value; + } /**/ + + //******************************************************** + // Loading variables + int[] varIds = new int[nvars]; + res = NetCDF.nc_inq_varids(ncid, out nvars, varIds); + HandleResult(res); + + for (int i = 0; i < nvars; i++) + { + Variable var = ReadNetCdfVariable(varIds[i]); + AddVariableToCollection(var); + } + + Variable globalMetaVar = new NetCDFGlobalMetadataVariable(this); + AddVariableToCollection(globalMetaVar); + + //******************************************************** + // Loading coordinate systems + // CS stored in two places: as global attributes and as variable's attributes + // for backward compatibility. We're going to use the global attributes, + // as it works even when there is no any variable containing the coordinate system. + for (int i = 0; ; i++) + { + string attName = "coordinates" + (i + 1).ToString(); + + // Inquire information about cs attribute + NcType type; + IntPtr len; + res = NetCDF.nc_inq_att(ncid, NcConst.NC_GLOBAL, attName, out type, out len); + if (res == (int)ResultCode.NC_ENOTATT) // that's all: no more cs + break; + if (res != (int)ResultCode.NC_NOERR) // an error has occurred + HandleResult(res); + + if (type != NcType.NC_CHAR) + throw new Exception("Coordinate system defining attribute is not a string"); + + // Getting the value of the attribute + string attValue = NcGetAttText(ncid, NcConst.NC_GLOBAL, attName, (int)len, out res); + HandleResult(res); + + string[] items = attValue.Split(' '); + string csName = items[0]; // csname axis1 axis2 ... + + // Creating the coordinate system instance + Variable[] axes = new Variable[items.Length - 1]; + bool csFound = true; + for (int j = 1; j < items.Length; j++) // axes names + { + int aid; + Variable a; + if (int.TryParse(items[j], out aid)) + { + csFound = Variables.TryGetByID(aid, out a); + if (!csFound) break; + a = Variables.GetByID(aid); + } + else + { + csFound = ContainsRecent(Variables, items[j]); + if (!csFound) break; + a = Variables[items[j], SchemaVersion.Recent]; + } + axes[j - 1] = a; + } + if (!csFound) break; + CoordinateSystem cs = CreateCoordinateSystem(csName, axes); + } + + // Loading coordinate systems for variable's metadata (backward compatibility) + foreach (Variable var in Variables) + { + if (var.Metadata.ContainsKey("coordinates", SchemaVersion.Recent) && + !var.Metadata.ContainsKey("coordinatesName", SchemaVersion.Recent)) + { + string[] axes = var.Metadata["coordinates", SchemaVersion.Recent].ToString().Split(' '); + List axesList = new List(); + bool axisNotFound = false; + foreach (var axis in axes) + { + try + { + Variable a = Variables[axis, SchemaVersion.Recent]; + axesList.Add(a); + } + catch + { + Trace.WriteLineIf(TraceNetCDFDataSet.TraceError, "Coordinate named " + axis + " is not found in nc file"); + axisNotFound = true; + break; + } + } + if (!axisNotFound) + { + // This is a standart NC file without SDS coordinate systems metadata! + //string csName = "_cs_" + var.Name + "_" + (Guid.NewGuid()).ToString(); + //CreateCoordinateSystem(csName, axesList.ToArray()); + break; + } + } + } + + // Adding coordinate systems to variables + foreach (Variable var in Variables) + { + // Each variable has an entry named coordinates#Name, # - integer index + var.Metadata.ForEach( + delegate (KeyValuePair entry) + { + if (!entry.Key.StartsWith("coordinates") || !entry.Key.EndsWith("Name")) + return; + + try + { + CoordinateSystem cs = CoordinateSystems[(string)entry.Value, SchemaVersion.Recent]; + var.AddCoordinateSystem(cs); + } + catch (Exception ex) + { + Trace.WriteLineIf(TraceNetCDFDataSet.TraceError, "Coordinate system with name " + entry.Value + " not found"); + } + }, + SchemaVersion.Proposed); + } + } // eo if exists + + private void InitializeFromUrl(string url) + { + try + { + initializing = true; + int res; + + if (NetCDFDataSet.TraceNetCDFDataSet.TraceInfo) + { + IntPtr cacheSize, nelems; + float preemption; + if (NetCDF.nc_get_chunk_cache(out cacheSize, out nelems, out preemption) == (int)ResultCode.NC_NOERR) + { + string s = String.Format("NetCDF chunk cache size: {0}, nelems: {1}, preemption: {2}", cacheSize, nelems, preemption); + Trace.WriteLineIf(NetCDFDataSet.TraceNetCDFDataSet.TraceInfo, s); + } + } + + res = NetCDF.nc_open_chunked(url, CreateMode.NC_NOWRITE /*| NetCDF.CreateMode.NC_SHARE*/, out ncid, new IntPtr(defaultCacheSize), new IntPtr(defaultCacheNElems), defaultCachePreemption); + HandleResult(res); + + LoadMetadata(url); + Commit(); + + SetCompleteReadOnly(); + initializing = false; + } + finally + { + IsAutocommitEnabled = false; + } } private void InitializeFromFile(string fileName, ResourceOpenMode openMode) @@ -270,6 +472,7 @@ private void InitializeFromFile(string fileName, ResourceOpenMode openMode) int res; bool exists = File.Exists(fileName); + if (openMode == ResourceOpenMode.Create && exists) { File.Delete(fileName); @@ -293,7 +496,7 @@ private void InitializeFromFile(string fileName, ResourceOpenMode openMode) if (openMode == ResourceOpenMode.CreateNew) throw new IOException("The open mode is createNew but the file already exists"); - if (((File.GetAttributes(fileName) & FileAttributes.ReadOnly) != 0) && openMode != ResourceOpenMode.ReadOnly) + if ((File.GetAttributes(fileName) & FileAttributes.ReadOnly) != 0 && openMode != ResourceOpenMode.ReadOnly) { openMode = ResourceOpenMode.ReadOnly; Trace.WriteLineIf(TraceNetCDFDataSet.TraceWarning, "NetCDFDataSet: Opening file in read only mode"); @@ -319,165 +522,7 @@ private void InitializeFromFile(string fileName, ResourceOpenMode openMode) HandleResult(res); - if (exists) - { - //******************************************************** - // Loading schema - int ndims, nvars, ngatts, unlimdimid; - res = NetCDF.nc_inq(ncid, out ndims, out nvars, out ngatts, out unlimdimid); - HandleResult(res); - - Debug.WriteLineIf(NetCDFDataSet.TraceNetCDFDataSet.TraceInfo, String.Format("NetCDF dataset {0} opened: {1} dimensions, {2} variables, {3} global attributes.", - fileName, ndims, nvars, ngatts)); - - /******************************************************** - // Reading Global Attributes - int nattrs; - res = NetCDF.nc_inq_varnatts(ncid, NetCDF.NC_GLOBAL, out nattrs); - AttributeTypeMap atm = new AttributeTypeMap(ncid, NetCDF.NC_GLOBAL); - NetCDFDataSet.HandleResult(res); - MetadataDictionary globalMetadata = Metadata; - for (int i = 0; i < nattrs; i++) - { - // Name - StringBuilder attName = new StringBuilder(512); - res = NetCDF.nc_inq_attname(ncid, NetCDF.NC_GLOBAL, i, attName); - NetCDFDataSet.HandleResult(res); - string aname = attName.ToString(); - - // Skip out internal attribute - if (aname == AttributeTypeMap.AttributeName) - continue; - - // Type - object value = ReadNetCdfAttribute(NetCDF.NC_GLOBAL, aname, atm); - globalMetadata[aname] = value; - } /**/ - - //******************************************************** - // Loading variables - int[] varIds = new int[nvars]; - res = NetCDF.nc_inq_varids(ncid, out nvars, varIds); - HandleResult(res); - - for (int i = 0; i < nvars; i++) - { - Variable var = ReadNetCdfVariable(varIds[i]); - AddVariableToCollection(var); - } - - Variable globalMetaVar = new NetCDFGlobalMetadataVariable(this); - AddVariableToCollection(globalMetaVar); - - //******************************************************** - // Loading coordinate systems - // CS stored in two places: as global attributes and as variable's attributes - // for backward compatibility. We're going to use the global attributes, - // as it works even when there is no any variable containing the coordinate system. - for (int i = 0; ; i++) - { - string attName = "coordinates" + (i + 1).ToString(); - - // Inquire information about cs attribute - NcType type; - IntPtr len; - res = NetCDF.nc_inq_att(ncid, NcConst.NC_GLOBAL, attName, out type, out len); - if (res == (int)ResultCode.NC_ENOTATT) // that's all: no more cs - break; - if (res != (int)ResultCode.NC_NOERR) // an error has occurred - HandleResult(res); - - if (type != NcType.NC_CHAR) - throw new Exception("Coordinate system defining attribute is not a string"); - - // Getting the value of the attribute - string attValue = NcGetAttText(ncid, NcConst.NC_GLOBAL, attName, (int)len, out res); - HandleResult(res); - - string[] items = attValue.Split(' '); - string csName = items[0]; // csname axis1 axis2 ... - - // Creating the coordinate system instance - Variable[] axes = new Variable[items.Length - 1]; - bool csFound = true; - for (int j = 1; j < items.Length; j++) // axes names - { - int aid; - Variable a; - if (int.TryParse(items[j], out aid)) - { - csFound = Variables.TryGetByID(aid, out a); - if (!csFound) break; - a = Variables.GetByID(aid); - } - else - { - csFound = ContainsRecent(Variables, items[j]); - if (!csFound) break; - a = Variables[items[j], SchemaVersion.Recent]; - } - axes[j - 1] = a; - } - if (!csFound) break; - CoordinateSystem cs = CreateCoordinateSystem(csName, axes); - } - - // Loading coordinate systems for variable's metadata (backward compatibility) - foreach (Variable var in Variables) - { - if (var.Metadata.ContainsKey("coordinates", SchemaVersion.Recent) && - !var.Metadata.ContainsKey("coordinatesName", SchemaVersion.Recent)) - { - string[] axes = var.Metadata["coordinates", SchemaVersion.Recent].ToString().Split(' '); - List axesList = new List(); - bool axisNotFound = false; - foreach (var axis in axes) - { - try - { - Variable a = Variables[axis, SchemaVersion.Recent]; - axesList.Add(a); - } - catch - { - Trace.WriteLineIf(TraceNetCDFDataSet.TraceError, "Coordinate named " + axis + " is not found in nc file"); - axisNotFound = true; - break; - } - } - if (!axisNotFound) - { - // This is a standart NC file without SDS coordinate systems metadata! - //string csName = "_cs_" + var.Name + "_" + (Guid.NewGuid()).ToString(); - //CreateCoordinateSystem(csName, axesList.ToArray()); - break; - } - } - } - - // Adding coordinate systems to variables - foreach (Variable var in Variables) - { - // Each variable has an entry named coordinates#Name, # - integer index - var.Metadata.ForEach( - delegate (KeyValuePair entry) - { - if (!entry.Key.StartsWith("coordinates") || !entry.Key.EndsWith("Name")) - return; - - try - { - CoordinateSystem cs = CoordinateSystems[(string)entry.Value, SchemaVersion.Recent]; - var.AddCoordinateSystem(cs); - } - catch (Exception ex) - { - Trace.WriteLineIf(TraceNetCDFDataSet.TraceError, "Coordinate system with name " + entry.Value + " not found"); - } - }, - SchemaVersion.Proposed); - } - } // eo if exists + if (exists) LoadMetadata(fileName); Commit(); @@ -505,7 +550,7 @@ private static bool ContainsRecent(ReadOnlyVariableCollection vars, string var) } /// - /// + /// /// /// /// @@ -522,7 +567,7 @@ internal static NetCDFDataSet Open(string fileName, bool enableRollback) } /// - /// + /// /// /// /// @@ -567,7 +612,7 @@ internal bool RollbackEnabled #region Overriden Methods /// - /// + /// /// /// /// @@ -581,7 +626,7 @@ protected override Variable CreateVariable(string varName, s } /// - /// + /// /// protected override void OnTransactionOpened() { @@ -614,7 +659,7 @@ private void UpdateDimIds() } /// - /// + /// /// /// /// @@ -626,7 +671,7 @@ protected override bool OnPrecommitting(DataSet.Changes changes) return true; } /// - /// + /// /// /// protected override void OnPrecommit(DataSet.Changes changes) @@ -657,7 +702,7 @@ protected override void OnPrecommit(DataSet.Changes changes) } } /// - /// + /// /// protected override void OnCommit() { @@ -673,7 +718,7 @@ protected override void OnCommit() tempFileName = null; } /// - /// + /// /// protected override void OnRollback() { @@ -719,7 +764,7 @@ protected override void OnRollback() #region Utils /// - /// + /// /// protected override void Dispose(bool disposing) { From e2205b84de623465e19193e509d2f724a0ff141c Mon Sep 17 00:00:00 2001 From: Jonas Juselius Date: Mon, 28 Feb 2022 15:27:48 +0100 Subject: [PATCH 4/6] Prevent file and url in the same uri object --- ScientificDataSet/Providers/NetCDF/NetCDFUri.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/ScientificDataSet/Providers/NetCDF/NetCDFUri.cs b/ScientificDataSet/Providers/NetCDF/NetCDFUri.cs index 8aa834f..1fffe6a 100644 --- a/ScientificDataSet/Providers/NetCDF/NetCDFUri.cs +++ b/ScientificDataSet/Providers/NetCDF/NetCDFUri.cs @@ -24,6 +24,8 @@ public NetCDFUri(string uri) throw new ArgumentException("The given uri must define single file path"); if (GetParameterOccurences("url") > 1) throw new ArgumentException("The given uri must define single DAP url"); + if (GetParameterOccurences("file") > 0 && GetParameterOccurences("url") > 0) + throw new ArgumentException("File and url cannot be specified at the same time"); } /// /// Instantiates an instance of the class with default parameters. From 2491f79991bf5d565ab9603c68df5a01c6254b7c Mon Sep 17 00:00:00 2001 From: Jonas Juselius Date: Wed, 27 Apr 2022 22:50:54 +0200 Subject: [PATCH 5/6] wip: AddDimension --- .../Providers/NetCDF/NetCDFDataSet.cs | 22 +++++++++++++++++++ 1 file changed, 22 insertions(+) diff --git a/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs b/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs index 71e1c7a..438c659 100644 --- a/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs +++ b/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs @@ -1157,6 +1157,28 @@ public void SetChunkSizes(int[] sizes) internal int[] ChunkSizes { get { return chunkSizes; } } #endregion + + /// + /// Create a new Dimension of length len + /// + public void AddDimension(string dim, int len) + { + int res; + int id; + res = NetCDF.nc_inq_dimid(this.NcId, dim, out id); + if (res == (int)ResultCode.NC_EBADDIM) + { + // Creating new dimension + res = NetCDF.nc_def_dim(this.NcId, dim, new IntPtr(len), out id); + chunkSizes = null; + Commit(); + NetCDFDataSet.HandleResult(res); + } + else + { + NetCDFDataSet.HandleResult(res); + } + } } internal class AttributeTypeMap From 33f1cb019b3927aca64860e72a5ab585fef3205d Mon Sep 17 00:00:00 2001 From: Jonas Juselius Date: Fri, 29 Apr 2022 11:09:04 +0200 Subject: [PATCH 6/6] feat: Create Dimensions with fixed length --- .../Providers/NetCDF/NetCDFDataSet.cs | 13 +++++----- .../Providers/NetCDF/NetCdfVariable.cs | 25 ++++++++++++++----- 2 files changed, 25 insertions(+), 13 deletions(-) diff --git a/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs b/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs index 438c659..ab5a28d 100644 --- a/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs +++ b/ScientificDataSet/Providers/NetCDF/NetCDFDataSet.cs @@ -814,7 +814,7 @@ private Variable CreateNetCdfVariable(string name, string[] dims, Type dataType) Invoke(null, new object[] { this, name, dims }); return v; } - + internal object ReadNetCdfAttribute(int varid, string aname, AttributeTypeMap atm) { NcType type; @@ -1159,24 +1159,23 @@ public void SetChunkSizes(int[] sizes) #endregion /// - /// Create a new Dimension of length len + /// Create a new Dimension of length N (N = 0 unlimited) /// - public void AddDimension(string dim, int len) + public void CreateDimension(string dim, int len) { int res; int id; res = NetCDF.nc_inq_dimid(this.NcId, dim, out id); if (res == (int)ResultCode.NC_EBADDIM) { - // Creating new dimension + StartChanges(); res = NetCDF.nc_def_dim(this.NcId, dim, new IntPtr(len), out id); - chunkSizes = null; Commit(); - NetCDFDataSet.HandleResult(res); + HandleResult(res); } else { - NetCDFDataSet.HandleResult(res); + HandleResult(res); } } } diff --git a/ScientificDataSet/Providers/NetCDF/NetCdfVariable.cs b/ScientificDataSet/Providers/NetCDF/NetCdfVariable.cs index 88c726f..2f487f4 100644 --- a/ScientificDataSet/Providers/NetCDF/NetCdfVariable.cs +++ b/ScientificDataSet/Providers/NetCDF/NetCdfVariable.cs @@ -124,13 +124,18 @@ internal static NetCdfVariable Create(NetCDFDataSet dataSet, string na /* Getting exsting dims and creating new */ int[] dimids = new int[dims.Length]; + int[] dimlen = new int[dims.Length]; for (int i = 0; i < dims.Length; i++) { int id; res = NetCDF.nc_inq_dimid(dataSet.NcId, dims[i], out id); if (res == (int)ResultCode.NC_NOERR) { + IntPtr lenp; + res = NetCDF.nc_inq_dimlen(dataSet.NcId, id, out lenp); + NetCDFDataSet.HandleResult(res); dimids[i] = id; + dimlen[i] = lenp.ToInt32(); } else if (res == (int)ResultCode.NC_EBADDIM) { @@ -138,6 +143,7 @@ internal static NetCdfVariable Create(NetCDFDataSet dataSet, string na res = NetCDF.nc_def_dim(dataSet.NcId, dims[i], new IntPtr(NcConst.NC_UNLIMITED), out id); NetCDFDataSet.HandleResult(res); dimids[i] = id; + dimlen[i] = NcConst.NC_UNLIMITED; } else { @@ -197,7 +203,14 @@ internal static NetCdfVariable Create(NetCDFDataSet dataSet, string na for (int i = 0; i < dims.Length; i++) { - chunksizes[i] = new IntPtr(chunk); + if (dimlen[i] < chunk) + { + chunksizes[i] = new IntPtr(dimlen[i]); + } + else + { + chunksizes[i] = new IntPtr(chunk); + } } } @@ -339,7 +352,7 @@ private int NcId #region Transactions /// - /// + /// /// /// internal protected override void OnPrecommit(Variable.Changes proposedChanges) @@ -381,7 +394,7 @@ internal protected override void OnPrecommit(Variable.Changes proposedChanges) * Common netCDF model supports for only one such attribute, * so all others CS except first are not compatible with that model. * Their attribute names are "coordinates2","coordinates3" etc. - * Names of coordinate systems are provided with corresponed + * Names of coordinate systems are provided with corresponed * attributes "coordinatesName","coordinates2Name",... */ if (proposedChanges.CoordinateSystems != null) @@ -642,7 +655,7 @@ private static Array ToMultidimArrayDateTime(double[] ncArray, int[] shape) #region Data Access /// - /// + /// /// /// protected override int[] ReadShape() @@ -698,7 +711,7 @@ private IntPtr[] ConvertToIntPtr(int[] array) return result; } /// - /// + /// /// /// /// @@ -973,7 +986,7 @@ protected override Array ReadData(int[] origin, int[] stride, int[] shape) } } /// - /// + /// /// /// ///