view m-toolbox/classes/+utils/@xml/xmlread.m @ 44:409a22968d5e default

Add unit tests
author Daniele Nicolodi <nicolodi@science.unitn.it>
date Tue, 06 Dec 2011 18:42:11 +0100
parents f0afece42f48
children
line wrap: on
line source

% XMLREAD Reads a XML object
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% DESCRIPTION:   XMLREAD Reads a XML object.
%
% CALL:          obj = xmlread(node);
%                obj = xmlread(node ,'class');
%                obj = xmlread(node ,'ao');
%                obj = xmlread(node ,'time');
%
% XML HIERARCHY: ----------------------   ltpda_object   ----------------------
%
%                <ltpda_object>
%
%                   --> <object> ...
%                   --> <cell>   ...
%
%                </ltpda_object>
%
%                -------------------------   object   -------------------------
%
%                <object> ('type' - attribute)
%
%                   <property> ...
%
%                </object>
%
%                ------------------------   property   ------------------------
%                --------------------------   cell   --------------------------
%
%                <property> ('type', 'prop_name' -attributes)
%            OR  <cell>     ('type'              -attribute)
%
%                   --> atomic element
%                         - empty cell
%                         - empty double
%                         - empty char
%                         - char
%                         - double
%                         - logical
%                         - java (necessary for timezone)
%                   --> <cell>      ...
%                   --> <object>    ...
%                   --> <real_data> ...
%                       <imag_data> ...
%
%                </property>
%            OR  </cell>
%
%                -------   real_data   --------|--------   imag_data   --------
%                                              |
%                <real_data>('type'-attribute) | <imag_data> ('type' -attribute)
%                           ('shape'-attribute)|             ('shape'-attribute)
%                                              |
%                   --> <matrix> ...           |    --> <matrix> ...
%                   --> <vector> ...           |    --> <vector> ...
%                                              |
%                </real_data>                  | </imag_data>
%                                              |
%                ---------   matrix   --------------------   vector   ---------
%                                              |
%                <matrix> ('type' -attribute)  | <vector> ('type' -attribute)
%                                              |
%                   row vector (double)        |    column vector (double)
%                                              |
%                </matrix>                     | </vector>
%
% SYMBOLS:       --> Marks a choice between alternatives.
%                ... Indicate that an element may be repeated.
%
% VERSION:       $Id: xmlread.m,v 1.8 2011/03/28 17:04:27 ingo Exp $
%
% HISTORY:       31-01-2008 Diepholz
%                   Creation
%
% SEE ALSO:      utils.xml.xmlwrite
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function values = xmlread(rootNode, obj_name)
  
