//BISMILLAHIRRAHMANIRRAHIM
/*
manus tar shopner soman boro
Author :: Shakil Ahmed
.............AUST_CSE27.........
prob ::
Type ::
verdict::
*/
#include <bits/stdc++.h>
#define pb push_back
#define mp make_pair
#define pi acos(-1.0)
#define ff first
#define ss second
#define re return
#define QI queue<int>
#define SI stack<int>
#define SZ(x) ((int) (x).size())
#define all(x) (x).begin(), (x).end()
#define sqr(x) ((x) * (x))
#define memo(a,b) memset((a),(b),sizeof(a))
#define G() getchar()
#define MAX3(a,b,c) max(a,max(b,c))
double const EPS=3e-8;
using namespace std;
#define FI freopen ("input.txt", "r", stdin)
#define FO freopen ("output.txt", "w", stdout)
typedef long long Long;
typedef long long int64;
typedef unsigned long long ull;
typedef vector<int> vi ;
typedef set<int> si;
typedef vector<Long>vl;
typedef pair<int,int>pii;
typedef pair<string,int>psi;
typedef pair<Long,Long>pll;
typedef pair<double,double>pdd;
typedef vector<pii> vpii;
// For loop
#define forab(i, a, b) for (__typeof (b) i = (a) ; i <= b ; ++i)
#define rep(i, n) forab (i, 0, (n) - 1)
#define For(i, n) forab (i, 1, n)
#define rofba(i, a, b) for (__typeof (b)i = (b) ; i >= a ; --i)
#define per(i, n) rofba (i, 0, (n) - 1)
#define rof(i, n) rofba (i, 1, n)
#define forstl(i, s) for (__typeof ((s).end ()) i = (s).begin (); i != (s).end (); ++i)
template< class T > T gcd(T a, T b) { return (b != 0 ? gcd<T>(b, a%b) : a); }
template< class T > T lcm(T a, T b) { return (a / gcd<T>(a, b) * b); }
//Fast Reader
template<class T>inline bool read(T &x){int c=getchar();int sgn=1;while(~c&&c<'0'||c>'9'){if(c=='-')sgn=-1;c=getchar();}for(x=0;~c&&'0'<=c&&c<='9';c=getchar())x=x*10+c-'0'; x*=sgn; return ~c;}
//int dx[]={1,0,-1,0};int dy[]={0,1,0,-1}; //4 Direction
//int dx[]={1,1,0,-1,-1,-1,0,1};int dy[]={0,1,1,1,0,-1,-1,-1};//8 direction
//int dx[]={2,1,-1,-2,-2,-1,1,2};int dy[]={1,2,2,1,-1,-2,-2,-1};//Knight Direction
//int dx[]={2,1,-1,-2,-1,1};int dy[]={0,1,1,0,-1,-1}; //Hexagonal Direction
/* ************************************** My code start here ****************************************** */
const int MX = 2000000 + 5 ;
int n , m , need[MX];
int C[3];
struct abc
{
int x , y , c ;
}inp[MX];
vector <int> g[MX] ;
int col[MX];
/*
int DP(int node,int isgurd,int par)
{
if(vis[node][isgurd] ) return dp[node][isgurd];
vis[node][isgurd] = 1 ;
col[node] = 1 ;
int i , sz = g[node].size();
int sum = 0 ;
for ( i = 0 ; i < sz ; i++ )
{
int v = g[node][i];
if( v != par )
{
if(!isgurd) sum += DP(v,1,node);
else sum += min(DP(v,0,node),DP(v,1,node));
}
}
return dp[node][isgurd] = sum + isgurd ;
} */
bool dfs(int node,int c)
{
queue < int > q ;
q.push(node);
col[node] = c ;
while(!q.empty())
{
int frnt = q.front();
C[col[frnt]]++;
q.pop();
int sz = g[frnt].size();
int i , v ;
for ( i = 0 ; i < sz ; i++ )
{
v = g[frnt][i];
if( col[v] == -1 )
{
col[v] = !col[frnt];
q.push(v);
}
else if ( col[v] == col[frnt] ) return 0 ;
}
}
return 1 ;
}
void ini()
{
int i ;
for ( int i = 0 ; i <= n+1 ; i++ ) { g[i].clear();
need[i] = -1 ; // no update yet
col[i] = -1 ;
}
}
int main()
{
// ios_base::sync_with_stdio(0); cin.tie(0);
while(scanf("%d %d",&n,&m)==2)
{
map < int , int > Mark;
int idx = 0 ;
ini();
int x , y , c ;
bool ok = 1;
set <int> ans ;
rep(i,m)
{
read(inp[i].x) ;
read(inp[i].y);
read(inp[i].c);
if( inp[i].c == 2 && (need[inp[i].x] == 0 || need[inp[i].y] == 0 ) )
{
//printf("edge :: %d need[inp[i].x] :: %d need[y]::%d\n2", i,need[inp[i].x],need[inp[i].y]);
ok = 0 ;
}
else if( inp[i].c == 2 )
{
need[inp[i].x] = 1 ;
need[inp[i].y] = 1 ;
ans.insert(inp[i].x);
ans.insert(inp[i].y);
}
if( inp[i].c == 0 && ( need[inp[i].x] == 1 || need[inp[i].y] == 1) )
{
ok = 0 ;
}
else if( inp[i].c == 0 )
{
need[inp[i].x] = 0 ;
need[inp[i].x] = 0 ;
}
}
if(!ok)
{
puts("impossible");
continue;
}
rep(i,m)
{
if( inp[i].c == 2 && (need[inp[i].x] == 0 || need[inp[i].y] == 0 ) )
{
//printf("edge :: %d need[inp[i].x] :: %d need[y]::%d\n2", i,need[inp[i].x],need[inp[i].y]);
ok = 0 ;
}
if( inp[i].c == 0 && ( need[inp[i].x] == 1 || need[inp[i].y] == 1) )
{
ok = 0 ;
}
if(!ok ) break;
if(inp[i].c == 2 || inp[i].c == 0 ) continue;
if( need[inp[i].x] == 0 && need[inp[i].y] == 0 )
{ok = 0 ; // printf("edge :: %d",i);
// printf("inp[i].x :: %d need :: %d inp[i].y :: %d need :: %d\n",inp[i].x,need[inp[i].x],inp[i].y,need[inp[i].y]);
}
else if( need[inp[i].x] == 1 && need[inp[i].y] == 1 )
{ ok = 0 ;
// printf("edge:: %d\n",i);
// printf("inp[i].x :: %d need :: %d inp[i].y :: %d need :: %d\n",inp[i].x,need[inp[i].x],inp[i].y,need[inp[i].y]);
// ans.insert(inp[i].x); // 2i khanei bosathe hobe tai
// ans.insert(inp[i].y);
}
else if( (need[inp[i].x] == 1 && need[inp[i].y] == -1) || (need[inp[i].x] ==-1 && need[inp[i].y] == 1 ) )
{
if(need[inp[i].x] == 1 )
{
ans.insert(inp[i].x);
}
else ans.insert(inp[i].y);
}
else if( (need[inp[i].x] == 0 && need[inp[i].y] == -1) || (need[inp[i].x] ==-1 && need[inp[i].y] == 0 ) )
{
if(need[inp[i].x] == -1 )
{
ans.insert(inp[i].x);
}
else ans.insert(inp[i].y);
}
else {
if(Mark[inp[i].x] == 0 ) Mark[inp[i].x] = ++idx;
if(Mark[inp[i].y] == 0 ) Mark[inp[i].y] = ++idx;
int _x = Mark[inp[i].x] ;
int _y = Mark[inp[i].y];
g[_x].pb(_y);
g[_y].pb(_x);
}
if( !ok ) break;
}
if(!ok)
{
puts("impossible");
continue;
}
rep(i,m)
{
if( inp[i].c == 2 && (need[inp[i].x] == 0 || need[inp[i].y] == 0 ) )
{
//printf("edge :: %d need[inp[i].x] :: %d need[y]::%d\n2", i,need[inp[i].x],need[inp[i].y]);
ok = 0 ;
}
if( inp[i].c == 0 && ( need[inp[i].x] == 1 || need[inp[i].y] == 1) )
{
ok = 0 ;
}
if(!ok ) break;
if(inp[i].c == 2 || inp[i].c == 0 ) continue;
if( need[inp[i].x] == 0 && need[inp[i].y] == 0 )
{ok = 0 ; // printf("edge :: %d",i);
// printf("inp[i].x :: %d need :: %d inp[i].y :: %d need :: %d\n",inp[i].x,need[inp[i].x],inp[i].y,need[inp[i].y]);
}
else if( need[inp[i].x] == 1 && need[inp[i].y] == 1 )
{ ok = 0 ;
// printf("edge:: %d\n",i);
// printf("inp[i].x :: %d need :: %d inp[i].y :: %d need :: %d\n",inp[i].x,need[inp[i].x],inp[i].y,need[inp[i].y]);
// ans.insert(inp[i].x); // 2i khanei bosathe hobe tai
// ans.insert(inp[i].y);
}
if(!ok) break;
}
if(!ok)
{
puts("impossible");
continue;
}
int add = 0 ;
int i ;
// printf("hre");
for ( i = 1 ; i <= idx && ok ; i++ )
{
if(col[i]==-1)
{
C[1] = C[0] = 0 ;
ok = dfs(i,1);
add += min(C[1],C[0]);
}
}
if(!ok)
{
puts("impossible");
continue;
}
cout << ans.size() + add << endl ;
}
return 0;
}
//BISMILLAHIRRAHMANIRRAHIM
/*
 manus tar shopner soman boro
 Author :: Shakil Ahmed
.............AUST_CSE27.........
 prob   ::
 Type   ::
 verdict::
 */

