[Q]: Как прикpyтить к файлу расширенный атрибут - исходник

[A]: Dmitry Zavalishin (2:5020/32)

Елки-палки, как долго я боялся за это браться. Оказалось, если не лезть в дебри, то все вполне терпимо.

Разъяснения:

static bool set_ea( const char *file_name, const char *ea_name, const char *ea_data, int ea_data_len );

Берет и втыкает в file_name расширенный атрибут по имени ea_name, стирая старый полностью. Значение (двоичное) берется из ea_data, длина его в байтах - из ea_data_len.

bool set_ea_ASCII( const char *fn, const char* ea_name, string data );

Кодирует строку data в соответствии с правилами полуоси и засовывает результат в указаный EA соответствующего файла. Это пригодно для EA типа “.SUBJECT”, “.LONGNAME”.

bool set_ea_MVMT_ASCII( const char *fn, const char* ea_name, vector <string> data );

Кодирует группу строк как мультитиповый мультиэлементный EA и пристегивает его к файлу. Это пригодно для EA типа “.HISTORY”, “.COMMENTS”. Вообще тут пошел бы и MVST, но, говорят, традиционно используется MVMT.

bool set_ea_MVST_ASCII( const char *fn, const char* ea_name, vector <string> data );

Кодирует группу строк как однотиповый мультиэлементный EA и пристегивает его к файлу. Это пригодно для EA типа “.KEYPHRASES”. Эту функцию на данный момент я даже не проверял в работе, так что если что - извините.

/*\ * The software included, file formats and basic algorithms are * copyright (C) 1995,96 by Dmitry Zavalishin. All rights reserved. * * Module: OS/2 EAs * * $Log: ea.C $ * Revision 1.1 1996/07/22 02:48:05 dz * Initial revision * * * * \*/

#include “frip.h” #include “ea.h” #ifdef OS2 #define INCL_DOSFILEMGR /* File Manager values */ #define INCL_DOSERRORS /* DOS error values */ #include <os2.h> #include <stdio.h> #include <string.h> #pragma pack(4) static bool set_ea( const char *file_name, const char *ea_name, const char *ea_data, int ea_data_len ) { APIRET rc = NO_ERROR; /* Return code */ EAOP2 op; char * databuf = new char[(64*2*1024)+1024]; twice 64K for EA data + 1024 for any case

  
  op.fpGEA2List = (PGEA2LIST)0;
  op.fpFEA2List = (PFEA2LIST)databuf;
  
  //char *attname = ".SUBJECT";
  int  ea_name_len = strlen( ea_name );
  if( ea_name_len > 255 )
      {
      Error("EA name too long: "+string(ea_name));
      return Err;
      }
  
  //char datname[] = "\xFD\xFF\x14\x00More Stupid Subject!\x0"; // FFFD, 2-byte len, text
  //char  datlen = sizeof( datname );
  
  
  char *databufp = databuf + sizeof(long);
  *((long*)databufp) = 0; // Next field offset is zero - just one field here
  databufp += sizeof(long);
  *databufp++ = 0; // not critical
  *databufp++ = (char)ea_name_len;
  *((short*)databufp) = ea_data_len;
  databufp += sizeof(short);
  memcpy( databufp, ea_name, ea_name_len+1 ); // with trailing zero
  databufp += ea_name_len+1;
  memcpy( databufp, ea_data, ea_data_len ); // with trailing zero
  databufp += ea_data_len;
  
  *((long*)databuf) = databufp-databuf; // Size of all that stuff
  
  rc = DosSetPathInfo( file_name, FIL_QUERYEASIZE, &op, sizeof(op), 0);
  if (rc != NO_ERROR)
      {
      Error("DosSetPathInfo error");
      return Err;
      }
  
  delete [] databuf;
  
  return Ok;
  }

#endif OS2 class binbuf { public: char *b; binbuf( int size ) { b = new char[size]; } ~binbuf() { delete [] b; } }; bool set_ea_ASCII( const char *fn, const char* ea_name, string data ) { #ifdef OS2 binbuf b(64*1024); char *buf = b.b; *((short*)buf) = EAT_ASCII; buf += sizeof(short); *((short*)buf) = data.length(); buf += sizeof(short); strcpy( buf, data.c_str() ); return set_ea( fn, ea_name, b.b, data.length() + 4 ); #else OS2

  return Ok;

#endif OS2 } bool set_ea_MVMT_ASCII( const char *fn, const char* ea_name, vector <string> data ) { #ifdef OS2 binbuf b(64*1024); char *buf = b.b; *((short*)buf) = EAT_MVMT; buf += sizeof(short); Default CodePage == 0

  int len = data.size();
  for( int i = 0; i < len; i++ )
      {
      if( (64*1024-1) < ((buf-b.b) + data[i].length() + 4) )
          {
          Error("vector too big to fit in EA, cut it off :(");
          break;
          }
      
      *((short*)buf) = EAT_ASCII;          buf += sizeof(short);
      *((short*)buf) = data[i].length();   buf += sizeof(short);
      strcpy( buf, data[i].c_str() );      buf += data[i].length();
      }
  
  return set_ea( fn, ea_name, b.b, buf-b.b );

#else OS2 return Ok; #endif OS2

  }

bool set_ea_MVST_ASCII( const char *fn, const char* ea_name, vector <string> data )

  {

#ifdef OS2

  binbuf b(64*1024);
  
  char *buf = b.b;
  
  *((short*)buf) = EAT_MVST;       buf += sizeof(short);
    // Default CodePage == 0
  *((short*)buf) = 0;              buf += sizeof(short);
  *((short*)buf) = data.size();    buf += sizeof(short);
  *((short*)buf) = EAT_ASCII;      buf += sizeof(short);
  
  int len = data.size();
  for( int i = 0; i < len; i++ )
      {
      
      if( (64*1024-1) < ((buf-b.b) + data[i].length() + 4) )
          {
          Error("vector too big to fit in EA, cut it off :(");
          break;
          }
      
      *((short*)buf) = data[i].length();   buf += sizeof(short);
      strcpy( buf, data[i].c_str() );      buf += data[i].length();
      }
  
  return set_ea( fn, ea_name, b.b, buf-b.b );

#else OS2 return Ok; #endif OS2

  }