Blob Blame History Raw
// Copyright (c) 1998, 1999 Matthias Clasen 
// See the file COPYING for copying permission.

#include "config.h"

#include "EntityApp.h"
#include "Entity.h"
#include "Text.h"
#include "Sd.h"
#include "sptchar.h"
#include "OutputCharStream.h"
#include "SpcatMessages.h"

#ifdef SP_NAMESPACE
using namespace SP_NAMESPACE;
#endif

class SpcatApp : public EntityApp {
public:
  SpcatApp();
  void processOption(AppChar opt, const AppChar *arg);
  int processArguments(int argc, AppChar **argv);
  int processSysid(const StringC &);
private:
  Boolean lastFound_;
};

SP_DEFINE_APP(SpcatApp)

SpcatApp::SpcatApp()
 : lastFound_(true)
{
  registerOption('P', SP_T("public-id"), 
                 SpcatMessages::literal, SpcatMessages::PHelp);
  registerOption('S', SP_T("system-id"), 
                 SpcatMessages::literal, SpcatMessages::SHelp);
  registerOption('p', SP_T("parameter-entity"), 
                 SpcatMessages::name, SpcatMessages::pHelp);
  registerOption('d', SP_T("doctype"), 
                 SpcatMessages::name, SpcatMessages::dHelp);
  registerOption('l', SP_T("linktype"),
                 SpcatMessages::name, SpcatMessages::lHelp);
  registerOption('e', SP_T("entity"), 
                 SpcatMessages::name, SpcatMessages::eHelp);
  registerOption('n', SP_T("notation"), 
                 SpcatMessages::name, SpcatMessages::nHelp);
  registerOption('s', SP_T("declaration"), 
                 SpcatMessages::name, SpcatMessages::sHelp);
  registerInfo(SpcatMessages::info1);
  registerInfo(SpcatMessages::info2);
  registerInfo(SpcatMessages::info3);
  registerInfo(SpcatMessages::info4);
}

void SpcatApp::processOption(AppChar opt, const AppChar *arg)
{
  //FIXME This is to avoid calling entityManager() for 
  //EntityApp options, since entityManager() fixes the set 
  //of catalogs and clearEntityManager() didn't work for me.
  switch (opt) {
  case 'P':
  case 'S':
  case 'e':
  case 'p':
  case 'd':
  case 'l':
  case 'n':
  case 's':
    break;
  default: 
    EntityApp::processOption(opt, arg);
    return ;
  }

  ExternalId exId;
  StringC name;
  EntityDecl::DeclType type = EntityDecl::generalEntity; 
  Ptr<EntityManager> ema(entityManager().pointer());
  Sd sd(ema);
  Syntax syntax(sd); 
  Text txt;
  Location loc;

  switch (opt) {
  case 'P':
    for (int i = 0; arg[i] != '\0'; i++) 
      txt.addChar(arg[i], loc);
    const MessageType1 *err, *err1;
    exId.setPublic(txt, systemCharset(), syntax.space(), err, err1);
    break;
  case 'S':
    for (int i = 0; arg[i] != '\0'; i++) 
      txt.addChar(arg[i], loc);
    exId.setSystem(txt);
    break;
  case 'e':
    type = EntityDecl::generalEntity;
    for (int i = 0; arg[i] != '\0'; i++) 
      name += arg[i]; 
    break;
  case 'p':
    type = EntityDecl::parameterEntity;
    //FIXME get a complete syntax from somewhere
    //so that we can say
    //name += syntax.peroDelim();
    name += '%';
    for (int i = 0; arg[i] != '\0'; i++) 
      name += arg[i]; 
    break;
  case 'd':
    type = EntityDecl::doctype;
    for (int i = 0; arg[i] != '\0'; i++) 
      name += arg[i]; 
    break;
  case 'l':
    type = EntityDecl::linktype;
    for (int i = 0; arg[i] != '\0'; i++) 
      name += arg[i]; 
    break;
  case 'n':
    type = EntityDecl::notation;
    for (int i = 0; arg[i] != '\0'; i++) 
      name += arg[i]; 
    break;
  case 's':
    type = EntityDecl::sgml;
    break;
  default:
    EntityApp::processOption(opt, arg);
    return ;
  }

  ExternalTextEntity ent(name, type, loc, exId);
  StringC ret;
  ConstPtr<EntityCatalog> cat = 
         entityManager()->makeCatalog(ret, systemCharset(), *this);
  StringC sysid;

  if ((type == EntityDecl::sgml && cat->sgmlDecl(systemCharset(), *this, sysid, ret)) 
      || cat->lookup(ent, syntax, systemCharset(), *this, ret)) {
    lastFound_ = true;
    //FIXME should use makeStdOut(), but it didn't work for me.
    //OutputCharStream *os = makeStdOut();
    //*os << ret << '\n';	
    //delete os;
    for (int i = 0; i < ret.size(); i++) 
      printf("%c", ret[i]);
    printf("\n");
  } 
  else
    lastFound_ = false;
}

int SpcatApp::processArguments(int argc, AppChar **argv)
{
  if (lastFound_) 
    return 0;
  else 
    return 1;
}

int SpcatApp::processSysid(const StringC &)
{
  return 0;
}