%   if nargin == 1
%     node_name = mchar(rootNode.getNodeName);
%     
%     switch node_name
%       case 'ltpda_object'
%         
%         % Get the xml version from the document node
%         % node.getOwnerDocument.item(0).hasAttribute('ltpda_version')
%         if rootNode.hasAttribute('ltpda_version')
%           xml_ver = strtok(mchar(rootNode.getAttribute('ltpda_version')));
%         else
%           xml_ver = '1.0';
%         end
%         
%         tbx_ver = strtok(getappdata(0, 'ltpda_version'));
%         
%         % set the application data 'xml_ver' if the xml version is lower than the
%         % toolbox version.
%         % This application data is nedded in the function xml_read_object to
%         % update the object string.
%         if utils.helper.ver2num(xml_ver) < utils.helper.ver2num(tbx_ver)
%           update = xml_ver;
%         else
%           update = '';
%         end
%         
%         values = xml_read_ltpda_object(rootNode, update);
%       case 'object'
%         values = xml_read_object(rootNode, false);
%       case 'property'
%         values = xml_read_property_cell(rootNode, false);
%       case 'cell'
%         values = xml_read_property_cell(rootNode, false);
%       case 'real_data'
%         values = xml_read_real_imag_data(rootNode);
%       case 'imag_data'
%         values = xml_read_real_imag_data(rootNode);
%       case 'matrix'
%         %%%%% row vector
%         values = sscanf(mchar(child_node.getTextContent), '%g ', [1,inf]);
%       case 'vector'
%         %%%%% column vector
%         values = sscanf(mchar(child_node.getTextContent), '%g ');
%       otherwise
%         %%%%% Search for the next valid node name.
%         for ii = 1:rootNode.getLength
%           item = rootNode.item(ii-1);
%           if item.hasChildNodes
%             values = utils.xml.xmlread(item);
%           end
%         end
%     end
  if nargin >= 1
    
    values      = [];
    valuesShape = [];
    h           = history.initObjectWithSize(1,0);
    
    queryNode = rootNode.getElementsByTagName('ltpda_object');
    
    for ii= 1:queryNode.getLength
      
      LTPDANode = queryNode.item(ii-1);
      if LTPDANode.getNodeType == LTPDANode.ELEMENT_NODE
        
        if LTPDANode.hasAttribute('ltpda_version')
          ltpda_version = strtok(utils.xml.mchar(LTPDANode.getAttribute('ltpda_version')));
        else
          ltpda_version = '1.0';
        end
        
        if  (utils.helper.ver2num(ltpda_version) > utils.helper.ver2num('2.3')) || ...
            (strcmp(strtok(ltpda_version), '2.3'))
          %%%%%%%%%%%%%%%%%%   reading of a new XML file   %%%%%%%%%%%%%%%%%%
          
          for jj = 1:LTPDANode.getLength
            
            objNode = LTPDANode.item(jj-1);
            if objNode.getNodeType == objNode.ELEMENT_NODE
              
              className = utils.xml.mchar(objNode.getNodeName());
              
              if strcmp(className, 'historyRoot')
                
                h = history(objNode, history.initObjectWithSize(1,0));
                
              else
                valuesShape = utils.xml.getShape(objNode);
                val = feval(className, objNode, h);
                if ~exist('obj_name', 'var') || strcmp(class(val), obj_name)
                  values = [values val];
                else
                  error('### Skip the read object because it is from the class [%s] and not from the class [%s].', class(val), obj_name);
                end
              end
              
            end
          end
          values = reshape(values, valuesShape);
          
        else
          %%%%%%%%%%%%%%%%%%   reading of a old XML file   %%%%%%%%%%%%%%%%%%
          
          for jj = 1:LTPDANode.getLength
            objNode = LTPDANode.item(jj-1);
            if objNode.hasChildNodes
              % Get node name
              node_name = utils.xml.mchar(objNode.getNodeName);
              switch node_name
                case 'object'
                  val =  xml_read_object(objNode, ltpda_version);
                  if isempty(valuesShape), valuesShape = getShape(objNode); end
                  if ~exist('obj_name', 'var') || strcmp(class(val), obj_name)
                    values = [values val];
                  else
                    error('### Skip the read object because it is from the class [%s] and not from the class [%s].', class(val), obj_name);
                  end
                otherwise
              end
            end
          end % over all childs
          
          if ~isempty(valuesShape)
            values = reshape(values, valuesShape);
          end
        end
        
      end % LTPDANode.ELEMENT_NODE
      
    end % ii= 1:queryNode.getLength
    
  else
    error('### Invalid command of this function');
  end
  
  
end


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% DESCRIPTION: XML_READ_LTPDA_OBJECT Reads a ltoda_object element with the form:
%
%              <ltpda_object>
%
%                 --> <object> ...
%                 --> <cell>   ...
%
%              </ltpda_object>
%
% SYMBOLS:     --> Marks a choice between alternatives.
%              ... Indicate that an element may be repeated.
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function values = xml_read_ltpda_object(node, update)
  
  values = [];
  for ii = 1:node.getLength
    child_node = node.item(ii-1);
    if child_node.hasChildNodes
      % Get Node name
      node_name = mchar(child_node.getNodeName);
      switch node_name
        case 'object'
          values = [values xml_read_object(child_node, update)];
        case 'cell'
          values{end+1} = xml_read_property_cell(child_node, update);
        otherwise
          error('### The ''ltpda_object'' element can not contain a [%s] element.', node_name);
      end
    end
  end
