Blame ext/repo_releasefile_products.c

Packit 54873f
/*
Packit 54873f
 * repo_products.c
Packit 54873f
 *
Packit 54873f
 * Parses all files below 'proddir'
Packit 54873f
 * See http://en.opensuse.org/Product_Management/Code11
Packit 54873f
 *
Packit 54873f
 *
Packit 54873f
 * Copyright (c) 2008, Novell Inc.
Packit 54873f
 *
Packit 54873f
 * This program is licensed under the BSD license, read LICENSE.BSD
Packit 54873f
 * for further information
Packit 54873f
 */
Packit 54873f
Packit 54873f
#include <sys/types.h>
Packit 54873f
#include <unistd.h>
Packit 54873f
#include <stdio.h>
Packit 54873f
#include <stdlib.h>
Packit 54873f
#include <string.h>
Packit 54873f
#include <assert.h>
Packit 54873f
#include <dirent.h>
Packit 54873f
#include <ctype.h>
Packit 54873f
#include <errno.h>
Packit 54873f
Packit 54873f
#include "pool.h"
Packit 54873f
#include "repo.h"
Packit 54873f
#include "util.h"
Packit 54873f
#define DISABLE_SPLIT
Packit 54873f
#include "tools_util.h"
Packit 54873f
#include "repo_releasefile_products.h"
Packit 54873f
Packit 54873f
#define BUFF_SIZE 8192
Packit 54873f
Packit 54873f
struct parsedata {
Packit 54873f
  Repo *repo;
Packit 54873f
  struct joindata jd;
Packit 54873f
};
Packit 54873f
Packit 54873f
static void
Packit 54873f
add_releasefile_product(struct parsedata *pd, FILE *fp)
Packit 54873f
{
Packit 54873f
  Repo *repo = pd->repo;
Packit 54873f
  Pool *pool = repo->pool;
Packit 54873f
  char buf[BUFF_SIZE];
Packit 54873f
  Id name = 0;
Packit 54873f
  Id arch = 0;
Packit 54873f
  Id version = 0;
Packit 54873f
  int lnum = 0; /* line number */
Packit 54873f
  char *ptr, *ptr1;
Packit 54873f
Packit 54873f
  /* parse /etc/<xyz>-release file */
Packit 54873f
  while (fgets(buf, sizeof(buf), fp))
Packit 54873f
    {
Packit 54873f
      /* remove trailing \n */
Packit 54873f
      int l = strlen(buf);
Packit 54873f
      if (l && buf[l - 1] == '\n')
Packit 54873f
	buf[--l] = 0;
Packit 54873f
      ++lnum;
Packit 54873f
Packit 54873f
      if (lnum == 1)
Packit 54873f
	{
Packit 54873f
	  /* 1st line, <name> [(<arch>)] */
Packit 54873f
	  ptr = strchr(buf, '(');
Packit 54873f
	  if (ptr)
Packit 54873f
	    {
Packit 54873f
	      ptr1 = ptr - 1;
Packit 54873f
	      *ptr++ = 0;
Packit 54873f
	    }
Packit 54873f
	  else
Packit 54873f
	    ptr1 = buf + l - 1;
Packit 54873f
Packit 54873f
	  /* track back until non-blank, non-digit */
Packit 54873f
	  while (ptr1 > buf
Packit 54873f
		 && (*ptr1 == ' ' || isdigit(*ptr1) || *ptr1 == '.'))
Packit 54873f
	    --ptr1;
Packit 54873f
	  *(++ptr1) = 0;
Packit 54873f
	  name = pool_str2id(pool, join2(&pd->jd, "product", ":", buf), 1);
Packit 54873f
Packit 54873f
	  if (ptr)
Packit 54873f
	    {
Packit 54873f
	      /* have arch */
Packit 54873f
	      char *ptr1 = strchr(ptr, ')');
Packit 54873f
	      if (ptr1)
Packit 54873f
		{
Packit 54873f
		  *ptr1 = 0;
Packit 54873f
		  /* downcase arch */
Packit 54873f
		  ptr1 = ptr;
Packit 54873f
		  while (*ptr1)
Packit 54873f
		    {
Packit 54873f
		      if (isupper(*ptr1))
Packit 54873f
			 *ptr1 = tolower(*ptr1);
Packit 54873f
		      ++ptr1;
Packit 54873f
		    }
Packit 54873f
		  arch = pool_str2id(pool, ptr, 1);
Packit 54873f
		}
Packit 54873f
	    }
Packit 54873f
	}
Packit 54873f
      else if (strncmp(buf, "VERSION", 7) == 0)
Packit 54873f
	{
Packit 54873f
	  ptr = strchr(buf + 7, '=');
Packit 54873f
	  if (ptr)
Packit 54873f
	    {
Packit 54873f
	      while (*++ptr == ' ')
Packit 54873f
		;
Packit 54873f
	      version = makeevr(pool, ptr);
Packit 54873f
	    }
Packit 54873f
	}
Packit 54873f
    }
Packit 54873f
  if (name)
Packit 54873f
    {
Packit 54873f
      Solvable *s = pool_id2solvable(pool, repo_add_solvable(repo));
Packit 54873f
      s->name = name;
Packit 54873f
      s->evr = version ? version : ID_EMPTY;
Packit 54873f
      s->arch = arch ? arch : ARCH_NOARCH;
Packit 54873f
      if (s->name && s->arch != ARCH_SRC && s->arch != ARCH_NOSRC)
Packit 54873f
	s->provides = repo_addid_dep(repo, s->provides, pool_rel2id(pool, s->name, s->evr, REL_EQ, 1), 0);
Packit 54873f
    }
Packit 54873f
}
Packit 54873f
Packit 54873f
Packit 54873f
int
Packit 54873f
repo_add_releasefile_products(Repo *repo, const char *dirpath, int flags)
Packit 54873f
{
Packit 54873f
  DIR *dir;
Packit 54873f
  struct dirent *entry;
Packit 54873f
  FILE *fp;
Packit 54873f
  char *fullpath;
Packit 54873f
  struct parsedata pd;
Packit 54873f
Packit 54873f
  if (!dirpath)
Packit 54873f
    dirpath = "/etc";
Packit 54873f
  if (flags & REPO_USE_ROOTDIR)
Packit 54873f
    dirpath = pool_prepend_rootdir(repo->pool, dirpath);
Packit 54873f
  dir = opendir(dirpath);
Packit 54873f
  if (!dir)
Packit 54873f
    {
Packit 54873f
      if (flags & REPO_USE_ROOTDIR)
Packit 54873f
        solv_free((char *)dirpath);
Packit 54873f
      return 0;
Packit 54873f
    }
Packit 54873f
Packit 54873f
  memset(&pd, 0, sizeof(pd));
Packit 54873f
  pd.repo = repo;
Packit 54873f
  while ((entry = readdir(dir)))
Packit 54873f
    {
Packit 54873f
      int len = strlen(entry->d_name);
Packit 54873f
      if (len > 8 && !strcmp(entry->d_name + len - 8, "-release"))
Packit 54873f
        {
Packit 54873f
	  /* skip /etc/lsb-release, thats not a product per-se */
Packit 54873f
	  if (strcmp(entry->d_name, "lsb-release") == 0)
Packit 54873f
	    continue;
Packit 54873f
	  fullpath = join2(&pd.jd, dirpath, "/", entry->d_name);
Packit 54873f
	  if ((fp = fopen(fullpath, "r")) == 0)
Packit 54873f
	    {
Packit 54873f
	      pool_error(repo->pool, 0, "%s: %s", fullpath, strerror(errno));
Packit 54873f
	      continue;
Packit 54873f
	    }
Packit 54873f
	  add_releasefile_product(&pd, fp);
Packit 54873f
	  fclose(fp);
Packit 54873f
	}
Packit 54873f
    }
Packit 54873f
  closedir(dir);
Packit 54873f
  join_freemem(&pd.jd);
Packit 54873f
  if (flags & REPO_USE_ROOTDIR)
Packit 54873f
    solv_free((char *)dirpath);
Packit 54873f
Packit 54873f
  if (!(flags & REPO_NO_INTERNALIZE) && (flags & REPO_REUSE_REPODATA) != 0)
Packit 54873f
    repodata_internalize(repo_last_repodata(repo));
Packit 54873f
  return 0;
Packit 54873f
}
Packit 54873f