#include <bits/stdc++.h>
#define pb push_back
#define mp make_pair
#define pi acos(-1.0)
#define ff first
#define ss second
#define re return
#define QI queue<int>
#define SI stack<int>
#define SZ(x) ((int) (x).size())
#define all(x) (x).begin(), (x).end()
#define sqr(x) ((x) * (x))
#define memo(a,b) memset((a),(b),sizeof(a))
#define G() getchar()
#define MAX3(a,b,c) max(a,max(b,c))

double const EPS=3e-8;
using namespace std;

#define FI freopen ("input.txt", "r", stdin)
#define FO freopen ("output.txt", "w", stdout)

typedef long long Long;
typedef long long int64;
typedef unsigned long long ull;
typedef vector<int> vi ;
typedef set<int> si;
typedef vector<Long>vl;
typedef pair<int,int>pii;
typedef pair<string,int>psi;
typedef pair<Long,Long>pll;
typedef pair<double,double>pdd;
typedef vector<pii> vpii;

// For loop

#define forab(i, a, b)	for (__typeof (b) i = (a) ; i <= b ; ++i)
#define rep(i, n)		forab (i, 0, (n) - 1)
#define For(i, n)		forab (i, 1, n)
#define rofba(i, a, b)	for (__typeof (b)i = (b) ; i >= a ; --i)
#define per(i, n)		rofba (i, 0, (n) - 1)
#define rof(i, n)		rofba (i, 1, n)
#define forstl(i, s)	for (__typeof ((s).end ()) i = (s).begin (); i != (s).end (); ++i)