end


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% DESCRIPTION: XML_READ_OBJECT Reads a object element with the form:
%
%              <object> ('type' - attribute)
%
%                 <property> ...
%
%              </object>
%
% SYMBOLS:     ... Indicate that an element may be repeated.
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function obj = xml_read_object(node, update)
  
  obj_type = mchar(node.getAttribute('type'));
  obj = [];
  
  % Set some properties which are stored in the attributes of the 'object' node
  obj = getSpecialObjectAttributes(obj, node, update);
  
  for ii = 1:node.getLength
    child_node = node.item(ii-1);
    %%% Read only the property if the node is a ELEMENT_NODE.
    if child_node.getNodeType == child_node.ELEMENT_NODE
      % Get node name
      node_name = mchar(child_node.getNodeName);
      % get property name from the attribute 'prop_name'
      prop_name = mchar(child_node.getAttribute('prop_name'));
      switch node_name
        case 'property'
          prop_value = xml_read_property_cell(child_node, update);
          
          try
            obj.(prop_name) = prop_value;
          catch
            warning('\n\n### Skip the unknown property [%s] in the object [%s]\n', prop_name, class(obj));
            disp([char(10) 'Code me up to convert me into the new structure !!!' char(10) ]);
          end
          
        otherwise
          error('### The ''object'' element can not contain a [%s] element.', node_name);
      end
    end
  end
  
  if isempty(update)
    % Make sure that the version we want to update to is far in the future.
    % In other words, we don't do an update if it's not necessary.
    update = '1000000000000000';
  else
    old_c = {'timeformat', 'pole', 'zero'};
    new_c = {'',           'pz',    'pz'};
    
    idx = strmatch(obj_type, char(old_c), 'exact');
    if ~isempty(idx)
      obj_type = new_c{idx};
    end
    
    %     if ~isempty(obj_type)
    %       % Update the structure to the current ltpda_version
    %       update_fcn = [obj_type '.update_struct'];
    %       obj = feval(update_fcn, obj, update);
    %     end
  end
  
  if ~isempty(obj_type) && ~strcmp(obj_type, 'struct')
    obj.class  = obj_type;
    obj.tbxver = update;
    obj = feval(obj_type, obj);
  elseif isempty(obj) && strcmp(obj_type, 'struct')
    obj = struct();
  end
end


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% DESCRIPTION: XML_READ_PROPERTY_CELL Reads a property element or a cell element
%              with the form:
%
%              <property> ('type', 'prop_name' -attributes)
% OR:          <cell>     ('type'              -attribute)
%
%                 --> atomic element
%                       - empty cell
%                       - empty double
%                       - empty char
%                       - char
%                       - double
%                       - logical
%                       - java (necessary for timezone)
%                 --> <cell>      ...
%                 --> <object>    ...
%                 --> <real_data> ...
%                     <imag_data> ...
%
%              </property>
% OR:          </cell>
%
% SYMBOLS:     --> Marks a choice between alternatives.
%              ... Indicate that an element may be repeated.
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function prop_value = xml_read_property_cell(node, update)
  
  prop_value = [];
  real_value = [];
  imag_value = [];
  shape      = [];
  
  %%% Is the length of the node is equal 1 or zero
  %%% then is the property a atomic element
  if node.getLength == 0 || node.getLength == 1
    prop_value = xml_read_atomic_element(node);
    return
  end
  
  for ii = 1:node.getLength
    child_node = node.item(ii-1);
    if child_node.getNodeType == child_node.ELEMENT_NODE
      %%% Get the node name
      node_name = mchar(child_node.getNodeName);
      switch node_name
        case 'cell'
          if isempty(shape), shape = getShape(node); end
          prop_value{end+1} = xml_read_property_cell(child_node, update);
        case 'object'
          if isempty(shape), shape = getShape(child_node); end
          prop_value = [prop_value xml_read_object(child_node, update)];
        case 'real_data'
          real_value = xml_read_real_imag_data(child_node);
        case 'imag_data'
          imag_value = xml_read_real_imag_data(child_node);
        otherwise
          error('### The ''property'' or ''cell'' element can not contain a [%s] element.', node_name);
      end
      
      %%% Is real_value and imag_value filled then return the complex value
      if ~isempty(real_value) && ~isempty(imag_value)
        prop_value = complex(real_value, imag_value);
        %%% Is real_value filled then return the read value
      elseif ~isempty(real_value)
        prop_value = real_value;
      end
    end
  end
  
  %%% Reshape the prop_value if necessary
  if ~isempty(shape)
    prop_value = reshape(prop_value, shape);
  end
  
end


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% DESCRIPTION: XML_READ_REAL_IMAG_DATA Reads a real_data element or a
%              imag_data element with the form:
%
%              <real_data> ('type' -attribute) <imag_data> ('type' -attribute)
%                          ('shape'-attribute)             ('shape'-attribute)
%
%                 --> <matrix> ...                --> <matrix> ...
%                 --> <vector> ...                --> <vector> ...
%
%              </real_data>                    </imag_data>
%
%              <matrix> ('type' -attribute)    <vector> ('type' -attribute)
%
%                 row vector (double)             column vector (double)
%
%              </matrix>                       </vector>
%
% SYMBOLS:     --> Marks a choice between alternatives.
%              ... Indicate that an element may be repeated.
%
% REMARK:      The matrix elements will be read as a row vector.
%              The vector elements will be read as a column vector.
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function value = xml_read_real_imag_data(node)
  
  value            = [];
  header_displayed = true;
  
  %%% Get the shape of the matrix/vector
  shape = getShape(node);
  
  %   value = zeros(shape);
  %   idx   = 1;
  
  node_name = '';
  for ii = 1:node.getLength
    child_node = node.item(ii-1);
    if child_node.hasChildNodes
      %%% Get node name
      node_name = mchar(child_node.getNodeName);
      %%% Get type
      node_type = mchar(child_node.getAttribute('type'));
      switch node_name
        case 'matrix'
          %%% row vector
          if strcmp(node_type, 'sym')
            matrix_row = sym(mchar(child_node.getTextContent));
          else
            matrix_row = sscanf(mchar(child_node.getTextContent), '%g ', [1,inf]);
          end
          value = [value; matrix_row];
          %           value(idx,:) = matrix_row;
          %           idx = idx + 1;
        case 'vector'
          %%% column vector
          if strcmp(node_type, 'sym')
            vector_column = sym(mchar(child_node.getTextContent));
          else
            vector_column = sscanf(mchar(child_node.getTextContent), '%g ');
          end
          value = [value; vector_column];
          %           value(idx:idx + length(vector_column) - 1) = vector_column;
          %           idx = idx + length(vector_column);
          header_displayed = TerminalOutput(node, header_displayed, node_name, length(value));
        otherwise
          error('### The ''real_data'' or ''imag_data'' element can not contain a [%s] element.', node_name);
      end
    end
  end
  
  if strcmp(node_name, 'matrix')
    header_displayed = true;
    TerminalOutput(node, header_displayed, node_name, size(value,1));
  end
  value = reshape(value, shape);
  