template< class T > T gcd(T a, T b) { return (b != 0 ? gcd<T>(b, a%b) : a); }
template< class T > T lcm(T a, T b) { return (a / gcd<T>(a, b) * b); }

//Fast Reader
template<class T>inline bool read(T &x){int c=getchar();int sgn=1;while(~c&&c<'0'||c>'9'){if(c=='-')sgn=-1;c=getchar();}for(x=0;~c&&'0'<=c&&c<='9';c=getchar())x=x*10+c-'0'; x*=sgn; return ~c;}

//int dx[]={1,0,-1,0};int dy[]={0,1,0,-1}; //4 Direction
//int dx[]={1,1,0,-1,-1,-1,0,1};int dy[]={0,1,1,1,0,-1,-1,-1};//8 direction
//int dx[]={2,1,-1,-2,-2,-1,1,2};int dy[]={1,2,2,1,-1,-2,-2,-1};//Knight Direction
//int dx[]={2,1,-1,-2,-1,1};int dy[]={0,1,1,0,-1,-1}; //Hexagonal Direction

/* **************************************  My code start here ****************************************** */
const int MX = 2000000 + 5 ;
int n , m , need[MX];
int C[3];
struct abc
{
    int x , y , c ;
}inp[MX];
vector <int> g[MX] ;
int  col[MX];
/*
int DP(int node,int isgurd,int par)
{
    if(vis[node][isgurd] ) return dp[node][isgurd];
    vis[node][isgurd] = 1 ;
    col[node] = 1 ;
    int i , sz = g[node].size();
    int sum = 0 ;
    for ( i = 0 ; i < sz ; i++ )
    {
        int v = g[node][i];
        if( v != par )
        {
            if(!isgurd) sum += DP(v,1,node);
            else sum += min(DP(v,0,node),DP(v,1,node));
        }
    }
    return dp[node][isgurd] = sum + isgurd ;
} */
bool dfs(int node,int c)
{
    queue < int > q ;
    q.push(node);
    col[node] = c ;
    while(!q.empty())
    {
        int frnt = q.front();
        C[col[frnt]]++;
        q.pop();
        int sz = g[frnt].size();
        int i , v ;
        for ( i = 0 ; i < sz ; i++ )
        {
            v = g[frnt][i];
            if( col[v] == -1 )
            {
                col[v] = !col[frnt];
                q.push(v);
            }
            else if ( col[v] == col[frnt] ) return 0 ;
        }
    }
    return 1 ;

}
void ini()
{
    int i ;
    for ( int i = 0 ; i <= n+1 ; i++ ) { g[i].clear();
    need[i] = -1 ; // no update yet

    col[i]  = -1 ;
    }
}
int main()
{
   // ios_base::sync_with_stdio(0); cin.tie(0);

    while(scanf("%d %d",&n,&m)==2)
    {
        map < int , int > Mark;
        int idx = 0 ;
      ini();
      int x , y , c ;
      bool ok = 1;
       set <int> ans ;
      rep(i,m)
      {
          read(inp[i].x) ;
          read(inp[i].y);
          read(inp[i].c);
          if( inp[i].c == 2 && (need[inp[i].x] == 0 || need[inp[i].y] == 0 ) )
          {
              //printf("edge :: %d need[inp[i].x] :: %d need[y]::%d\n2", i,need[inp[i].x],need[inp[i].y]);
              ok = 0 ;
          }
          else if( inp[i].c == 2 )
          {
              need[inp[i].x] = 1 ;
              need[inp[i].y] = 1 ;
              ans.insert(inp[i].x);
              ans.insert(inp[i].y);
          }
          if( inp[i].c == 0 && ( need[inp[i].x] == 1  || need[inp[i].y] == 1) )
          {

              ok = 0 ;
          }
          else if( inp[i].c == 0 )
          {
              need[inp[i].x] = 0 ;
              need[inp[i].x] = 0 ;
          }
      }
      if(!ok)
      {
          puts("impossible");
          continue;
      }

      rep(i,m)
      {
             if( inp[i].c == 2 && (need[inp[i].x] == 0 || need[inp[i].y] == 0 ) )
          {
              //printf("edge :: %d need[inp[i].x] :: %d need[y]::%d\n2", i,need[inp[i].x],need[inp[i].y]);
              ok = 0 ;
          }
            if( inp[i].c == 0 && ( need[inp[i].x] == 1  || need[inp[i].y] == 1) )
          {

              ok = 0 ;
          }
          if(!ok ) break;
          if(inp[i].c == 2 || inp[i].c == 0 ) continue;
          if( need[inp[i].x] == 0 && need[inp[i].y] == 0 )
          {ok = 0 ; // printf("edge :: %d",i);
         // printf("inp[i].x :: %d need :: %d inp[i].y :: %d need :: %d\n",inp[i].x,need[inp[i].x],inp[i].y,need[inp[i].y]);
          }
          else if( need[inp[i].x] == 1 && need[inp[i].y] == 1 )
          { ok = 0 ;
       //   printf("edge:: %d\n",i);
        //   printf("inp[i].x :: %d need :: %d inp[i].y :: %d need :: %d\n",inp[i].x,need[inp[i].x],inp[i].y,need[inp[i].y]);
           // ans.insert(inp[i].x); // 2i khanei bosathe hobe tai
           // ans.insert(inp[i].y);
          }
          else if( (need[inp[i].x] == 1 && need[inp[i].y] == -1) || (need[inp[i].x] ==-1 && need[inp[i].y] == 1 ) )
          {
              if(need[inp[i].x] == 1 )
              {
                  ans.insert(inp[i].x);
              }
              else ans.insert(inp[i].y);
          }
             else if( (need[inp[i].x] == 0 && need[inp[i].y] == -1) || (need[inp[i].x] ==-1 && need[inp[i].y] == 0 ) )
          {
              if(need[inp[i].x] == -1 )
              {
                  ans.insert(inp[i].x);
              }
              else ans.insert(inp[i].y);
          }
          else {
            if(Mark[inp[i].x] == 0 ) Mark[inp[i].x] = ++idx;
            if(Mark[inp[i].y] == 0 ) Mark[inp[i].y] = ++idx;
            int _x = Mark[inp[i].x] ;
            int _y = Mark[inp[i].y];

          g[_x].pb(_y);
          g[_y].pb(_x);
          }
          if( !ok ) break;


      }
        if(!ok)
      {
          puts("impossible");
          continue;
      }
      rep(i,m)
      {
              if( inp[i].c == 2 && (need[inp[i].x] == 0 || need[inp[i].y] == 0 ) )
          {
              //printf("edge :: %d need[inp[i].x] :: %d need[y]::%d\n2", i,need[inp[i].x],need[inp[i].y]);
              ok = 0 ;
          }
            if( inp[i].c == 0 && ( need[inp[i].x] == 1  || need[inp[i].y] == 1) )
          {

              ok = 0 ;
          }
          if(!ok ) break;
          if(inp[i].c == 2 || inp[i].c == 0 ) continue;
          if( need[inp[i].x] == 0 && need[inp[i].y] == 0 )
          {ok = 0 ; // printf("edge :: %d",i);
         // printf("inp[i].x :: %d need :: %d inp[i].y :: %d need :: %d\n",inp[i].x,need[inp[i].x],inp[i].y,need[inp[i].y]);
          }
          else if( need[inp[i].x] == 1 && need[inp[i].y] == 1 )
          { ok = 0 ;
       //   printf("edge:: %d\n",i);
        //   printf("inp[i].x :: %d need :: %d inp[i].y :: %d need :: %d\n",inp[i].x,need[inp[i].x],inp[i].y,need[inp[i].y]);
           // ans.insert(inp[i].x); // 2i khanei bosathe hobe tai
           // ans.insert(inp[i].y);
          }
          if(!ok) break;
      }
        if(!ok)
      {
          puts("impossible");
          continue;
      }
      int add = 0 ;
      int i ;
    //  printf("hre");
      for ( i =  1 ; i <= idx && ok ; i++ )
      {
          if(col[i]==-1)
          {
              C[1] = C[0] = 0 ;
              ok = dfs(i,1);
              add += min(C[1],C[0]);
          }

      }
        if(!ok)
      {
          puts("impossible");
          continue;
      }
      cout << ans.size() + add << endl ;




    }



    return 0;
}