end


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% DESCRIPTION: xml_read_atomic_element Reads a atomic element.
%              This function identify the following types:
%                 --> atomic element
%                       - empty cell
%                       - empty double
%                       - empty char
%                       - char
%                       - double
%                       - logical
%                       - java (necessary for timezone)
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function values = xml_read_atomic_element(node)
  
  values = '';
  
  type = mchar(node.getAttribute('type'));
  switch type
    case {'', 'char'}
      shape = getShape(node);
      if (any(shape==1))
        values = mchar(node.getTextContent);
      else
        values = reshape(mchar(node.getTextContent), shape);
      end
      values = strrep(values, '<NEW_LINE>', '\n');
      values = strrep(values, 'CVS_TAG', '$');
      %%% Special case if 'values' is empty because
      %%% there are more than an one empty char fields.
      %%% For example: '' AND char('', '') AND char('', '', '') AND ...
      if isempty(values)
        if any(shape)
          char_str = 'char(';
          for ii = 1:max(shape)
            char_str = [char_str, ''''', '];
          end
          char_str = char_str(1:end-2);
          cmd = strcat('values = ', char_str, ');');
          eval(cmd);
        else
          values = '';
        end
      end
      
    case {'double', 'uint64', 'uint32', 'int32', 'int64'}
      number = mchar(node.getTextContent);
      
      %%% Special case if 'values' is empty because
      %%% there are more than an one empty double fields.
      %%% For example: [] AND zeros(1,0) AND zeros(0,1) AND zeros(2,0) AND ...
      if isempty(number)
        shape = getShape(node);
        values = zeros(shape);
      else
        values = str2double(number);
      end
      
      % cast to other number type
      if strcmp(type, 'uint64')
        values = uint64(values);
      elseif strcmp(type, 'uint32')
        values = uint32(values);
      elseif strcmp(type, 'int64')
        values = int64(values);
      elseif strcmp(type, 'int32')
        values = int32(values);
      end
      
    case 'sym'
      number = mchar(node.getTextContent);
      
      %%% Special case if 'values' is empty because
      %%% there are more than an one empty double fields.
      %%% For example: [] AND zeros(1,0) AND zeros(0,1) AND zeros(2,0) AND ...
      if isempty(number)
        shape = getShape(node);
        values = sym(shape);
      else
        values = sym(number);
      end
      
    case 'cell'
      cell_str = mchar(node.getTextContent);
      
      %%% Special case if 'values' is empty because
      %%% there are more than an one empty cell fields.
      %%% For example: {} AND cell(1,0) AND cell(0,1) AND cell(2,0) AND ...
      if isempty(cell_str)
        shape = getShape(node);
        values = cell(shape);
      else
        error('### Should not happen because a not empty ''cell'' node have several child nodes.');
      end
      
    case 'logical'
      cmd = ['logical(' mchar(node.getTextContent) ');'];
      values = eval(cmd);
      
    case 'sun.util.calendar.ZoneInfo'
      values = java.util.TimeZone.getTimeZone(mchar(node.getTextContent));
      
    otherwise
      
      if any(strcmp(utils.helper.ltpda_non_abstract_classes, type))
        %%% Special case for reading an object with the size Nx0 or 0xN
        shape = getShape(node);
        cmd = sprintf('%s.initObjectWithSize', type);
        values = feval(cmd, shape(1), shape(2));
      else
        error('### Unknown type attribute [%s].', type');
      end
  end
end


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% FUNCTION:    shape = getShape(node)
%
% INPUTS:      node - A DOM node with an attribute 'shape'
%
% DESCRIPTION: getShape Helper function to get the shape from the
%              attribute 'shape' as a set of double.
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function shape = getShape(node)
  shape = [NaN NaN];
  if node.hasAttribute('shape')
    shape_str = mchar(node.getAttribute('shape'));
    x_idx     = strfind(shape_str, 'x');
    shape     = [str2double(shape_str(1:x_idx-1)) str2double(shape_str(x_idx+1:end))];
  end
end


%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% FUNCTION:    obj = getSpecialObjectAttributes(obj, node)
%
% INPUTS:      obj  - a object structure or empty array
%              node - A DOM node with the node name 'object'
%
% DESCRIPTION:
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function obj = getSpecialObjectAttributes(obj, node, update)
  
  if strcmp(mchar(node.getNodeName), 'object')
    
    if strcmp(mchar(node.getAttribute('type')), 'time')
      
      % Get attributes for a time object
      % Check if the attribute exist (necessary for backwards compability)
      if (node.hasAttribute('utc'))
        obj.utc_epoch_milli = str2double(mchar(node.getAttribute('utc')));
        obj.timezone        = java.util.TimeZone.getTimeZone(mchar(node.getAttribute('timezone')));
        obj.timeformat      = mchar(node.getAttribute('timeformat'));
      end
      
    elseif strcmp(mchar(node.getAttribute('type')), 'provenance')
      
      % Get attributes for a provenance object
      % Check if the attribute exist (necessary for backwards compability)
      if (node.hasAttribute('info'))
        objStr = mchar(node.getAttribute('info'));
        obj = provenance.setFromEncodedInfo(obj, objStr);
      end
      
    elseif strcmp(mchar(node.getAttribute('type')), 'minfo')
      
      % Get attributes for a minfo object
      % Check if the attribute exist (necessary for backwards compability)
      if (node.hasAttribute('info'))
        objStr = mchar(node.getAttribute('info'));
        obj = minfo.setFromEncodedInfo(obj, objStr);
      end
      
    elseif strcmp(mchar(node.getAttribute('type')), 'param')
      
      % Get attributes for a param object
      % Check if the attribute exist (necessary for backwards compability)
      if (node.hasAttribute('key'))
        obj.key = mchar(node.getAttribute('key'));
        
        for ii = 1:node.getLength
          childNode = node.item(ii-1);
          %%% Read only the property if the node is a ELEMENT_NODE.
          if childNode.getNodeType == childNode.ELEMENT_NODE
            obj.val = xml_read_property_cell(childNode, update);
          end
        end
      end
      
    else
      % Get general attributes.
      if node.hasAttribute('name')
        obj.name = mchar(node.getAttribute('name'));
      end
      if node.hasAttribute('description')
        obj.description = mchar(node.getAttribute('description'));
      end
      if node.hasAttribute('UUID')
        obj.UUID = mchar(node.getAttribute('UUID'));
      end
      if node.hasAttribute('created')
        obj.created = str2double(mchar(node.getAttribute('created')));
      end
      if node.hasAttribute('proctime')
        obj.proctime = str2double(mchar(node.getAttribute('proctime')));
      end
    end
    
    
  end
end

%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%
% FUNCTION:    header_displayed = TerminalOutput(node, header_displayed, node_name, number)
%
% DESCRIPTION: Displays the terminal output.
%
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
function header_displayed = TerminalOutput(node, header_displayed, node_name, number)
  
  import utils.const.*
  
  THRESHOLD_DISP_MATRIX = 10;
  THRESHOLD_DISP_VECTOR = 1000;
  
  showing = false;
  
  if strcmp(node_name, 'matrix')
    if number >= THRESHOLD_DISP_MATRIX
      showing = true;
      add_text = 'matrix lines';
    end
  else
    if number >= THRESHOLD_DISP_VECTOR
      showing = true;
      add_text = 'data samples';
    end
  end
  
  if showing == true
    
    parent = node.getParentNode; % The parent node have the 'property name' information
    
    if header_displayed == true
      if parent.hasAttribute('prop_name')
        disp_prop_name = mchar(parent.getAttribute('prop_name'));
      else
        disp_prop_name = 'Unknown Property Name';
      end
      utils.helper.msg(msg.PROC2, 'Reading property: %s', disp_prop_name);
      
      if strcmp(mchar(node.getNodeName), 'real_data')
        utils.helper.msg(msg.PROC2, 'Reading real data');
      else
        utils.helper.msg(msg.PROC2, 'Reading imag data');
      end
      header_displayed = false;
    end
    
    utils.helper.msg(msg.PROC2, 'Read %d %s', number, add_text);
  end
end

function c = mchar(s)
  c = cell(s);
  c = c{1};
end